83 lines
2.9 KiB
JavaScript
83 lines
2.9 KiB
JavaScript
import 'dotenv/config';
|
|
import express from 'express';
|
|
import { pool } from './lib/db/pool.js';
|
|
import { log } from './lib/log.js';
|
|
import { mountApi } from './lib/api/index.js';
|
|
import * as queue from './lib/jobs/queue.js';
|
|
import { registerWorkers } from './lib/jobs/index.js';
|
|
import { router as ingestRouter } from './lib/api/routes/ingest.js';
|
|
import { router as iconsRouter } from './lib/api/routes/icons.js';
|
|
import { startCron } from './lib/cron/index.js';
|
|
import { seedFromConfig } from './lib/health/registry.js';
|
|
import { mcpAuth } from './lib/api/middleware/mcp_auth.js';
|
|
import { handleMcp } from './lib/mcp/http.js';
|
|
|
|
const VERSION = '2.0.0-alpha.16';
|
|
|
|
export function createApp() {
|
|
const app = express();
|
|
app.use(express.json({
|
|
limit: '10mb',
|
|
verify: (req, _res, buf) => { req.rawBody = buf; }
|
|
}));
|
|
// no-cache (not no-store): the browser may cache but must revalidate, so a
|
|
// deploy's new CSS/JS takes effect immediately instead of serving stale assets.
|
|
app.use(express.static('public', {
|
|
setHeaders: (res) => res.setHeader('Cache-Control', 'no-cache')
|
|
}));
|
|
|
|
// /api/ingest/* bypasses agentOrOwner — webhooks authenticate via HMAC
|
|
// and need access to req.rawBody captured above.
|
|
app.use('/api/ingest', ingestRouter);
|
|
|
|
// /api/icons/* bypasses agentOrOwner — <img> tags can't send bearer headers;
|
|
// slugs are sanitized to [a-z0-9-] to prevent path traversal.
|
|
app.use('/api/icons', iconsRouter);
|
|
|
|
app.get('/health', async (_req, res) => {
|
|
let db_ok = false;
|
|
try {
|
|
await pool.query('SELECT 1');
|
|
db_ok = true;
|
|
} catch (e) {
|
|
log.error({ err: e }, 'healthcheck db ping failed');
|
|
}
|
|
res.json({ ok: true, db_ok, version: VERSION });
|
|
});
|
|
|
|
mountApi(app);
|
|
|
|
// MCP Streamable HTTP for external agents (read + suggest-only, space-scoped).
|
|
app.all('/mcp', mcpAuth, handleMcp);
|
|
|
|
app.use((_req, res) => res.status(404).json({ error: { code: 'not_found' } }));
|
|
|
|
app.use((err, _req, res, _next) => {
|
|
log.error({ err }, 'unhandled');
|
|
res.status(500).json({ error: { code: 'internal', message: 'internal server error' } });
|
|
});
|
|
|
|
return app;
|
|
}
|
|
|
|
if (import.meta.url === `file://${process.argv[1]}`) {
|
|
const port = process.env.PORT || 3000;
|
|
const app = createApp();
|
|
queue.start()
|
|
.then(registerWorkers)
|
|
.then(() => log.info('job queue ready'))
|
|
.catch(err => log.error({ err }, 'queue boot failed'));
|
|
startCron();
|
|
// One-time bootstrap of the service registry from config/services.json if empty.
|
|
seedFromConfig().then(n => { if (n) log.info({ seeded: n }, 'monitored_services seeded from config'); })
|
|
.catch(err => log.error({ err }, 'service registry seed failed'));
|
|
app.listen(port, () => log.info({ port }, 'void-server listening'));
|
|
for (const sig of ['SIGTERM', 'SIGINT']) {
|
|
process.on(sig, async () => {
|
|
log.info({ sig }, 'shutting down');
|
|
try { await queue.stop(); } catch { /* */ }
|
|
process.exit(0);
|
|
});
|
|
}
|
|
}
|