trustgraph/ts/packages/flow/src/gateway/server.ts

238 lines
6.9 KiB
TypeScript
Raw Normal View History

2026-04-05 21:09:33 -05:00
/**
* API Gateway HTTP + WebSocket server.
*
* Replaces the Python aiohttp gateway with Fastify.
2026-04-05 22:44:45 -05:00
* Uses the Mux class for WebSocket multiplexing (queue-based request
* buffering, concurrency control, proper task lifecycle).
2026-04-05 21:09:33 -05:00
*
* Python reference: trustgraph-flow/trustgraph/gateway/service.py
*/
import Fastify from "fastify";
import websocketPlugin from "@fastify/websocket";
2026-04-05 22:44:45 -05:00
import { registry } from "@trustgraph/base";
2026-04-05 21:09:33 -05:00
import { DispatcherManager } from "./dispatch/manager.js";
2026-04-05 22:44:45 -05:00
import { Mux, type MuxRequest, type MuxHandler } from "./dispatch/mux.js";
2026-04-05 21:09:33 -05:00
export interface GatewayConfig {
port: number;
metricsPort: number;
secret?: string;
natsUrl?: string;
}
export async function createGateway(config: GatewayConfig) {
const app = Fastify({ logger: true });
await app.register(websocketPlugin);
const dispatcher = new DispatcherManager(config);
await dispatcher.start();
// Authentication middleware
app.addHook("onRequest", async (request, reply) => {
if (request.url === "/api/v1/metrics") return;
2026-04-05 22:44:45 -05:00
if (request.url.startsWith("/api/v1/socket")) return; // Socket auth via query param
2026-04-05 21:09:33 -05:00
if (config.secret) {
const auth = request.headers.authorization;
if (!auth || auth !== `Bearer ${config.secret}`) {
reply.code(401).send({ error: "Unauthorized" });
}
}
});
2026-04-05 22:44:45 -05:00
// REST endpoint: POST /api/v1/:kind (global services)
2026-04-05 21:09:33 -05:00
app.post<{ Params: { kind: string } }>("/api/v1/:kind", async (request, reply) => {
const { kind } = request.params;
const body = request.body as Record<string, unknown>;
try {
const result = await dispatcher.dispatchGlobalService(kind, body);
return result;
} catch (err) {
reply.code(500).send({ error: { type: "internal", message: String(err) } });
}
});
2026-04-05 22:44:45 -05:00
// REST endpoint: POST /api/v1/flow/:flow/service/:kind (flow-scoped services)
2026-04-05 21:09:33 -05:00
app.post<{ Params: { flow: string; kind: string } }>(
"/api/v1/flow/:flow/service/:kind",
async (request, reply) => {
const { flow, kind } = request.params;
const body = request.body as Record<string, unknown>;
try {
const result = await dispatcher.dispatchFlowService(flow, kind, body);
return result;
} catch (err) {
reply.code(500).send({ error: { type: "internal", message: String(err) } });
}
},
);
// REST endpoint: POST /api/v1/flow/:flow/load (trigger document processing)
app.post<{ Params: { flow: string } }>(
"/api/v1/flow/:flow/load",
async (request, reply) => {
const { flow } = request.params;
const body = request.body as {
documentId?: string;
user?: string;
collection?: string;
};
if (!body.documentId) {
return reply.code(400).send({
error: { type: "bad-request", message: "documentId is required" },
});
}
try {
const user = body.user ?? "default";
const collection = body.collection ?? "default";
const documentId = body.documentId;
// Publish Document message to the decode-input topic
const topic = "tg.flow.document";
const metadata = {
id: `load-${Date.now()}-${Math.random().toString(36).slice(2, 8)}`,
root: documentId,
user,
collection,
};
await dispatcher.publishToTopic(topic, { metadata, documentId });
return { status: "processing", documentId, flow };
} catch (err) {
reply.code(500).send({
error: { type: "internal", message: String(err) },
});
}
},
);
2026-04-05 21:09:33 -05:00
// WebSocket endpoint: /api/v1/socket
2026-04-05 22:44:45 -05:00
// Uses Mux for queue-based request buffering and concurrency control.
2026-04-05 21:09:33 -05:00
app.get("/api/v1/socket", { websocket: true }, (socket, request) => {
// Auth via query param
const url = new URL(request.url, `http://${request.headers.host}`);
const token = url.searchParams.get("token");
if (config.secret && token !== config.secret) {
socket.close(4001, "Unauthorized");
return;
}
2026-04-05 22:44:45 -05:00
// Build the MuxHandler that dispatches to the DispatcherManager
const handler: MuxHandler = async (muxReq, respond) => {
if (muxReq.flow) {
await dispatcher.dispatchFlowServiceStreaming(
muxReq.flow,
muxReq.service,
muxReq.request,
respond,
);
} else {
await dispatcher.dispatchGlobalServiceStreaming(
muxReq.service,
muxReq.request,
respond,
);
}
};
const mux = new Mux(handler);
2026-04-05 21:09:33 -05:00
2026-04-05 22:44:45 -05:00
// Start the Mux run loop — sends responses back over the socket
const runPromise = mux.run((data) => {
// Only send if the socket is still open (readyState 1 = OPEN)
if (socket.readyState === 1) {
socket.send(data);
}
});
// Incoming messages get queued into the Mux
socket.on("message", (data) => {
try {
const msg = JSON.parse(data.toString()) as {
id?: string;
service?: string;
flow?: string;
request?: Record<string, unknown>;
2026-04-05 21:09:33 -05:00
};
2026-04-05 22:44:45 -05:00
if (!msg.id || !msg.service || !msg.request) {
socket.send(
JSON.stringify({
id: msg.id ?? null,
error: { type: "bad-request", message: "Missing id, service, or request" },
complete: true,
}),
);
return;
2026-04-05 21:09:33 -05:00
}
2026-04-05 22:44:45 -05:00
const muxReq: MuxRequest = {
id: msg.id,
service: msg.service,
flow: msg.flow,
request: msg.request,
};
mux.receive(muxReq);
2026-04-05 21:09:33 -05:00
} catch (err) {
socket.send(
JSON.stringify({
2026-04-05 22:44:45 -05:00
error: { type: "parse-error", message: String(err) },
2026-04-05 21:09:33 -05:00
complete: true,
}),
);
}
});
socket.on("close", () => {
2026-04-05 22:44:45 -05:00
mux.stop();
});
socket.on("error", () => {
mux.stop();
});
// Ensure runPromise errors don't go unhandled
runPromise.catch((err) => {
console.error("[Gateway] Mux run loop error:", err);
mux.stop();
if (socket.readyState === 1) {
socket.close(1011, "Internal server error");
}
2026-04-05 21:09:33 -05:00
});
});
2026-04-05 22:44:45 -05:00
// Metrics endpoint — returns Prometheus metrics from prom-client
app.get("/api/v1/metrics", async (_, reply) => {
reply.header("content-type", registry.contentType);
2026-04-05 21:09:33 -05:00
return registry.metrics();
});
return {
start: () => app.listen({ port: config.port, host: "0.0.0.0" }),
stop: async () => {
await app.close();
await dispatcher.stop();
},
};
}
export async function run(): Promise<void> {
const config: GatewayConfig = {
port: parseInt(process.env.GATEWAY_PORT ?? "8088", 10),
metricsPort: parseInt(process.env.METRICS_PORT ?? "8000", 10),
secret: process.env.GATEWAY_SECRET,
natsUrl: process.env.NATS_URL,
};
const gateway = await createGateway(config);
await gateway.start();
console.log(`[Gateway] Listening on port ${config.port}`);
}