improve event processing by adding asynchronous handling for better performance

This commit is contained in:
ImBenji 2026-04-21 13:03:00 +01:00
parent 07cabf43e3
commit 81bcf40a8d
5 changed files with 79 additions and 36 deletions

View file

@ -110,15 +110,25 @@ Returns one article by numeric ID. Same usability filter as the list endpoint
### `GET /events`
Returns a single event and its articles.
Without `id` — returns a paginated list of events. With `id` — returns a single event and its articles.
#### Query params
| Param | Description |
|---|---|
| `id` | Event ID (required) |
| `id` | Event ID. If present, returns that event with its articles instead of the list |
| `limit` | Rows to return (list mode only). Default `20`, max `100` |
| `offset` | Pagination offset (list mode only). Default `0` |
#### Response shape
#### List response shape
```json
[
{ "id": 1, "title": "...", "created_at": "2025-01-01T12:35:10.000Z" }
]
```
#### Single event response shape
```json
{

View file

@ -19,7 +19,8 @@
"tickers": []
},
"openRouter": {
"apiKey": "[OFF]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",
"enabled": false,
"apiKey": "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",
"embeddingModel": "qwen/qwen3-embedding-8b"
},
"gdelt": {

View file

@ -72,6 +72,9 @@ const BODY_PREFIX_BLOCKLIST = [
// yahoo finance serves its global nav when the article body is js-rendered
// and the plain fetch only gets the static shell
"today's news us politics world weather",
// cnbc paywall shell — no article body, just site nav
"subscribe to cnbc pro subscribe to investing club",
];

View file

@ -91,6 +91,12 @@ const upsertQueryEmbedding = db.prepare(`
const VEC0_DIM = 8192;
function isOpenRouterEnabled() {
if (!config.openRouter) return false;
if (config.openRouter.enabled === false) return false;
return Boolean(config.openRouter.apiKey && String(config.openRouter.apiKey).trim());
}
function serializeEmbedding(values) {
return Buffer.from(new Float32Array(values).buffer);
}
@ -279,11 +285,7 @@ async function requestEmbedding(input) {
}
async function generateAndStoreEmbedding(id) {
const apiKey = config.openRouter && config.openRouter.apiKey
? String(config.openRouter.apiKey).trim()
: '';
if (!apiKey) {
if (!isOpenRouterEnabled()) {
return { stored: false, shouldPauseBatch: false };
}
@ -368,11 +370,7 @@ async function backfillMissingEmbeddings(limit = 256, batchSize = 16) {
return { processed: 0, paused: false };
}
const apiKey = config.openRouter && config.openRouter.apiKey
? String(config.openRouter.apiKey).trim()
: '';
if (!apiKey) {
if (!isOpenRouterEnabled()) {
return { processed: 0, paused: false };
}

View file

@ -1,14 +1,20 @@
const db = require('../db');
function parseLimit(value) {
const n = Number.parseInt(value, 10);
return Number.isFinite(n) && n > 0 ? Math.min(n, 100) : 20;
}
function parseOffset(value) {
const n = Number.parseInt(value, 10);
return Number.isFinite(n) && n >= 0 ? n : 0;
}
async function eventRoutes(fastify) {
fastify.get('/events', async (request, reply) => {
const query = request.query || {};
if (!query.id) {
reply.code(400);
return { error: 'id is required' };
}
if (query.id) {
const id = Number.parseInt(query.id, 10);
if (!Number.isFinite(id)) {
reply.code(400);
@ -31,6 +37,31 @@ async function eventRoutes(fastify) {
`).all(id);
return { ...event, articles };
}
const limit = parseLimit(query.limit);
const offset = parseOffset(query.offset);
const SORT_COLUMNS = {
created_at: 'e.created_at',
id: 'e.id',
article_count: 'article_count',
};
const sortBy = SORT_COLUMNS[query.sort_by] || SORT_COLUMNS.created_at;
const order = String(query.order || '').toLowerCase() === 'asc' ? 'ASC' : 'DESC';
return db.prepare(`
SELECT e.id, e.title, e.created_at,
COUNT(a.id) AS article_count
FROM events e
LEFT JOIN articles a ON a.event_id = e.id
AND a.content IS NOT NULL AND a.content != ''
AND a.is_index_page = 0
GROUP BY e.id
ORDER BY ${sortBy} ${order}, e.id ${order}
LIMIT ? OFFSET ?
`).all(limit, offset);
});
}