- Ingestion API: batch cost events, Welford baseline update with optimistic locking (version column), anomaly detection inline - Anomaly API: list (filtered), acknowledge, snooze (1-168h), mark expected, dashboard summary with hourly trend - Governance API: mode status, promotion eligibility check with FP rate calculation - Baseline API: list with computed stddev, reset per resource - Data layer: withTenant() RLS wrapper, Zod config with ANOMALY_THRESHOLD - Fastify server entry point
25 lines
715 B
TypeScript
25 lines
715 B
TypeScript
import pg from 'pg';
|
|
import pino from 'pino';
|
|
import { config } from '../config/index.js';
|
|
|
|
const logger = pino({ name: 'data' });
|
|
|
|
export const pool = new pg.Pool({ connectionString: config.DATABASE_URL });
|
|
|
|
export async function withTenant<T>(tenantId: string, fn: (client: pg.PoolClient) => Promise<T>): Promise<T> {
|
|
const client = await pool.connect();
|
|
try {
|
|
await client.query('BEGIN');
|
|
await client.query(`SET LOCAL app.tenant_id = '${tenantId}'`);
|
|
const result = await fn(client);
|
|
await client.query('COMMIT');
|
|
return result;
|
|
} catch (err) {
|
|
await client.query('ROLLBACK');
|
|
throw err;
|
|
} finally {
|
|
await client.query('RESET app.tenant_id');
|
|
client.release();
|
|
}
|
|
}
|