bonecode 1.1.0 → 1.2.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +145 -9
- package/bin/bonecode +47 -42
- package/compat/opencode_adapter.ts +188 -17
- package/dist/bone/output/agent/src/algorithms.d.ts +1 -0
- package/dist/bone/output/agent/src/algorithms.js +3 -0
- package/dist/bone/output/agent/src/algorithms.js.map +1 -0
- package/dist/bone/output/agent/src/audit.d.ts +3 -0
- package/dist/bone/output/agent/src/audit.js +40 -0
- package/dist/bone/output/agent/src/audit.js.map +1 -0
- package/dist/bone/output/agent/src/auth.d.ts +8 -0
- package/dist/bone/output/agent/src/auth.js +56 -0
- package/dist/bone/output/agent/src/auth.js.map +1 -0
- package/dist/bone/output/agent/src/db.d.ts +6 -0
- package/dist/bone/output/agent/src/db.js +63 -0
- package/dist/bone/output/agent/src/db.js.map +1 -0
- package/dist/bone/output/agent/src/events.d.ts +25 -0
- package/dist/bone/output/agent/src/events.js +184 -0
- package/dist/bone/output/agent/src/events.js.map +1 -0
- package/dist/bone/output/agent/src/logger.d.ts +28 -0
- package/dist/bone/output/agent/src/logger.js +45 -0
- package/dist/bone/output/agent/src/logger.js.map +1 -0
- package/dist/bone/output/agent/src/metrics.d.ts +5 -0
- package/dist/bone/output/agent/src/metrics.js +60 -0
- package/dist/bone/output/agent/src/metrics.js.map +1 -0
- package/dist/bone/output/agent/src/routes/agent_instance.d.ts +1 -0
- package/dist/bone/output/agent/src/routes/agent_instance.js +253 -0
- package/dist/bone/output/agent/src/routes/agent_instance.js.map +1 -0
- package/dist/bone/output/agent/src/routes/build_step.d.ts +1 -0
- package/dist/bone/output/agent/src/routes/build_step.js +133 -0
- package/dist/bone/output/agent/src/routes/build_step.js.map +1 -0
- package/dist/bone/output/agent/src/routes/plan.d.ts +1 -0
- package/dist/bone/output/agent/src/routes/plan.js +119 -0
- package/dist/bone/output/agent/src/routes/plan.js.map +1 -0
- package/dist/bone/output/agent/src/routes/task.d.ts +1 -0
- package/dist/bone/output/agent/src/routes/task.js +133 -0
- package/dist/bone/output/agent/src/routes/task.js.map +1 -0
- package/dist/bone/output/agent/src/routes/tool_call.d.ts +1 -0
- package/dist/bone/output/agent/src/routes/tool_call.js +190 -0
- package/dist/bone/output/agent/src/routes/tool_call.js.map +1 -0
- package/dist/bone/output/agent/src/state_machines/agent_instance.d.ts +9 -0
- package/dist/bone/output/agent/src/state_machines/agent_instance.js +22 -0
- package/dist/bone/output/agent/src/state_machines/agent_instance.js.map +1 -0
- package/dist/bone/output/agent/src/state_machines/build_step.d.ts +9 -0
- package/dist/bone/output/agent/src/state_machines/build_step.js +20 -0
- package/dist/bone/output/agent/src/state_machines/build_step.js.map +1 -0
- package/dist/bone/output/agent/src/state_machines/plan.d.ts +9 -0
- package/dist/bone/output/agent/src/state_machines/plan.js +20 -0
- package/dist/bone/output/agent/src/state_machines/plan.js.map +1 -0
- package/dist/bone/output/agent/src/state_machines/task.d.ts +9 -0
- package/dist/bone/output/agent/src/state_machines/task.js +20 -0
- package/dist/bone/output/agent/src/state_machines/task.js.map +1 -0
- package/dist/bone/output/agent/src/state_machines/tool_call.d.ts +9 -0
- package/dist/bone/output/agent/src/state_machines/tool_call.js +20 -0
- package/dist/bone/output/agent/src/state_machines/tool_call.js.map +1 -0
- package/dist/bone/output/rag/src/algorithms.d.ts +1 -0
- package/dist/bone/output/rag/src/algorithms.js +3 -0
- package/dist/bone/output/rag/src/algorithms.js.map +1 -0
- package/dist/bone/output/rag/src/auth.d.ts +8 -0
- package/dist/bone/output/rag/src/auth.js +56 -0
- package/dist/bone/output/rag/src/auth.js.map +1 -0
- package/dist/bone/output/rag/src/db.d.ts +6 -0
- package/dist/bone/output/rag/src/db.js +63 -0
- package/dist/bone/output/rag/src/db.js.map +1 -0
- package/dist/bone/output/rag/src/events.d.ts +25 -0
- package/dist/bone/output/rag/src/events.js +184 -0
- package/dist/bone/output/rag/src/events.js.map +1 -0
- package/dist/bone/output/rag/src/extensions.d.ts +83 -0
- package/dist/bone/output/rag/src/extensions.js +329 -0
- package/dist/bone/output/rag/src/extensions.js.map +1 -0
- package/dist/bone/output/rag/src/flows.d.ts +24 -0
- package/dist/bone/output/rag/src/flows.js +236 -0
- package/dist/bone/output/rag/src/flows.js.map +1 -0
- package/dist/bone/output/rag/src/logger.d.ts +28 -0
- package/dist/bone/output/rag/src/logger.js +45 -0
- package/dist/bone/output/rag/src/logger.js.map +1 -0
- package/dist/bone/output/rag/src/metrics.d.ts +5 -0
- package/dist/bone/output/rag/src/metrics.js +60 -0
- package/dist/bone/output/rag/src/metrics.js.map +1 -0
- package/dist/bone/output/rag/src/routes/code_chunk.d.ts +1 -0
- package/dist/bone/output/rag/src/routes/code_chunk.js +100 -0
- package/dist/bone/output/rag/src/routes/code_chunk.js.map +1 -0
- package/dist/bone/output/rag/src/routes/code_file.d.ts +1 -0
- package/dist/bone/output/rag/src/routes/code_file.js +127 -0
- package/dist/bone/output/rag/src/routes/code_file.js.map +1 -0
- package/dist/bone/output/rag/src/routes/indexing_job.d.ts +1 -0
- package/dist/bone/output/rag/src/routes/indexing_job.js +113 -0
- package/dist/bone/output/rag/src/routes/indexing_job.js.map +1 -0
- package/dist/bone/output/rag/src/routes/knowledge_base.d.ts +1 -0
- package/dist/bone/output/rag/src/routes/knowledge_base.js +242 -0
- package/dist/bone/output/rag/src/routes/knowledge_base.js.map +1 -0
- package/dist/bone/output/rag/src/routes/memory_entry.d.ts +1 -0
- package/dist/bone/output/rag/src/routes/memory_entry.js +113 -0
- package/dist/bone/output/rag/src/routes/memory_entry.js.map +1 -0
- package/dist/bone/output/rag/src/state_machines/code_file.d.ts +9 -0
- package/dist/bone/output/rag/src/state_machines/code_file.js +21 -0
- package/dist/bone/output/rag/src/state_machines/code_file.js.map +1 -0
- package/dist/bone/output/rag/src/state_machines/indexing_job.d.ts +9 -0
- package/dist/bone/output/rag/src/state_machines/indexing_job.js +20 -0
- package/dist/bone/output/rag/src/state_machines/indexing_job.js.map +1 -0
- package/dist/bone/output/rag/src/state_machines/knowledge_base.d.ts +9 -0
- package/dist/bone/output/rag/src/state_machines/knowledge_base.js +21 -0
- package/dist/bone/output/rag/src/state_machines/knowledge_base.js.map +1 -0
- package/dist/bone/output/rag/src/state_machines/memory_entry.d.ts +9 -0
- package/dist/bone/output/rag/src/state_machines/memory_entry.js +18 -0
- package/dist/bone/output/rag/src/state_machines/memory_entry.js.map +1 -0
- package/dist/bone/output/session/src/algorithms.d.ts +1 -0
- package/dist/bone/output/session/src/algorithms.js +3 -0
- package/dist/bone/output/session/src/algorithms.js.map +1 -0
- package/dist/bone/output/session/src/audit.d.ts +3 -0
- package/dist/bone/output/session/src/audit.js +40 -0
- package/dist/bone/output/session/src/audit.js.map +1 -0
- package/dist/bone/output/session/src/auth.d.ts +8 -0
- package/dist/bone/output/session/src/auth.js +56 -0
- package/dist/bone/output/session/src/auth.js.map +1 -0
- package/dist/bone/output/session/src/db.d.ts +6 -0
- package/dist/bone/output/session/src/db.js +63 -0
- package/dist/bone/output/session/src/db.js.map +1 -0
- package/dist/bone/output/session/src/events.d.ts +26 -0
- package/dist/bone/output/session/src/events.js +212 -0
- package/dist/bone/output/session/src/events.js.map +1 -0
- package/dist/bone/output/session/src/extensions.d.ts +41 -0
- package/dist/bone/output/session/src/extensions.js +217 -0
- package/dist/bone/output/session/src/extensions.js.map +1 -0
- package/dist/bone/output/session/src/logger.d.ts +28 -0
- package/dist/bone/output/session/src/logger.js +44 -0
- package/dist/bone/output/session/src/logger.js.map +1 -0
- package/dist/bone/output/session/src/metrics.d.ts +5 -0
- package/dist/bone/output/session/src/metrics.js +60 -0
- package/dist/bone/output/session/src/metrics.js.map +1 -0
- package/dist/bone/output/session/src/routes/message.d.ts +1 -0
- package/dist/bone/output/session/src/routes/message.js +120 -0
- package/dist/bone/output/session/src/routes/message.js.map +1 -0
- package/dist/bone/output/session/src/routes/part.d.ts +1 -0
- package/dist/bone/output/session/src/routes/part.js +106 -0
- package/dist/bone/output/session/src/routes/part.js.map +1 -0
- package/dist/bone/output/session/src/routes/permission.d.ts +1 -0
- package/dist/bone/output/session/src/routes/permission.js +106 -0
- package/dist/bone/output/session/src/routes/permission.js.map +1 -0
- package/dist/bone/output/session/src/routes/project.d.ts +1 -0
- package/dist/bone/output/session/src/routes/project.js +106 -0
- package/dist/bone/output/session/src/routes/project.js.map +1 -0
- package/dist/bone/output/session/src/routes/session.d.ts +1 -0
- package/dist/bone/output/session/src/routes/session.js +308 -0
- package/dist/bone/output/session/src/routes/session.js.map +1 -0
- package/dist/bone/output/session/src/state_machines/session.d.ts +9 -0
- package/dist/bone/output/session/src/state_machines/session.js +21 -0
- package/dist/bone/output/session/src/state_machines/session.js.map +1 -0
- package/dist/bone/output/session/src/websocket.d.ts +15 -0
- package/dist/bone/output/session/src/websocket.js +215 -0
- package/dist/bone/output/session/src/websocket.js.map +1 -0
- package/dist/bone/output/workspace/src/algorithms.d.ts +1 -0
- package/dist/bone/output/workspace/src/algorithms.js +3 -0
- package/dist/bone/output/workspace/src/algorithms.js.map +1 -0
- package/dist/bone/output/workspace/src/auth.d.ts +8 -0
- package/dist/bone/output/workspace/src/auth.js +56 -0
- package/dist/bone/output/workspace/src/auth.js.map +1 -0
- package/dist/bone/output/workspace/src/db.d.ts +6 -0
- package/dist/bone/output/workspace/src/db.js +63 -0
- package/dist/bone/output/workspace/src/db.js.map +1 -0
- package/dist/bone/output/workspace/src/events.d.ts +25 -0
- package/dist/bone/output/workspace/src/events.js +184 -0
- package/dist/bone/output/workspace/src/events.js.map +1 -0
- package/dist/bone/output/workspace/src/logger.d.ts +28 -0
- package/dist/bone/output/workspace/src/logger.js +45 -0
- package/dist/bone/output/workspace/src/logger.js.map +1 -0
- package/dist/bone/output/workspace/src/metrics.d.ts +5 -0
- package/dist/bone/output/workspace/src/metrics.js +60 -0
- package/dist/bone/output/workspace/src/metrics.js.map +1 -0
- package/dist/bone/output/workspace/src/routes/codebase.d.ts +1 -0
- package/dist/bone/output/workspace/src/routes/codebase.js +113 -0
- package/dist/bone/output/workspace/src/routes/codebase.js.map +1 -0
- package/dist/bone/output/workspace/src/routes/snapshot.d.ts +1 -0
- package/dist/bone/output/workspace/src/routes/snapshot.js +151 -0
- package/dist/bone/output/workspace/src/routes/snapshot.js.map +1 -0
- package/dist/bone/output/workspace/src/routes/workspace.d.ts +1 -0
- package/dist/bone/output/workspace/src/routes/workspace.js +209 -0
- package/dist/bone/output/workspace/src/routes/workspace.js.map +1 -0
- package/dist/bone/output/workspace/src/state_machines/codebase.d.ts +9 -0
- package/dist/bone/output/workspace/src/state_machines/codebase.js +19 -0
- package/dist/bone/output/workspace/src/state_machines/codebase.js.map +1 -0
- package/dist/bone/output/workspace/src/state_machines/snapshot.d.ts +9 -0
- package/dist/bone/output/workspace/src/state_machines/snapshot.js +18 -0
- package/dist/bone/output/workspace/src/state_machines/snapshot.js.map +1 -0
- package/dist/bone/output/workspace/src/state_machines/workspace.d.ts +9 -0
- package/dist/bone/output/workspace/src/state_machines/workspace.js +19 -0
- package/dist/bone/output/workspace/src/state_machines/workspace.js.map +1 -0
- package/dist/compat/opencode_adapter.d.ts +25 -0
- package/dist/compat/opencode_adapter.js +599 -0
- package/dist/compat/opencode_adapter.js.map +1 -0
- package/dist/extensions/chunker.d.ts +24 -0
- package/dist/extensions/chunker.js +360 -0
- package/dist/extensions/chunker.js.map +1 -0
- package/dist/extensions/embedding_provider.d.ts +18 -0
- package/dist/extensions/embedding_provider.js +150 -0
- package/dist/extensions/embedding_provider.js.map +1 -0
- package/dist/extensions/llm_provider.d.ts +33 -0
- package/dist/extensions/llm_provider.js +338 -0
- package/dist/extensions/llm_provider.js.map +1 -0
- package/dist/extensions/mcp_bridge.d.ts +44 -0
- package/dist/extensions/mcp_bridge.js +151 -0
- package/dist/extensions/mcp_bridge.js.map +1 -0
- package/dist/extensions/rag_search.d.ts +38 -0
- package/dist/extensions/rag_search.js +242 -0
- package/dist/extensions/rag_search.js.map +1 -0
- package/dist/extensions/snapshot.d.ts +14 -0
- package/dist/extensions/snapshot.js +158 -0
- package/dist/extensions/snapshot.js.map +1 -0
- package/dist/extensions/tool_executor.d.ts +28 -0
- package/dist/extensions/tool_executor.js +268 -0
- package/dist/extensions/tool_executor.js.map +1 -0
- package/dist/src/cli.d.ts +15 -0
- package/dist/src/cli.js +687 -0
- package/dist/src/cli.js.map +1 -0
- package/dist/src/config.d.ts +44 -0
- package/dist/src/config.js +165 -0
- package/dist/src/config.js.map +1 -0
- package/dist/src/context_builder.d.ts +51 -0
- package/dist/src/context_builder.js +558 -0
- package/dist/src/context_builder.js.map +1 -0
- package/dist/src/db_adapter.d.ts +24 -0
- package/dist/src/db_adapter.js +341 -0
- package/dist/src/db_adapter.js.map +1 -0
- package/dist/src/engine/session/compaction_logic.d.ts +11 -0
- package/dist/src/engine/session/compaction_logic.js +113 -0
- package/dist/src/engine/session/compaction_logic.js.map +1 -0
- package/dist/src/engine/session/instruction_loader.d.ts +5 -0
- package/dist/src/engine/session/instruction_loader.js +78 -0
- package/dist/src/engine/session/instruction_loader.js.map +1 -0
- package/dist/src/engine/session/overflow_check.d.ts +14 -0
- package/dist/src/engine/session/overflow_check.js +45 -0
- package/dist/src/engine/session/overflow_check.js.map +1 -0
- package/dist/src/engine/session/prompt.d.ts +45 -0
- package/dist/src/engine/session/prompt.js +584 -0
- package/dist/src/engine/session/prompt.js.map +1 -0
- package/dist/src/engine/session/provider_transform.d.ts +59 -0
- package/dist/src/engine/session/provider_transform.js +193 -0
- package/dist/src/engine/session/provider_transform.js.map +1 -0
- package/dist/src/engine/session/retry_logic.d.ts +12 -0
- package/dist/src/engine/session/retry_logic.js +72 -0
- package/dist/src/engine/session/retry_logic.js.map +1 -0
- package/dist/src/engine/session/system_prompt.d.ts +9 -0
- package/dist/src/engine/session/system_prompt.js +96 -0
- package/dist/src/engine/session/system_prompt.js.map +1 -0
- package/dist/src/engine/session/tool_registry.d.ts +5 -0
- package/dist/src/engine/session/tool_registry.js +117 -0
- package/dist/src/engine/session/tool_registry.js.map +1 -0
- package/dist/src/export.d.ts +13 -0
- package/dist/src/export.js +103 -0
- package/dist/src/export.js.map +1 -0
- package/dist/src/mdns.d.ts +7 -0
- package/dist/src/mdns.js +60 -0
- package/dist/src/mdns.js.map +1 -0
- package/dist/src/rag_worker.d.ts +38 -0
- package/dist/src/rag_worker.js +435 -0
- package/dist/src/rag_worker.js.map +1 -0
- package/dist/src/server.d.ts +11 -0
- package/dist/src/server.js +214 -0
- package/dist/src/server.js.map +1 -0
- package/dist/src/stats.d.ts +45 -0
- package/dist/src/stats.js +233 -0
- package/dist/src/stats.js.map +1 -0
- package/dist/src/tui.d.ts +29 -0
- package/dist/src/tui.js +1053 -0
- package/dist/src/tui.js.map +1 -0
- package/package.json +7 -4
- package/src/cli.ts +247 -5
- package/src/export.ts +122 -0
- package/src/mdns.ts +53 -0
- package/src/server.ts +32 -0
- package/src/stats.ts +290 -0
- package/src/tui.ts +749 -248
|
@@ -0,0 +1,215 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
// Generated by BoneScript compiler. DO NOT EDIT.
|
|
3
|
+
// WebSocket server for realtime channels
|
|
4
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
5
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
6
|
+
};
|
|
7
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
8
|
+
exports.broadcastToChannel = exports.unsubscribeFromChannel = exports.subscribeToChannel = exports.setupWebSocketServer = void 0;
|
|
9
|
+
const ws_1 = require("ws");
|
|
10
|
+
const jsonwebtoken_1 = __importDefault(require("jsonwebtoken"));
|
|
11
|
+
const logger_1 = require("./logger");
|
|
12
|
+
const JWT_SECRET = process.env.JWT_SECRET || "bonescript-dev-secret-change-in-production";
|
|
13
|
+
// Redis pub/sub for multi-instance WebSocket broadcasting
|
|
14
|
+
let redisSub = null;
|
|
15
|
+
let redisPub = null;
|
|
16
|
+
if (process.env.REDIS_URL) {
|
|
17
|
+
try {
|
|
18
|
+
const { createClient } = require("redis");
|
|
19
|
+
redisSub = createClient({ url: process.env.REDIS_URL });
|
|
20
|
+
redisPub = createClient({ url: process.env.REDIS_URL });
|
|
21
|
+
Promise.all([redisSub.connect(), redisPub.connect()]).then(() => {
|
|
22
|
+
logger_1.logger.info("redis_connected", { event: "startup" });
|
|
23
|
+
}).catch((e) => {
|
|
24
|
+
logger_1.logger.error("redis_connect_failed", { event: "startup", metadata: { error: e.message } });
|
|
25
|
+
redisSub = null;
|
|
26
|
+
redisPub = null;
|
|
27
|
+
});
|
|
28
|
+
}
|
|
29
|
+
catch {
|
|
30
|
+
redisSub = null;
|
|
31
|
+
redisPub = null;
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
const clients = new Map();
|
|
35
|
+
const CHANNELS = {
|
|
36
|
+
"session_events": {
|
|
37
|
+
ordering: "causal",
|
|
38
|
+
persistence: "last_1000",
|
|
39
|
+
max_size: 100000,
|
|
40
|
+
},
|
|
41
|
+
"part_stream": {
|
|
42
|
+
ordering: "fifo",
|
|
43
|
+
persistence: "last_100",
|
|
44
|
+
max_size: 10000,
|
|
45
|
+
},
|
|
46
|
+
};
|
|
47
|
+
const messageBuffers = new Map();
|
|
48
|
+
function getBuffer(channel) {
|
|
49
|
+
const buf = messageBuffers.get(channel);
|
|
50
|
+
if (buf)
|
|
51
|
+
return buf;
|
|
52
|
+
const fresh = [];
|
|
53
|
+
messageBuffers.set(channel, fresh);
|
|
54
|
+
return fresh;
|
|
55
|
+
}
|
|
56
|
+
function persistMessage(channel, msg) {
|
|
57
|
+
const cfg = CHANNELS[channel];
|
|
58
|
+
if (!cfg || cfg.persistence === "none")
|
|
59
|
+
return;
|
|
60
|
+
const buf = getBuffer(channel);
|
|
61
|
+
buf.push(msg);
|
|
62
|
+
// Honor persistence config (last_N, full)
|
|
63
|
+
const match = cfg.persistence.match(/^last_(\d+)$/);
|
|
64
|
+
if (match) {
|
|
65
|
+
const limit = parseInt(match[1], 10);
|
|
66
|
+
while (buf.length > limit)
|
|
67
|
+
buf.shift();
|
|
68
|
+
}
|
|
69
|
+
else if (buf.length > cfg.max_size) {
|
|
70
|
+
buf.shift();
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
function setupWebSocketServer(httpServer) {
|
|
74
|
+
const wss = new ws_1.WebSocketServer({ server: httpServer, path: "/ws" });
|
|
75
|
+
wss.on("connection", (socket, req) => {
|
|
76
|
+
const url = new URL(req.url || "/", `http://${req.headers.host}`);
|
|
77
|
+
const channel = url.searchParams.get("channel") || "";
|
|
78
|
+
const token = url.searchParams.get("token") || "";
|
|
79
|
+
if (!CHANNELS[channel]) {
|
|
80
|
+
socket.send(JSON.stringify({ type: "error", message: "Unknown channel: " + channel }));
|
|
81
|
+
socket.close();
|
|
82
|
+
return;
|
|
83
|
+
}
|
|
84
|
+
let userId;
|
|
85
|
+
try {
|
|
86
|
+
const decoded = jsonwebtoken_1.default.verify(token, JWT_SECRET);
|
|
87
|
+
userId = decoded.sub;
|
|
88
|
+
}
|
|
89
|
+
catch {
|
|
90
|
+
socket.send(JSON.stringify({ type: "error", message: "Authentication failed" }));
|
|
91
|
+
socket.close();
|
|
92
|
+
return;
|
|
93
|
+
}
|
|
94
|
+
const client = { socket, user_id: userId, channel, topics: new Set() };
|
|
95
|
+
const set = clients.get(channel) || new Set();
|
|
96
|
+
set.add(client);
|
|
97
|
+
clients.set(channel, set);
|
|
98
|
+
console.log(`[ws] User ${userId} connected to ${channel} (${set.size} active)`);
|
|
99
|
+
// Send buffered history
|
|
100
|
+
const history = messageBuffers.get(channel) || [];
|
|
101
|
+
for (const msg of history) {
|
|
102
|
+
socket.send(JSON.stringify(msg));
|
|
103
|
+
}
|
|
104
|
+
socket.send(JSON.stringify({ type: "connected", channel, history_size: history.length }));
|
|
105
|
+
socket.on("message", (data) => {
|
|
106
|
+
try {
|
|
107
|
+
const msg = JSON.parse(data.toString());
|
|
108
|
+
if (msg.type === "subscribe" && msg.topic) {
|
|
109
|
+
client.topics.add(msg.topic);
|
|
110
|
+
return;
|
|
111
|
+
}
|
|
112
|
+
if (msg.type === "unsubscribe" && msg.topic) {
|
|
113
|
+
client.topics.delete(msg.topic);
|
|
114
|
+
return;
|
|
115
|
+
}
|
|
116
|
+
// Broadcast
|
|
117
|
+
const broadcast = {
|
|
118
|
+
type: msg.type || "message",
|
|
119
|
+
payload: msg.payload || msg,
|
|
120
|
+
from: userId,
|
|
121
|
+
timestamp: new Date().toISOString(),
|
|
122
|
+
};
|
|
123
|
+
persistMessage(channel, broadcast);
|
|
124
|
+
broadcastToChannel(channel, broadcast, client);
|
|
125
|
+
}
|
|
126
|
+
catch (e) {
|
|
127
|
+
socket.send(JSON.stringify({ type: "error", message: e.message }));
|
|
128
|
+
}
|
|
129
|
+
});
|
|
130
|
+
socket.on("close", () => {
|
|
131
|
+
const set = clients.get(channel);
|
|
132
|
+
if (set)
|
|
133
|
+
set.delete(client);
|
|
134
|
+
console.log(`[ws] User ${userId} disconnected from ${channel}`);
|
|
135
|
+
});
|
|
136
|
+
// Heartbeat
|
|
137
|
+
const heartbeat = setInterval(() => {
|
|
138
|
+
if (socket.readyState === ws_1.WebSocket.OPEN) {
|
|
139
|
+
socket.ping();
|
|
140
|
+
}
|
|
141
|
+
else {
|
|
142
|
+
clearInterval(heartbeat);
|
|
143
|
+
}
|
|
144
|
+
}, 30000);
|
|
145
|
+
});
|
|
146
|
+
// Bridge: forward eventBus events to WebSocket clients
|
|
147
|
+
// Channel 'session_events' bridge
|
|
148
|
+
// Channel 'part_stream' bridge
|
|
149
|
+
// Subscribe to Redis channels for cross-instance delivery
|
|
150
|
+
if (redisSub) {
|
|
151
|
+
redisSub.subscribe("ws:session_events", (message) => {
|
|
152
|
+
const set = clients.get("session_events");
|
|
153
|
+
if (!set)
|
|
154
|
+
return;
|
|
155
|
+
for (const client of set) {
|
|
156
|
+
if (client.socket.readyState === ws_1.WebSocket.OPEN)
|
|
157
|
+
client.socket.send(message);
|
|
158
|
+
}
|
|
159
|
+
}).catch(() => { });
|
|
160
|
+
redisSub.subscribe("ws:part_stream", (message) => {
|
|
161
|
+
const set = clients.get("part_stream");
|
|
162
|
+
if (!set)
|
|
163
|
+
return;
|
|
164
|
+
for (const client of set) {
|
|
165
|
+
if (client.socket.readyState === ws_1.WebSocket.OPEN)
|
|
166
|
+
client.socket.send(message);
|
|
167
|
+
}
|
|
168
|
+
}).catch(() => { });
|
|
169
|
+
}
|
|
170
|
+
return wss;
|
|
171
|
+
}
|
|
172
|
+
exports.setupWebSocketServer = setupWebSocketServer;
|
|
173
|
+
function broadcastToChannel(channel, message, exclude) {
|
|
174
|
+
const set = clients.get(channel);
|
|
175
|
+
const data = JSON.stringify(message);
|
|
176
|
+
// Broadcast to local clients
|
|
177
|
+
if (set) {
|
|
178
|
+
for (const client of set) {
|
|
179
|
+
if (client === exclude)
|
|
180
|
+
continue;
|
|
181
|
+
if (client.socket.readyState === ws_1.WebSocket.OPEN) {
|
|
182
|
+
client.socket.send(data);
|
|
183
|
+
}
|
|
184
|
+
}
|
|
185
|
+
}
|
|
186
|
+
// Publish to Redis for cross-instance delivery
|
|
187
|
+
if (redisPub) {
|
|
188
|
+
redisPub.publish(`ws:${channel}`, data).catch(() => { });
|
|
189
|
+
}
|
|
190
|
+
// Notify in-process subscribers (used by SSE adapter)
|
|
191
|
+
const subs = inProcessSubs.get(channel);
|
|
192
|
+
if (subs) {
|
|
193
|
+
for (const handler of subs) {
|
|
194
|
+
try {
|
|
195
|
+
handler(message);
|
|
196
|
+
}
|
|
197
|
+
catch { /* non-fatal */ }
|
|
198
|
+
}
|
|
199
|
+
}
|
|
200
|
+
}
|
|
201
|
+
exports.broadcastToChannel = broadcastToChannel;
|
|
202
|
+
const inProcessSubs = new Map();
|
|
203
|
+
function subscribeToChannel(channel, handler) {
|
|
204
|
+
const set = inProcessSubs.get(channel) || new Set();
|
|
205
|
+
set.add(handler);
|
|
206
|
+
inProcessSubs.set(channel, set);
|
|
207
|
+
}
|
|
208
|
+
exports.subscribeToChannel = subscribeToChannel;
|
|
209
|
+
function unsubscribeFromChannel(channel, handler) {
|
|
210
|
+
const set = inProcessSubs.get(channel);
|
|
211
|
+
if (set)
|
|
212
|
+
set.delete(handler);
|
|
213
|
+
}
|
|
214
|
+
exports.unsubscribeFromChannel = unsubscribeFromChannel;
|
|
215
|
+
//# sourceMappingURL=websocket.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"websocket.js","sourceRoot":"","sources":["../../../../../bone/output/session/src/websocket.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"algorithms.js","sourceRoot":"","sources":["../../../../../bone/output/workspace/src/algorithms.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
import { Request, Response, NextFunction } from "express";
|
|
2
|
+
export interface AuthContext {
|
|
3
|
+
authenticated: boolean;
|
|
4
|
+
actor_id: string | null;
|
|
5
|
+
trace_id: string;
|
|
6
|
+
}
|
|
7
|
+
export declare function authMiddleware(req: Request, res: Response, next: NextFunction): void;
|
|
8
|
+
export declare function requireAuth(req: Request, res: Response, next: NextFunction): void;
|
|
@@ -0,0 +1,56 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.requireAuth = exports.authMiddleware = void 0;
|
|
7
|
+
const jsonwebtoken_1 = __importDefault(require("jsonwebtoken"));
|
|
8
|
+
// JWT_SECRET must be set in production. The server will refuse to start without it
|
|
9
|
+
// when NODE_ENV is "production" to prevent accidental use of a weak fallback.
|
|
10
|
+
const JWT_SECRET = (() => {
|
|
11
|
+
const secret = process.env.JWT_SECRET;
|
|
12
|
+
if (!secret) {
|
|
13
|
+
if (process.env.NODE_ENV === "production") {
|
|
14
|
+
console.error("[FATAL] JWT_SECRET environment variable is not set. Refusing to start in production.");
|
|
15
|
+
process.exit(1);
|
|
16
|
+
}
|
|
17
|
+
console.warn("[WARN] JWT_SECRET is not set. Using insecure default — do not use in production.");
|
|
18
|
+
return "bonescript-dev-secret-do-not-use-in-production";
|
|
19
|
+
}
|
|
20
|
+
if (secret.length < 32) {
|
|
21
|
+
console.warn("[WARN] JWT_SECRET is shorter than 32 characters. Use a longer secret in production.");
|
|
22
|
+
}
|
|
23
|
+
return secret;
|
|
24
|
+
})();
|
|
25
|
+
function authMiddleware(req, res, next) {
|
|
26
|
+
const header = req.headers.authorization;
|
|
27
|
+
if (!header || !header.startsWith("Bearer ")) {
|
|
28
|
+
req.auth = { authenticated: false, actor_id: null, trace_id: req.headers["x-trace-id"] || "" };
|
|
29
|
+
next();
|
|
30
|
+
return;
|
|
31
|
+
}
|
|
32
|
+
try {
|
|
33
|
+
const token = header.slice(7);
|
|
34
|
+
const decoded = jsonwebtoken_1.default.verify(token, JWT_SECRET);
|
|
35
|
+
req.auth = {
|
|
36
|
+
authenticated: true,
|
|
37
|
+
actor_id: decoded.sub,
|
|
38
|
+
trace_id: req.headers["x-trace-id"] || "",
|
|
39
|
+
};
|
|
40
|
+
}
|
|
41
|
+
catch {
|
|
42
|
+
req.auth = { authenticated: false, actor_id: null, trace_id: req.headers["x-trace-id"] || "" };
|
|
43
|
+
}
|
|
44
|
+
next();
|
|
45
|
+
}
|
|
46
|
+
exports.authMiddleware = authMiddleware;
|
|
47
|
+
function requireAuth(req, res, next) {
|
|
48
|
+
const auth = req.auth;
|
|
49
|
+
if (!auth || !auth.authenticated) {
|
|
50
|
+
res.status(401).json({ error: { code: "UNAUTHORIZED", message: "Authentication required" } });
|
|
51
|
+
return;
|
|
52
|
+
}
|
|
53
|
+
next();
|
|
54
|
+
}
|
|
55
|
+
exports.requireAuth = requireAuth;
|
|
56
|
+
//# sourceMappingURL=auth.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"auth.js","sourceRoot":"","sources":["../../../../../bone/output/workspace/src/auth.ts"],"names":[],"mappings":";;;;;;AAEA,gEAA+B;AAE/B,mFAAmF;AACnF,8EAA8E;AAC9E,MAAM,UAAU,GAAG,CAAC,GAAG,EAAE;IACvB,MAAM,MAAM,GAAG,OAAO,CAAC,GAAG,CAAC,UAAU,CAAC;IACtC,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,IAAI,OAAO,CAAC,GAAG,CAAC,QAAQ,KAAK,YAAY,EAAE,CAAC;YAC1C,OAAO,CAAC,KAAK,CAAC,sFAAsF,CAAC,CAAC;YACtG,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QAClB,CAAC;QACD,OAAO,CAAC,IAAI,CAAC,kFAAkF,CAAC,CAAC;QACjG,OAAO,gDAAgD,CAAC;IAC1D,CAAC;IACD,IAAI,MAAM,CAAC,MAAM,GAAG,EAAE,EAAE,CAAC;QACvB,OAAO,CAAC,IAAI,CAAC,qFAAqF,CAAC,CAAC;IACtG,CAAC;IACD,OAAO,MAAM,CAAC;AAChB,CAAC,CAAC,EAAE,CAAC;AAQL,SAAgB,cAAc,CAAC,GAAY,EAAE,GAAa,EAAE,IAAkB;IAC5E,MAAM,MAAM,GAAG,GAAG,CAAC,OAAO,CAAC,aAAa,CAAC;IACzC,IAAI,CAAC,MAAM,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,SAAS,CAAC,EAAE,CAAC;QAC5C,GAAW,CAAC,IAAI,GAAG,EAAE,aAAa,EAAE,KAAK,EAAE,QAAQ,EAAE,IAAI,EAAE,QAAQ,EAAE,GAAG,CAAC,OAAO,CAAC,YAAY,CAAW,IAAI,EAAE,EAAE,CAAC;QAClH,IAAI,EAAE,CAAC;QACP,OAAO;IACT,CAAC;IACD,IAAI,CAAC;QACH,MAAM,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;QAC9B,MAAM,OAAO,GAAG,sBAAG,CAAC,MAAM,CAAC,KAAK,EAAE,UAAU,CAAoB,CAAC;QAChE,GAAW,CAAC,IAAI,GAAG;YAClB,aAAa,EAAE,IAAI;YACnB,QAAQ,EAAE,OAAO,CAAC,GAAG;YACrB,QAAQ,EAAE,GAAG,CAAC,OAAO,CAAC,YAAY,CAAW,IAAI,EAAE;SACpD,CAAC;IACJ,CAAC;IAAC,MAAM,CAAC;QACN,GAAW,CAAC,IAAI,GAAG,EAAE,aAAa,EAAE,KAAK,EAAE,QAAQ,EAAE,IAAI,EAAE,QAAQ,EAAE,GAAG,CAAC,OAAO,CAAC,YAAY,CAAW,IAAI,EAAE,EAAE,CAAC;IACpH,CAAC;IACD,IAAI,EAAE,CAAC;AACT,CAAC;AAnBD,wCAmBC;AAED,SAAgB,WAAW,CAAC,GAAY,EAAE,GAAa,EAAE,IAAkB;IACzE,MAAM,IAAI,GAAiB,GAAW,CAAC,IAAI,CAAC;IAC5C,IAAI,CAAC,IAAI,IAAI,CAAC,IAAI,CAAC,aAAa,EAAE,CAAC;QACjC,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,EAAE,IAAI,EAAE,cAAc,EAAE,OAAO,EAAE,yBAAyB,EAAE,EAAE,CAAC,CAAC;QAC9F,OAAO;IACT,CAAC;IACD,IAAI,EAAE,CAAC;AACT,CAAC;AAPD,kCAOC"}
|
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
import { PoolClient } from "pg";
|
|
2
|
+
export declare const pool: any;
|
|
3
|
+
export declare function query<T = any>(text: string, params?: any[]): Promise<T[]>;
|
|
4
|
+
export declare function queryOne<T = any>(text: string, params?: any[]): Promise<T | null>;
|
|
5
|
+
export declare function execute(text: string, params?: any[]): Promise<number>;
|
|
6
|
+
export declare function transaction<T>(fn: (client: PoolClient) => Promise<T>): Promise<T>;
|
|
@@ -0,0 +1,63 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.transaction = exports.execute = exports.queryOne = exports.query = exports.pool = void 0;
|
|
4
|
+
// Generated by BoneScript compiler. DO NOT EDIT.
|
|
5
|
+
const pg_1 = require("pg");
|
|
6
|
+
// Lazy pool — created on first use so DATABASE_URL is read after dotenv loads
|
|
7
|
+
let _pool = null;
|
|
8
|
+
function getPool() {
|
|
9
|
+
if (!_pool) {
|
|
10
|
+
_pool = new pg_1.Pool({ connectionString: process.env.DATABASE_URL || "postgresql://localhost:5432/workspace_domain", max: 20 });
|
|
11
|
+
_pool.on("error", (err) => console.error("[DB] Pool error (non-fatal):", err.message));
|
|
12
|
+
}
|
|
13
|
+
return _pool;
|
|
14
|
+
}
|
|
15
|
+
exports.pool = new Proxy({}, { get(_t, p) { return getPool()[p]; } });
|
|
16
|
+
async function query(text, params) {
|
|
17
|
+
try {
|
|
18
|
+
const result = await exports.pool.query(text, params);
|
|
19
|
+
return result.rows;
|
|
20
|
+
}
|
|
21
|
+
catch (e) {
|
|
22
|
+
throw new Error(`DB query failed: ${e.message}`);
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
exports.query = query;
|
|
26
|
+
async function queryOne(text, params) {
|
|
27
|
+
try {
|
|
28
|
+
const rows = await query(text, params);
|
|
29
|
+
return rows[0] || null;
|
|
30
|
+
}
|
|
31
|
+
catch (e) {
|
|
32
|
+
throw new Error(`DB query failed: ${e.message}`);
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
exports.queryOne = queryOne;
|
|
36
|
+
async function execute(text, params) {
|
|
37
|
+
try {
|
|
38
|
+
const result = await exports.pool.query(text, params);
|
|
39
|
+
return result.rowCount || 0;
|
|
40
|
+
}
|
|
41
|
+
catch (e) {
|
|
42
|
+
throw new Error(`DB execute failed: ${e.message}`);
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
exports.execute = execute;
|
|
46
|
+
async function transaction(fn) {
|
|
47
|
+
const client = await exports.pool.connect();
|
|
48
|
+
try {
|
|
49
|
+
await client.query("BEGIN");
|
|
50
|
+
const result = await fn(client);
|
|
51
|
+
await client.query("COMMIT");
|
|
52
|
+
return result;
|
|
53
|
+
}
|
|
54
|
+
catch (e) {
|
|
55
|
+
await client.query("ROLLBACK");
|
|
56
|
+
throw e;
|
|
57
|
+
}
|
|
58
|
+
finally {
|
|
59
|
+
client.release();
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
exports.transaction = transaction;
|
|
63
|
+
//# sourceMappingURL=db.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"db.js","sourceRoot":"","sources":["../../../../../bone/output/workspace/src/db.ts"],"names":[],"mappings":";;;AAAA,iDAAiD;AACjD,2BAAsC;AAEtC,8EAA8E;AAC9E,IAAI,KAAK,GAAgB,IAAI,CAAC;AAC9B,SAAS,OAAO;IACd,IAAI,CAAC,KAAK,EAAE,CAAC;QACb,KAAK,GAAG,IAAI,SAAI,CAAC,EAAE,gBAAgB,EAAE,OAAO,CAAC,GAAG,CAAC,YAAY,IAAI,8CAA8C,EAAE,GAAG,EAAE,EAAE,EAAE,CAAC,CAAC;QAC5H,KAAK,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,GAAU,EAAE,EAAE,CAAC,OAAO,CAAC,KAAK,CAAC,8BAA8B,EAAE,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC;IAC9F,CAAC;IACD,OAAO,KAAK,CAAC;AACf,CAAC;AACY,QAAA,IAAI,GAAG,IAAI,KAAK,CAAC,EAAU,EAAE,EAAE,GAAG,CAAC,EAAO,EAAE,CAAM,IAAI,OAAQ,OAAO,EAAU,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;AAE/F,KAAK,UAAU,KAAK,CAAU,IAAY,EAAE,MAAc;IAC/D,IAAI,CAAC;QAAC,MAAM,MAAM,GAAG,MAAM,YAAI,CAAC,KAAK,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC;QAAC,OAAO,MAAM,CAAC,IAAW,CAAC;IAAC,CAAC;IACjF,OAAO,CAAM,EAAE,CAAC;QAAC,MAAM,IAAI,KAAK,CAAC,oBAAoB,CAAC,CAAC,OAAO,EAAE,CAAC,CAAC;IAAC,CAAC;AACtE,CAAC;AAHD,sBAGC;AACM,KAAK,UAAU,QAAQ,CAAU,IAAY,EAAE,MAAc;IAClE,IAAI,CAAC;QAAC,MAAM,IAAI,GAAG,MAAM,KAAK,CAAI,IAAI,EAAE,MAAM,CAAC,CAAC;QAAC,OAAO,IAAI,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC;IAAC,CAAC;IAC1E,OAAO,CAAM,EAAE,CAAC;QAAC,MAAM,IAAI,KAAK,CAAC,oBAAoB,CAAC,CAAC,OAAO,EAAE,CAAC,CAAC;IAAC,CAAC;AACtE,CAAC;AAHD,4BAGC;AACM,KAAK,UAAU,OAAO,CAAC,IAAY,EAAE,MAAc;IACxD,IAAI,CAAC;QAAC,MAAM,MAAM,GAAG,MAAM,YAAI,CAAC,KAAK,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC;QAAC,OAAO,MAAM,CAAC,QAAQ,IAAI,CAAC,CAAC;IAAC,CAAC;IACnF,OAAO,CAAM,EAAE,CAAC;QAAC,MAAM,IAAI,KAAK,CAAC,sBAAsB,CAAC,CAAC,OAAO,EAAE,CAAC,CAAC;IAAC,CAAC;AACxE,CAAC;AAHD,0BAGC;AACM,KAAK,UAAU,WAAW,CAAI,EAAsC;IACzE,MAAM,MAAM,GAAG,MAAM,YAAI,CAAC,OAAO,EAAE,CAAC;IACpC,IAAI,CAAC;QAAC,MAAM,MAAM,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;QAAC,MAAM,MAAM,GAAG,MAAM,EAAE,CAAC,MAAM,CAAC,CAAC;QAAC,MAAM,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC;QAAC,OAAO,MAAM,CAAC;IAAC,CAAC;IAClH,OAAO,CAAC,EAAE,CAAC;QAAC,MAAM,MAAM,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;QAAC,MAAM,CAAC,CAAC;IAAC,CAAC;YAC9C,CAAC;QAAC,MAAM,CAAC,OAAO,EAAE,CAAC;IAAC,CAAC;AAC/B,CAAC;AALD,kCAKC"}
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
/// <reference types="node" />
|
|
2
|
+
/// <reference types="node" />
|
|
3
|
+
/// <reference types="node" />
|
|
4
|
+
/// <reference types="node" />
|
|
5
|
+
/// <reference types="node" />
|
|
6
|
+
import { PoolClient } from "pg";
|
|
7
|
+
export type EventDeliveryMode = "in_process" | "durable";
|
|
8
|
+
export interface EventMetadata {
|
|
9
|
+
source: string;
|
|
10
|
+
timestamp: Date;
|
|
11
|
+
correlation_id: string;
|
|
12
|
+
causation_id: string;
|
|
13
|
+
}
|
|
14
|
+
export interface SystemEvent {
|
|
15
|
+
type: string;
|
|
16
|
+
payload: Record<string, unknown>;
|
|
17
|
+
metadata: EventMetadata;
|
|
18
|
+
}
|
|
19
|
+
type Handler = (event: SystemEvent) => Promise<void>;
|
|
20
|
+
export declare const eventBus: {
|
|
21
|
+
subscribe(type: string, handler: Handler): void;
|
|
22
|
+
publish(type: string, payload: Record<string, unknown>, source: string, correlationId?: string, client?: PoolClient): Promise<void>;
|
|
23
|
+
startWorker(intervalMs?: number): NodeJS.Timeout | null;
|
|
24
|
+
};
|
|
25
|
+
export {};
|
|
@@ -0,0 +1,184 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
// Generated by BoneScript compiler. DO NOT EDIT.
|
|
3
|
+
// Durable event bus with transactional outbox pattern.
|
|
4
|
+
// Set EVENT_MODE=durable in .env to enable.
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.eventBus = void 0;
|
|
7
|
+
const uuid_1 = require("uuid");
|
|
8
|
+
const db_1 = require("./db");
|
|
9
|
+
const logger_1 = require("./logger");
|
|
10
|
+
const metrics_1 = require("./metrics");
|
|
11
|
+
const MODE = process.env.EVENT_MODE || "in_process";
|
|
12
|
+
// Events requiring exactly_once delivery (deduplicated)
|
|
13
|
+
const EXACTLY_ONCE_EVENTS = new Set(["SnapshotCreated"]);
|
|
14
|
+
// Events requiring at_least_once delivery (retried until ack)
|
|
15
|
+
const AT_LEAST_ONCE_EVENTS = new Set(["WorkspaceInitialized"]);
|
|
16
|
+
// ─── In-Process Bus ──────────────────────────────────────────────────────────
|
|
17
|
+
class InProcessBus {
|
|
18
|
+
constructor() {
|
|
19
|
+
this.handlers = new Map();
|
|
20
|
+
}
|
|
21
|
+
subscribe(type, handler) {
|
|
22
|
+
const existing = this.handlers.get(type) || [];
|
|
23
|
+
existing.push(handler);
|
|
24
|
+
this.handlers.set(type, existing);
|
|
25
|
+
}
|
|
26
|
+
async publish(type, payload, source, correlationId) {
|
|
27
|
+
const event = {
|
|
28
|
+
type,
|
|
29
|
+
payload,
|
|
30
|
+
metadata: {
|
|
31
|
+
source,
|
|
32
|
+
timestamp: new Date(),
|
|
33
|
+
correlation_id: correlationId || (0, uuid_1.v4)(),
|
|
34
|
+
causation_id: (0, uuid_1.v4)(),
|
|
35
|
+
},
|
|
36
|
+
};
|
|
37
|
+
(0, metrics_1.counter)("event.published", { type, mode: "in_process" });
|
|
38
|
+
const handlers = this.handlers.get(type) || [];
|
|
39
|
+
for (const handler of handlers) {
|
|
40
|
+
try {
|
|
41
|
+
await handler(event);
|
|
42
|
+
(0, metrics_1.counter)("event.delivered", { type, mode: "in_process" });
|
|
43
|
+
}
|
|
44
|
+
catch (e) {
|
|
45
|
+
(0, metrics_1.counter)("event.delivery_failed", { type, mode: "in_process" });
|
|
46
|
+
logger_1.logger.error("event_handler_failed", { event: type, metadata: { error: e.message } });
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
// ─── Durable Bus (Transactional Outbox) ──────────────────────────────────────
|
|
52
|
+
class DurableBus {
|
|
53
|
+
constructor() {
|
|
54
|
+
this.handlers = new Map();
|
|
55
|
+
}
|
|
56
|
+
subscribe(type, handler) {
|
|
57
|
+
const existing = this.handlers.get(type) || [];
|
|
58
|
+
existing.push(handler);
|
|
59
|
+
this.handlers.set(type, existing);
|
|
60
|
+
}
|
|
61
|
+
// Write event to outbox within the current transaction (or a new one)
|
|
62
|
+
async publish(type, payload, source, correlationId, client) {
|
|
63
|
+
const eventId = (0, uuid_1.v4)();
|
|
64
|
+
const corrId = correlationId || (0, uuid_1.v4)();
|
|
65
|
+
const sql = `
|
|
66
|
+
INSERT INTO event_outbox (id, event_type, payload, source, correlation_id)
|
|
67
|
+
VALUES ($1, $2, $3, $4, $5)
|
|
68
|
+
`;
|
|
69
|
+
const params = [eventId, type, JSON.stringify({ ...payload, _event_id: eventId }), source, corrId];
|
|
70
|
+
if (client) {
|
|
71
|
+
// Write within caller's transaction — atomicity guaranteed
|
|
72
|
+
await client.query(sql, params);
|
|
73
|
+
}
|
|
74
|
+
else {
|
|
75
|
+
await db_1.pool.query(sql, params);
|
|
76
|
+
}
|
|
77
|
+
(0, metrics_1.counter)("event.outboxed", { type });
|
|
78
|
+
}
|
|
79
|
+
// Called by the background worker
|
|
80
|
+
async flush() {
|
|
81
|
+
const client = await db_1.pool.connect();
|
|
82
|
+
try {
|
|
83
|
+
await client.query("BEGIN");
|
|
84
|
+
// Fetch pending events (lock rows to prevent concurrent processing)
|
|
85
|
+
const { rows } = await client.query(`
|
|
86
|
+
SELECT id, event_type, payload, source, correlation_id, attempts
|
|
87
|
+
FROM event_outbox
|
|
88
|
+
WHERE status = 'pending' AND scheduled_at <= NOW()
|
|
89
|
+
ORDER BY scheduled_at ASC
|
|
90
|
+
LIMIT 50
|
|
91
|
+
FOR UPDATE SKIP LOCKED
|
|
92
|
+
`);
|
|
93
|
+
for (const row of rows) {
|
|
94
|
+
try {
|
|
95
|
+
// exactly_once: check deduplication table
|
|
96
|
+
if (EXACTLY_ONCE_EVENTS.has(row.event_type)) {
|
|
97
|
+
const { rows: dup } = await client.query("SELECT 1 FROM event_processed WHERE event_id = $1", [row.payload._event_id || row.id]);
|
|
98
|
+
if (dup.length > 0) {
|
|
99
|
+
await client.query("UPDATE event_outbox SET status = 'delivered', delivered_at = NOW() WHERE id = $1", [row.id]);
|
|
100
|
+
continue;
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
const event = {
|
|
104
|
+
type: row.event_type,
|
|
105
|
+
payload: row.payload,
|
|
106
|
+
metadata: {
|
|
107
|
+
source: row.source,
|
|
108
|
+
timestamp: new Date(),
|
|
109
|
+
correlation_id: row.correlation_id,
|
|
110
|
+
causation_id: (0, uuid_1.v4)(),
|
|
111
|
+
},
|
|
112
|
+
};
|
|
113
|
+
const handlers = this.handlers.get(row.event_type) || [];
|
|
114
|
+
for (const handler of handlers) {
|
|
115
|
+
await handler(event);
|
|
116
|
+
}
|
|
117
|
+
// Mark delivered
|
|
118
|
+
await client.query("UPDATE event_outbox SET status = 'delivered', delivered_at = NOW(), attempts = attempts + 1 WHERE id = $1", [row.id]);
|
|
119
|
+
// Record for exactly_once deduplication
|
|
120
|
+
if (EXACTLY_ONCE_EVENTS.has(row.event_type)) {
|
|
121
|
+
await client.query("INSERT INTO event_processed (event_id, event_type) VALUES ($1, $2) ON CONFLICT DO NOTHING", [row.payload._event_id || row.id, row.event_type]);
|
|
122
|
+
}
|
|
123
|
+
(0, metrics_1.counter)("event.delivered", { type: row.event_type, mode: "durable" });
|
|
124
|
+
}
|
|
125
|
+
catch (e) {
|
|
126
|
+
const maxAttempts = AT_LEAST_ONCE_EVENTS.has(row.event_type) ? 10 : 3;
|
|
127
|
+
const newAttempts = row.attempts + 1;
|
|
128
|
+
const status = newAttempts >= maxAttempts ? "dead_letter" : "pending";
|
|
129
|
+
const backoffMs = Math.min(1000 * Math.pow(2, newAttempts), 300000);
|
|
130
|
+
await client.query(`UPDATE event_outbox
|
|
131
|
+
SET attempts = $1, last_error = $2, status = $3,
|
|
132
|
+
scheduled_at = NOW() + ($4 || ' milliseconds')::interval
|
|
133
|
+
WHERE id = $5`, [newAttempts, e.message, status, backoffMs, row.id]);
|
|
134
|
+
(0, metrics_1.counter)("event.delivery_failed", { type: row.event_type, mode: "durable" });
|
|
135
|
+
logger_1.logger.error("event_delivery_failed", { event: row.event_type, metadata: { error: e.message, attempts: newAttempts } });
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
await client.query("COMMIT");
|
|
139
|
+
}
|
|
140
|
+
catch (e) {
|
|
141
|
+
await client.query("ROLLBACK");
|
|
142
|
+
throw e;
|
|
143
|
+
}
|
|
144
|
+
finally {
|
|
145
|
+
client.release();
|
|
146
|
+
}
|
|
147
|
+
}
|
|
148
|
+
// Start background worker
|
|
149
|
+
startWorker(intervalMs = 1000) {
|
|
150
|
+
logger_1.logger.info("event_worker_started", { event: "startup", metadata: { interval_ms: intervalMs } });
|
|
151
|
+
return setInterval(async () => {
|
|
152
|
+
try {
|
|
153
|
+
await this.flush();
|
|
154
|
+
}
|
|
155
|
+
catch (e) {
|
|
156
|
+
logger_1.logger.error("event_worker_error", { event: "flush_failed", metadata: { error: e.message } });
|
|
157
|
+
}
|
|
158
|
+
}, intervalMs);
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
// ─── Unified Interface ────────────────────────────────────────────────────────
|
|
162
|
+
const inProcess = new InProcessBus();
|
|
163
|
+
const durable = new DurableBus();
|
|
164
|
+
exports.eventBus = {
|
|
165
|
+
subscribe(type, handler) {
|
|
166
|
+
inProcess.subscribe(type, handler);
|
|
167
|
+
durable.subscribe(type, handler);
|
|
168
|
+
},
|
|
169
|
+
async publish(type, payload, source, correlationId, client) {
|
|
170
|
+
if (MODE === "durable") {
|
|
171
|
+
await durable.publish(type, payload, source, correlationId, client);
|
|
172
|
+
}
|
|
173
|
+
else {
|
|
174
|
+
await inProcess.publish(type, payload, source, correlationId);
|
|
175
|
+
}
|
|
176
|
+
},
|
|
177
|
+
startWorker(intervalMs) {
|
|
178
|
+
if (MODE === "durable") {
|
|
179
|
+
return durable.startWorker(intervalMs);
|
|
180
|
+
}
|
|
181
|
+
return null;
|
|
182
|
+
},
|
|
183
|
+
};
|
|
184
|
+
//# sourceMappingURL=events.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"events.js","sourceRoot":"","sources":["../../../../../bone/output/workspace/src/events.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
export type LogLevel = "debug" | "info" | "warn" | "error" | "fatal";
|
|
2
|
+
export type LogStatus = "success" | "failure" | "timeout" | "rejected";
|
|
3
|
+
export interface LogEntry {
|
|
4
|
+
timestamp: string;
|
|
5
|
+
level: LogLevel;
|
|
6
|
+
service: string;
|
|
7
|
+
trace_id: string;
|
|
8
|
+
span_id: string;
|
|
9
|
+
event: string;
|
|
10
|
+
duration_ms?: number;
|
|
11
|
+
status: LogStatus;
|
|
12
|
+
entity_id?: string;
|
|
13
|
+
actor_id?: string | null;
|
|
14
|
+
error_code?: string | null;
|
|
15
|
+
metadata?: Record<string, unknown>;
|
|
16
|
+
}
|
|
17
|
+
declare class Logger {
|
|
18
|
+
private service;
|
|
19
|
+
constructor(service: string);
|
|
20
|
+
emit(entry: Omit<LogEntry, "timestamp" | "service" | "span_id">): void;
|
|
21
|
+
info(event: string, fields?: Partial<LogEntry>): void;
|
|
22
|
+
warn(event: string, fields?: Partial<LogEntry>): void;
|
|
23
|
+
error(event: string, fields?: Partial<LogEntry>): void;
|
|
24
|
+
debug(event: string, fields?: Partial<LogEntry>): void;
|
|
25
|
+
}
|
|
26
|
+
export declare function createLogger(service: string): Logger;
|
|
27
|
+
export declare const logger: Logger;
|
|
28
|
+
export {};
|