camelagi 0.5.0
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/LICENSE +21 -0
- package/README.md +224 -0
- package/camelagi.mjs +2 -0
- package/config.example.yaml +107 -0
- package/dist/agent/agent-openai.js +206 -0
- package/dist/agent/agent-openai.js.map +1 -0
- package/dist/agent/agent-sdk.js +209 -0
- package/dist/agent/agent-sdk.js.map +1 -0
- package/dist/agent/tool-adapter.js +31 -0
- package/dist/agent/tool-adapter.js.map +1 -0
- package/dist/agent/types.js +3 -0
- package/dist/agent/types.js.map +1 -0
- package/dist/agent.js +17 -0
- package/dist/agent.js.map +1 -0
- package/dist/approval-forward.js +42 -0
- package/dist/approval-forward.js.map +1 -0
- package/dist/approvals.js +151 -0
- package/dist/approvals.js.map +1 -0
- package/dist/boot.js +34 -0
- package/dist/boot.js.map +1 -0
- package/dist/bootstrap.js +451 -0
- package/dist/bootstrap.js.map +1 -0
- package/dist/camelagi-gateway.mjs +93611 -0
- package/dist/camelagi-gateway.mjs.map +7 -0
- package/dist/channels/adapter.js +10 -0
- package/dist/channels/adapter.js.map +1 -0
- package/dist/channels/discord.js +232 -0
- package/dist/channels/discord.js.map +1 -0
- package/dist/channels/handler.js +349 -0
- package/dist/channels/handler.js.map +1 -0
- package/dist/channels/index.js +19 -0
- package/dist/channels/index.js.map +1 -0
- package/dist/channels/registry.js +71 -0
- package/dist/channels/registry.js.map +1 -0
- package/dist/channels/telegram.js +83 -0
- package/dist/channels/telegram.js.map +1 -0
- package/dist/channels/types.js +3 -0
- package/dist/channels/types.js.map +1 -0
- package/dist/chunker.js +102 -0
- package/dist/chunker.js.map +1 -0
- package/dist/cli/cmd-agents.js +65 -0
- package/dist/cli/cmd-agents.js.map +1 -0
- package/dist/cli/cmd-bootstrap.js +10 -0
- package/dist/cli/cmd-bootstrap.js.map +1 -0
- package/dist/cli/cmd-chat.js +32 -0
- package/dist/cli/cmd-chat.js.map +1 -0
- package/dist/cli/cmd-config.js +88 -0
- package/dist/cli/cmd-config.js.map +1 -0
- package/dist/cli/cmd-cron.js +120 -0
- package/dist/cli/cmd-cron.js.map +1 -0
- package/dist/cli/cmd-daemon.js +37 -0
- package/dist/cli/cmd-daemon.js.map +1 -0
- package/dist/cli/cmd-doctor.js +18 -0
- package/dist/cli/cmd-doctor.js.map +1 -0
- package/dist/cli/cmd-logs.js +30 -0
- package/dist/cli/cmd-logs.js.map +1 -0
- package/dist/cli/cmd-pairing.js +41 -0
- package/dist/cli/cmd-pairing.js.map +1 -0
- package/dist/cli/cmd-reset.js +39 -0
- package/dist/cli/cmd-reset.js.map +1 -0
- package/dist/cli/cmd-serve.js +30 -0
- package/dist/cli/cmd-serve.js.map +1 -0
- package/dist/cli/cmd-sessions.js +56 -0
- package/dist/cli/cmd-sessions.js.map +1 -0
- package/dist/cli/cmd-setup.js +11 -0
- package/dist/cli/cmd-setup.js.map +1 -0
- package/dist/cli/cmd-soul.js +43 -0
- package/dist/cli/cmd-soul.js.map +1 -0
- package/dist/cli/parse.js +50 -0
- package/dist/cli/parse.js.map +1 -0
- package/dist/cli/registry.js +15 -0
- package/dist/cli/registry.js.map +1 -0
- package/dist/cli.js +103 -0
- package/dist/cli.js.map +1 -0
- package/dist/compact.js +92 -0
- package/dist/compact.js.map +1 -0
- package/dist/config.js +153 -0
- package/dist/config.js.map +1 -0
- package/dist/constants.js +21 -0
- package/dist/constants.js.map +1 -0
- package/dist/core/config.js +212 -0
- package/dist/core/config.js.map +1 -0
- package/dist/core/constants.js +21 -0
- package/dist/core/constants.js.map +1 -0
- package/dist/core/errors.js +5 -0
- package/dist/core/errors.js.map +1 -0
- package/dist/core/log.js +41 -0
- package/dist/core/log.js.map +1 -0
- package/dist/core/models.js +123 -0
- package/dist/core/models.js.map +1 -0
- package/dist/core/types.js +3 -0
- package/dist/core/types.js.map +1 -0
- package/dist/core/update-check.js +51 -0
- package/dist/core/update-check.js.map +1 -0
- package/dist/cron.js +81 -0
- package/dist/cron.js.map +1 -0
- package/dist/daemon.js +109 -0
- package/dist/daemon.js.map +1 -0
- package/dist/doctor.js +194 -0
- package/dist/doctor.js.map +1 -0
- package/dist/errors.js +5 -0
- package/dist/errors.js.map +1 -0
- package/dist/extensions/approval-forward.js +42 -0
- package/dist/extensions/approval-forward.js.map +1 -0
- package/dist/extensions/approvals.js +144 -0
- package/dist/extensions/approvals.js.map +1 -0
- package/dist/extensions/cron.js +306 -0
- package/dist/extensions/cron.js.map +1 -0
- package/dist/extensions/hooks.js +72 -0
- package/dist/extensions/hooks.js.map +1 -0
- package/dist/extensions/skills.js +97 -0
- package/dist/extensions/skills.js.map +1 -0
- package/dist/gateway/csrf.js +44 -0
- package/dist/gateway/csrf.js.map +1 -0
- package/dist/gateway/logger.js +81 -0
- package/dist/gateway/logger.js.map +1 -0
- package/dist/gateway/rate-limit.js +33 -0
- package/dist/gateway/rate-limit.js.map +1 -0
- package/dist/gateway/routes.js +315 -0
- package/dist/gateway/routes.js.map +1 -0
- package/dist/gateway/state.js +54 -0
- package/dist/gateway/state.js.map +1 -0
- package/dist/gateway/ws-handler.js +200 -0
- package/dist/gateway/ws-handler.js.map +1 -0
- package/dist/gateway-entry.js +16 -0
- package/dist/gateway-entry.js.map +1 -0
- package/dist/hooks.js +72 -0
- package/dist/hooks.js.map +1 -0
- package/dist/lanes.js +62 -0
- package/dist/lanes.js.map +1 -0
- package/dist/model.js +30 -0
- package/dist/model.js.map +1 -0
- package/dist/policy.js +22 -0
- package/dist/policy.js.map +1 -0
- package/dist/queue.js +45 -0
- package/dist/queue.js.map +1 -0
- package/dist/retry.js +96 -0
- package/dist/retry.js.map +1 -0
- package/dist/runs.js +83 -0
- package/dist/runs.js.map +1 -0
- package/dist/runtime/compact.js +99 -0
- package/dist/runtime/compact.js.map +1 -0
- package/dist/runtime/lanes.js +66 -0
- package/dist/runtime/lanes.js.map +1 -0
- package/dist/runtime/orchestrate.js +121 -0
- package/dist/runtime/orchestrate.js.map +1 -0
- package/dist/runtime/queue.js +50 -0
- package/dist/runtime/queue.js.map +1 -0
- package/dist/runtime/retry.js +127 -0
- package/dist/runtime/retry.js.map +1 -0
- package/dist/runtime/runs.js +105 -0
- package/dist/runtime/runs.js.map +1 -0
- package/dist/serve.js +209 -0
- package/dist/serve.js.map +1 -0
- package/dist/session.js +75 -0
- package/dist/session.js.map +1 -0
- package/dist/setup.js +254 -0
- package/dist/setup.js.map +1 -0
- package/dist/skills.js +89 -0
- package/dist/skills.js.map +1 -0
- package/dist/subagent.js +71 -0
- package/dist/subagent.js.map +1 -0
- package/dist/system-prompt.js +157 -0
- package/dist/system-prompt.js.map +1 -0
- package/dist/telegram/admin-bot.js +705 -0
- package/dist/telegram/admin-bot.js.map +1 -0
- package/dist/telegram/agent-bot.js +551 -0
- package/dist/telegram/agent-bot.js.map +1 -0
- package/dist/telegram/bot-approval.js +63 -0
- package/dist/telegram/bot-approval.js.map +1 -0
- package/dist/telegram/draft-stream.js +86 -0
- package/dist/telegram/draft-stream.js.map +1 -0
- package/dist/telegram/format.js +106 -0
- package/dist/telegram/format.js.map +1 -0
- package/dist/telegram/helpers.js +87 -0
- package/dist/telegram/helpers.js.map +1 -0
- package/dist/telegram/pairing-notify.js +52 -0
- package/dist/telegram/pairing-notify.js.map +1 -0
- package/dist/telegram/pairing.js +138 -0
- package/dist/telegram/pairing.js.map +1 -0
- package/dist/telegram/resolve.js +33 -0
- package/dist/telegram/resolve.js.map +1 -0
- package/dist/telegram/transcribe.js +77 -0
- package/dist/telegram/transcribe.js.map +1 -0
- package/dist/telegram/types.js +3 -0
- package/dist/telegram/types.js.map +1 -0
- package/dist/telegram/voice-wizard.js +84 -0
- package/dist/telegram/voice-wizard.js.map +1 -0
- package/dist/telegram/wizard.js +89 -0
- package/dist/telegram/wizard.js.map +1 -0
- package/dist/telegram/wizards.js +297 -0
- package/dist/telegram/wizards.js.map +1 -0
- package/dist/telegram-admin.js +800 -0
- package/dist/telegram-admin.js.map +1 -0
- package/dist/telegram.js +118 -0
- package/dist/telegram.js.map +1 -0
- package/dist/tools/cron.js +94 -0
- package/dist/tools/cron.js.map +1 -0
- package/dist/tools/edit.js +29 -0
- package/dist/tools/edit.js.map +1 -0
- package/dist/tools/exec.js +38 -0
- package/dist/tools/exec.js.map +1 -0
- package/dist/tools/fetch.js +28 -0
- package/dist/tools/fetch.js.map +1 -0
- package/dist/tools/index.js +16 -0
- package/dist/tools/index.js.map +1 -0
- package/dist/tools/memory.js +164 -0
- package/dist/tools/memory.js.map +1 -0
- package/dist/tools/patch.js +284 -0
- package/dist/tools/patch.js.map +1 -0
- package/dist/tools/read.js +26 -0
- package/dist/tools/read.js.map +1 -0
- package/dist/tools/search.js +62 -0
- package/dist/tools/search.js.map +1 -0
- package/dist/tools/subagent.js +48 -0
- package/dist/tools/subagent.js.map +1 -0
- package/dist/tools/write.js +22 -0
- package/dist/tools/write.js.map +1 -0
- package/dist/tui/commands.js +450 -0
- package/dist/tui/commands.js.map +1 -0
- package/dist/tui/components/assistant-message.js +26 -0
- package/dist/tui/components/assistant-message.js.map +1 -0
- package/dist/tui/components/chat-log.js +94 -0
- package/dist/tui/components/chat-log.js.map +1 -0
- package/dist/tui/components/custom-editor.js +40 -0
- package/dist/tui/components/custom-editor.js.map +1 -0
- package/dist/tui/components/hint-bar.js +13 -0
- package/dist/tui/components/hint-bar.js.map +1 -0
- package/dist/tui/components/tool-execution.js +73 -0
- package/dist/tui/components/tool-execution.js.map +1 -0
- package/dist/tui/components/user-message.js +19 -0
- package/dist/tui/components/user-message.js.map +1 -0
- package/dist/tui/components/welcome.js +147 -0
- package/dist/tui/components/welcome.js.map +1 -0
- package/dist/tui/context.js +3 -0
- package/dist/tui/context.js.map +1 -0
- package/dist/tui/theme.js +91 -0
- package/dist/tui/theme.js.map +1 -0
- package/dist/tui/tui.js +389 -0
- package/dist/tui/tui.js.map +1 -0
- package/dist/tui/ws-handler.js +154 -0
- package/dist/tui/ws-handler.js.map +1 -0
- package/dist/types.js +3 -0
- package/dist/types.js.map +1 -0
- package/dist/usage.js +88 -0
- package/dist/usage.js.map +1 -0
- package/dist/workspace.js +245 -0
- package/dist/workspace.js.map +1 -0
- package/package.json +74 -0
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
// CSRF protection — blocks cross-origin mutation requests from browsers
|
|
2
|
+
const LOOPBACK_HOSTS = new Set(["localhost", "127.0.0.1", "::1"]);
|
|
3
|
+
const SAFE_METHODS = new Set(["GET", "HEAD", "OPTIONS"]);
|
|
4
|
+
function isLoopbackOrigin(raw) {
|
|
5
|
+
try {
|
|
6
|
+
const url = new URL(raw);
|
|
7
|
+
return LOOPBACK_HOSTS.has(url.hostname);
|
|
8
|
+
}
|
|
9
|
+
catch {
|
|
10
|
+
return false;
|
|
11
|
+
}
|
|
12
|
+
}
|
|
13
|
+
/**
|
|
14
|
+
* Blocks browser-originated cross-site mutation requests.
|
|
15
|
+
* Non-browser clients (curl, Node fetch) that don't send Origin/Sec-Fetch-Site pass through.
|
|
16
|
+
*/
|
|
17
|
+
export function csrfProtection() {
|
|
18
|
+
return (req, res, next) => {
|
|
19
|
+
if (SAFE_METHODS.has(req.method)) {
|
|
20
|
+
next();
|
|
21
|
+
return;
|
|
22
|
+
}
|
|
23
|
+
// Sec-Fetch-Site: if present, only allow same-origin / same-site / none
|
|
24
|
+
const fetchSite = req.headers["sec-fetch-site"];
|
|
25
|
+
if (fetchSite === "cross-site") {
|
|
26
|
+
res.status(403).json({ error: "Cross-site requests are not allowed" });
|
|
27
|
+
return;
|
|
28
|
+
}
|
|
29
|
+
// Origin header: if present, must be loopback
|
|
30
|
+
const origin = req.headers.origin;
|
|
31
|
+
if (origin && !isLoopbackOrigin(origin)) {
|
|
32
|
+
res.status(403).json({ error: "Non-local origin is not allowed" });
|
|
33
|
+
return;
|
|
34
|
+
}
|
|
35
|
+
// Referer header: if present and no Origin, must be loopback
|
|
36
|
+
const referer = req.headers.referer;
|
|
37
|
+
if (!origin && referer && !isLoopbackOrigin(referer)) {
|
|
38
|
+
res.status(403).json({ error: "Non-local referer is not allowed" });
|
|
39
|
+
return;
|
|
40
|
+
}
|
|
41
|
+
next();
|
|
42
|
+
};
|
|
43
|
+
}
|
|
44
|
+
//# sourceMappingURL=csrf.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"csrf.js","sourceRoot":"","sources":["../../src/gateway/csrf.ts"],"names":[],"mappings":"AAAA,wEAAwE;AAIxE,MAAM,cAAc,GAAG,IAAI,GAAG,CAAC,CAAC,WAAW,EAAE,WAAW,EAAE,KAAK,CAAC,CAAC,CAAC;AAClE,MAAM,YAAY,GAAG,IAAI,GAAG,CAAC,CAAC,KAAK,EAAE,MAAM,EAAE,SAAS,CAAC,CAAC,CAAC;AAEzD,SAAS,gBAAgB,CAAC,GAAW;IACnC,IAAI,CAAC;QACH,MAAM,GAAG,GAAG,IAAI,GAAG,CAAC,GAAG,CAAC,CAAC;QACzB,OAAO,cAAc,CAAC,GAAG,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;IAC1C,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,KAAK,CAAC;IACf,CAAC;AACH,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,cAAc;IAC5B,OAAO,CAAC,GAAY,EAAE,GAAa,EAAE,IAAkB,EAAQ,EAAE;QAC/D,IAAI,YAAY,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC;YAAC,IAAI,EAAE,CAAC;YAAC,OAAO;QAAC,CAAC;QAErD,wEAAwE;QACxE,MAAM,SAAS,GAAG,GAAG,CAAC,OAAO,CAAC,gBAAgB,CAAuB,CAAC;QACtE,IAAI,SAAS,KAAK,YAAY,EAAE,CAAC;YAC/B,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,qCAAqC,EAAE,CAAC,CAAC;YACvE,OAAO;QACT,CAAC;QAED,8CAA8C;QAC9C,MAAM,MAAM,GAAG,GAAG,CAAC,OAAO,CAAC,MAAM,CAAC;QAClC,IAAI,MAAM,IAAI,CAAC,gBAAgB,CAAC,MAAM,CAAC,EAAE,CAAC;YACxC,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,iCAAiC,EAAE,CAAC,CAAC;YACnE,OAAO;QACT,CAAC;QAED,6DAA6D;QAC7D,MAAM,OAAO,GAAG,GAAG,CAAC,OAAO,CAAC,OAAO,CAAC;QACpC,IAAI,CAAC,MAAM,IAAI,OAAO,IAAI,CAAC,gBAAgB,CAAC,OAAO,CAAC,EAAE,CAAC;YACrD,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,kCAAkC,EAAE,CAAC,CAAC;YACpE,OAAO;QACT,CAAC;QAED,IAAI,EAAE,CAAC;IACT,CAAC,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1,81 @@
|
|
|
1
|
+
// JSON-line request logger — writes to ~/.camelagi/logs/server.log
|
|
2
|
+
import fs from "node:fs";
|
|
3
|
+
import path from "node:path";
|
|
4
|
+
import os from "node:os";
|
|
5
|
+
const LOG_DIR = path.join(os.homedir(), ".camelagi", "logs");
|
|
6
|
+
const LOG_FILE = path.join(LOG_DIR, "server.log");
|
|
7
|
+
const MAX_AGE_DAYS = 7;
|
|
8
|
+
function ensureLogDir() {
|
|
9
|
+
fs.mkdirSync(LOG_DIR, { recursive: true });
|
|
10
|
+
}
|
|
11
|
+
/** Delete rotated log files older than MAX_AGE_DAYS */
|
|
12
|
+
export function rotateOldLogs() {
|
|
13
|
+
ensureLogDir();
|
|
14
|
+
const cutoff = Date.now() - MAX_AGE_DAYS * 86_400_000;
|
|
15
|
+
try {
|
|
16
|
+
for (const f of fs.readdirSync(LOG_DIR)) {
|
|
17
|
+
if (!f.startsWith("server-") || !f.endsWith(".log"))
|
|
18
|
+
continue;
|
|
19
|
+
const stat = fs.statSync(path.join(LOG_DIR, f));
|
|
20
|
+
if (stat.mtimeMs < cutoff)
|
|
21
|
+
fs.unlinkSync(path.join(LOG_DIR, f));
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
catch { /* best effort */ }
|
|
25
|
+
}
|
|
26
|
+
/** Rotate current log if it's from a previous day */
|
|
27
|
+
function maybeRotate() {
|
|
28
|
+
try {
|
|
29
|
+
if (!fs.existsSync(LOG_FILE))
|
|
30
|
+
return;
|
|
31
|
+
const stat = fs.statSync(LOG_FILE);
|
|
32
|
+
const logDate = new Date(stat.mtimeMs).toISOString().slice(0, 10);
|
|
33
|
+
const today = new Date().toISOString().slice(0, 10);
|
|
34
|
+
if (logDate !== today) {
|
|
35
|
+
fs.renameSync(LOG_FILE, path.join(LOG_DIR, `server-${logDate}.log`));
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
catch { /* best effort */ }
|
|
39
|
+
}
|
|
40
|
+
function writeLog(entry) {
|
|
41
|
+
try {
|
|
42
|
+
fs.appendFileSync(LOG_FILE, JSON.stringify(entry) + "\n");
|
|
43
|
+
}
|
|
44
|
+
catch { /* best effort */ }
|
|
45
|
+
}
|
|
46
|
+
/** Express middleware that logs each request as a JSON line */
|
|
47
|
+
export function requestLogger() {
|
|
48
|
+
ensureLogDir();
|
|
49
|
+
maybeRotate();
|
|
50
|
+
rotateOldLogs();
|
|
51
|
+
return (req, res, next) => {
|
|
52
|
+
const start = Date.now();
|
|
53
|
+
res.on("finish", () => {
|
|
54
|
+
const entry = {
|
|
55
|
+
ts: new Date().toISOString(),
|
|
56
|
+
method: req.method,
|
|
57
|
+
path: req.path,
|
|
58
|
+
status: res.statusCode,
|
|
59
|
+
ms: Date.now() - start,
|
|
60
|
+
};
|
|
61
|
+
const sid = req.body?.session ?? req.params?.id;
|
|
62
|
+
if (sid)
|
|
63
|
+
entry.sessionId = sid;
|
|
64
|
+
if (res.statusCode >= 400)
|
|
65
|
+
entry.error = res.statusMessage;
|
|
66
|
+
writeLog(entry);
|
|
67
|
+
});
|
|
68
|
+
next();
|
|
69
|
+
};
|
|
70
|
+
}
|
|
71
|
+
/** Tail the current log file — used by `camelagi logs` */
|
|
72
|
+
export function tailLog(lines = 50) {
|
|
73
|
+
if (!fs.existsSync(LOG_FILE))
|
|
74
|
+
return "No logs yet.";
|
|
75
|
+
const content = fs.readFileSync(LOG_FILE, "utf-8").trim();
|
|
76
|
+
if (!content)
|
|
77
|
+
return "No logs yet.";
|
|
78
|
+
const allLines = content.split("\n");
|
|
79
|
+
return allLines.slice(-lines).join("\n");
|
|
80
|
+
}
|
|
81
|
+
//# sourceMappingURL=logger.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"logger.js","sourceRoot":"","sources":["../../src/gateway/logger.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
// Simple in-memory rate limiter — no dependencies
|
|
2
|
+
const hits = new Map();
|
|
3
|
+
export function rateLimit(opts) {
|
|
4
|
+
// Periodically clean up expired entries
|
|
5
|
+
setInterval(() => {
|
|
6
|
+
const now = Date.now();
|
|
7
|
+
for (const [key, entry] of hits) {
|
|
8
|
+
if (now >= entry.resetAt)
|
|
9
|
+
hits.delete(key);
|
|
10
|
+
}
|
|
11
|
+
}, opts.windowMs).unref();
|
|
12
|
+
return (req, res, next) => {
|
|
13
|
+
const key = req.ip ?? "unknown";
|
|
14
|
+
const now = Date.now();
|
|
15
|
+
let entry = hits.get(key);
|
|
16
|
+
if (!entry || now >= entry.resetAt) {
|
|
17
|
+
entry = { count: 0, resetAt: now + opts.windowMs };
|
|
18
|
+
hits.set(key, entry);
|
|
19
|
+
}
|
|
20
|
+
entry.count++;
|
|
21
|
+
res.setHeader("X-RateLimit-Limit", opts.max);
|
|
22
|
+
res.setHeader("X-RateLimit-Remaining", Math.max(0, opts.max - entry.count));
|
|
23
|
+
res.setHeader("X-RateLimit-Reset", Math.ceil(entry.resetAt / 1000));
|
|
24
|
+
if (entry.count > opts.max) {
|
|
25
|
+
const retryAfter = Math.ceil((entry.resetAt - now) / 1000);
|
|
26
|
+
res.setHeader("Retry-After", retryAfter);
|
|
27
|
+
res.status(429).json({ error: "Too many requests, please try again later" });
|
|
28
|
+
return;
|
|
29
|
+
}
|
|
30
|
+
next();
|
|
31
|
+
};
|
|
32
|
+
}
|
|
33
|
+
//# sourceMappingURL=rate-limit.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"rate-limit.js","sourceRoot":"","sources":["../../src/gateway/rate-limit.ts"],"names":[],"mappings":"AAAA,kDAAkD;AASlD,MAAM,IAAI,GAAG,IAAI,GAAG,EAA8C,CAAC;AAEnE,MAAM,UAAU,SAAS,CAAC,IAAmB;IAC3C,wCAAwC;IACxC,WAAW,CAAC,GAAG,EAAE;QACf,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QACvB,KAAK,MAAM,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,IAAI,EAAE,CAAC;YAChC,IAAI,GAAG,IAAI,KAAK,CAAC,OAAO;gBAAE,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;QAC7C,CAAC;IACH,CAAC,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC,KAAK,EAAE,CAAC;IAE1B,OAAO,CAAC,GAAY,EAAE,GAAa,EAAE,IAAkB,EAAQ,EAAE;QAC/D,MAAM,GAAG,GAAG,GAAG,CAAC,EAAE,IAAI,SAAS,CAAC;QAChC,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QAEvB,IAAI,KAAK,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;QAC1B,IAAI,CAAC,KAAK,IAAI,GAAG,IAAI,KAAK,CAAC,OAAO,EAAE,CAAC;YACnC,KAAK,GAAG,EAAE,KAAK,EAAE,CAAC,EAAE,OAAO,EAAE,GAAG,GAAG,IAAI,CAAC,QAAQ,EAAE,CAAC;YACnD,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,KAAK,CAAC,CAAC;QACvB,CAAC;QAED,KAAK,CAAC,KAAK,EAAE,CAAC;QAEd,GAAG,CAAC,SAAS,CAAC,mBAAmB,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC;QAC7C,GAAG,CAAC,SAAS,CAAC,uBAAuB,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,GAAG,GAAG,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC;QAC5E,GAAG,CAAC,SAAS,CAAC,mBAAmB,EAAE,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,OAAO,GAAG,IAAI,CAAC,CAAC,CAAC;QAEpE,IAAI,KAAK,CAAC,KAAK,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;YAC3B,MAAM,UAAU,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,KAAK,CAAC,OAAO,GAAG,GAAG,CAAC,GAAG,IAAI,CAAC,CAAC;YAC3D,GAAG,CAAC,SAAS,CAAC,aAAa,EAAE,UAAU,CAAC,CAAC;YACzC,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,2CAA2C,EAAE,CAAC,CAAC;YAC7E,OAAO;QACT,CAAC;QAED,IAAI,EAAE,CAAC;IACT,CAAC,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1,315 @@
|
|
|
1
|
+
// Gateway REST API routes
|
|
2
|
+
import { loadConfig, saveConfig } from "../core/config.js";
|
|
3
|
+
import { createClient } from "../model.js";
|
|
4
|
+
import { buildSystemPrompt } from "../system-prompt.js";
|
|
5
|
+
import { agentMemoryDir, seedAgentWorkspace } from "../workspace.js";
|
|
6
|
+
import { loadMessages, listSessions, deleteSession } from "../session.js";
|
|
7
|
+
import { getActiveRunCount } from "../runtime/runs.js";
|
|
8
|
+
import { getLaneStats } from "../runtime/lanes.js";
|
|
9
|
+
import { orchestrate } from "../runtime/orchestrate.js";
|
|
10
|
+
import { errorMessage } from "../core/errors.js";
|
|
11
|
+
import { submitDecision } from "../extensions/approvals.js";
|
|
12
|
+
import fs from "node:fs";
|
|
13
|
+
import path from "node:path";
|
|
14
|
+
import { listPendingRequests, approveRequest, denyRequest } from "../telegram/pairing.js";
|
|
15
|
+
import { listPendingBotApprovals, approveBotApproval, denyBotApproval } from "../telegram/bot-approval.js";
|
|
16
|
+
import { notifyUserApproved, notifyUserOfDenial } from "../telegram/pairing-notify.js";
|
|
17
|
+
import { checkAuth, logMessage } from "./state.js";
|
|
18
|
+
export function registerRoutes(app, state) {
|
|
19
|
+
app.get("/health", (_req, res) => {
|
|
20
|
+
res.json({
|
|
21
|
+
status: "ok",
|
|
22
|
+
uptime: Math.floor((Date.now() - state.startTime) / 1000),
|
|
23
|
+
sessions: listSessions().length,
|
|
24
|
+
clients: state.clients.size,
|
|
25
|
+
activeRuns: getActiveRunCount(),
|
|
26
|
+
lanes: getLaneStats(),
|
|
27
|
+
});
|
|
28
|
+
});
|
|
29
|
+
app.post("/chat", async (req, res) => {
|
|
30
|
+
if (!checkAuth(state, req.headers.authorization)) {
|
|
31
|
+
res.status(401).json({ error: "Unauthorized" });
|
|
32
|
+
return;
|
|
33
|
+
}
|
|
34
|
+
const { message, session } = req.body;
|
|
35
|
+
if (!message || typeof message !== "string") {
|
|
36
|
+
res.status(400).json({ error: "message is required" });
|
|
37
|
+
return;
|
|
38
|
+
}
|
|
39
|
+
const sid = session ?? `http-${Date.now()}`;
|
|
40
|
+
logMessage(state, "http", "in", sid, message);
|
|
41
|
+
try {
|
|
42
|
+
const result = await orchestrate({
|
|
43
|
+
sessionId: sid,
|
|
44
|
+
message,
|
|
45
|
+
config: state.config,
|
|
46
|
+
systemPrompt: state.systemPrompt,
|
|
47
|
+
client: state.client,
|
|
48
|
+
});
|
|
49
|
+
if (result.response) {
|
|
50
|
+
logMessage(state, "http", "out", sid, result.response);
|
|
51
|
+
}
|
|
52
|
+
res.json({ response: result.response, session: sid });
|
|
53
|
+
}
|
|
54
|
+
catch (err) {
|
|
55
|
+
logMessage(state, "http", "out", sid, `ERROR: ${errorMessage(err)}`);
|
|
56
|
+
res.status(500).json({ error: errorMessage(err) });
|
|
57
|
+
}
|
|
58
|
+
});
|
|
59
|
+
// Sessions
|
|
60
|
+
app.get("/sessions", (req, res) => {
|
|
61
|
+
if (!checkAuth(state, req.headers.authorization)) {
|
|
62
|
+
res.status(401).json({ error: "Unauthorized" });
|
|
63
|
+
return;
|
|
64
|
+
}
|
|
65
|
+
res.json(listSessions());
|
|
66
|
+
});
|
|
67
|
+
app.get("/sessions/:id/messages", (req, res) => {
|
|
68
|
+
if (!checkAuth(state, req.headers.authorization)) {
|
|
69
|
+
res.status(401).json({ error: "Unauthorized" });
|
|
70
|
+
return;
|
|
71
|
+
}
|
|
72
|
+
const messages = loadMessages(req.params.id);
|
|
73
|
+
res.json(messages.map((m) => ({ role: m.role, content: m.content })));
|
|
74
|
+
});
|
|
75
|
+
app.delete("/sessions/:id", (req, res) => {
|
|
76
|
+
if (!checkAuth(state, req.headers.authorization)) {
|
|
77
|
+
res.status(401).json({ error: "Unauthorized" });
|
|
78
|
+
return;
|
|
79
|
+
}
|
|
80
|
+
deleteSession(req.params.id);
|
|
81
|
+
res.json({ ok: true });
|
|
82
|
+
});
|
|
83
|
+
// Agents
|
|
84
|
+
app.get("/agents", async (req, res) => {
|
|
85
|
+
if (!checkAuth(state, req.headers.authorization)) {
|
|
86
|
+
res.status(401).json({ error: "Unauthorized" });
|
|
87
|
+
return;
|
|
88
|
+
}
|
|
89
|
+
let runningIds = [];
|
|
90
|
+
try {
|
|
91
|
+
const { getAllChannels } = await import("../channels/registry.js");
|
|
92
|
+
runningIds = getAllChannels().flatMap(c => c.getActiveAgentIds());
|
|
93
|
+
}
|
|
94
|
+
catch { /* channels may not be loaded */ }
|
|
95
|
+
const agents = Object.entries(state.config.agents).map(([id, a]) => ({
|
|
96
|
+
id,
|
|
97
|
+
name: a.name,
|
|
98
|
+
admin: a.admin ?? false,
|
|
99
|
+
model: a.model ?? state.config.model,
|
|
100
|
+
telegram: !!a.telegram?.botToken,
|
|
101
|
+
running: runningIds.includes(id),
|
|
102
|
+
dir: agentMemoryDir(id),
|
|
103
|
+
}));
|
|
104
|
+
res.json(agents);
|
|
105
|
+
});
|
|
106
|
+
app.post("/agents", (req, res) => {
|
|
107
|
+
if (!checkAuth(state, req.headers.authorization)) {
|
|
108
|
+
res.status(401).json({ error: "Unauthorized" });
|
|
109
|
+
return;
|
|
110
|
+
}
|
|
111
|
+
const { id, name, model, description, telegramToken, allowedUsers } = req.body;
|
|
112
|
+
if (!id || !name) {
|
|
113
|
+
res.status(400).json({ error: "id and name are required" });
|
|
114
|
+
return;
|
|
115
|
+
}
|
|
116
|
+
if (state.config.agents[id]) {
|
|
117
|
+
res.status(409).json({ error: `Agent "${id}" already exists` });
|
|
118
|
+
return;
|
|
119
|
+
}
|
|
120
|
+
seedAgentWorkspace(id, name, description);
|
|
121
|
+
const agentConfig = { name };
|
|
122
|
+
if (model && model !== state.config.model)
|
|
123
|
+
agentConfig.model = model;
|
|
124
|
+
if (telegramToken) {
|
|
125
|
+
agentConfig.telegram = { botToken: telegramToken, allowedUsers: allowedUsers ?? [] };
|
|
126
|
+
}
|
|
127
|
+
const agents = { ...state.config.agents, [id]: agentConfig };
|
|
128
|
+
saveConfig({ agents });
|
|
129
|
+
state.config = loadConfig();
|
|
130
|
+
res.status(201).json({ id, name, dir: agentMemoryDir(id) });
|
|
131
|
+
});
|
|
132
|
+
app.delete("/agents/:id", (req, res) => {
|
|
133
|
+
if (!checkAuth(state, req.headers.authorization)) {
|
|
134
|
+
res.status(401).json({ error: "Unauthorized" });
|
|
135
|
+
return;
|
|
136
|
+
}
|
|
137
|
+
const { id } = req.params;
|
|
138
|
+
if (!state.config.agents[id]) {
|
|
139
|
+
res.status(404).json({ error: `Agent "${id}" not found` });
|
|
140
|
+
return;
|
|
141
|
+
}
|
|
142
|
+
const agents = { ...state.config.agents };
|
|
143
|
+
delete agents[id];
|
|
144
|
+
saveConfig({ agents });
|
|
145
|
+
state.config = loadConfig();
|
|
146
|
+
res.json({ ok: true });
|
|
147
|
+
});
|
|
148
|
+
// SOUL.md
|
|
149
|
+
app.get("/agents/:id/soul", (req, res) => {
|
|
150
|
+
if (!checkAuth(state, req.headers.authorization)) {
|
|
151
|
+
res.status(401).json({ error: "Unauthorized" });
|
|
152
|
+
return;
|
|
153
|
+
}
|
|
154
|
+
const { id } = req.params;
|
|
155
|
+
if (!state.config.agents[id]) {
|
|
156
|
+
res.status(404).json({ error: `Agent "${id}" not found` });
|
|
157
|
+
return;
|
|
158
|
+
}
|
|
159
|
+
const soulPath = path.join(agentMemoryDir(id), "SOUL.md");
|
|
160
|
+
if (!fs.existsSync(soulPath)) {
|
|
161
|
+
res.json({ content: "" });
|
|
162
|
+
return;
|
|
163
|
+
}
|
|
164
|
+
res.json({ content: fs.readFileSync(soulPath, "utf-8") });
|
|
165
|
+
});
|
|
166
|
+
app.put("/agents/:id/soul", (req, res) => {
|
|
167
|
+
if (!checkAuth(state, req.headers.authorization)) {
|
|
168
|
+
res.status(401).json({ error: "Unauthorized" });
|
|
169
|
+
return;
|
|
170
|
+
}
|
|
171
|
+
const { id } = req.params;
|
|
172
|
+
if (!state.config.agents[id]) {
|
|
173
|
+
res.status(404).json({ error: `Agent "${id}" not found` });
|
|
174
|
+
return;
|
|
175
|
+
}
|
|
176
|
+
const { content } = req.body;
|
|
177
|
+
if (typeof content !== "string") {
|
|
178
|
+
res.status(400).json({ error: "content is required" });
|
|
179
|
+
return;
|
|
180
|
+
}
|
|
181
|
+
const dir = agentMemoryDir(id);
|
|
182
|
+
fs.mkdirSync(dir, { recursive: true });
|
|
183
|
+
fs.writeFileSync(path.join(dir, "SOUL.md"), content);
|
|
184
|
+
res.json({ ok: true });
|
|
185
|
+
});
|
|
186
|
+
// Config
|
|
187
|
+
app.get("/config", (req, res) => {
|
|
188
|
+
if (!checkAuth(state, req.headers.authorization)) {
|
|
189
|
+
res.status(401).json({ error: "Unauthorized" });
|
|
190
|
+
return;
|
|
191
|
+
}
|
|
192
|
+
const safe = { ...state.config, apiKey: state.config.apiKey ? `***${state.config.apiKey.slice(-4)}` : undefined };
|
|
193
|
+
res.json(safe);
|
|
194
|
+
});
|
|
195
|
+
app.patch("/config", (req, res) => {
|
|
196
|
+
if (!checkAuth(state, req.headers.authorization)) {
|
|
197
|
+
res.status(401).json({ error: "Unauthorized" });
|
|
198
|
+
return;
|
|
199
|
+
}
|
|
200
|
+
const updates = req.body;
|
|
201
|
+
if (!updates || typeof updates !== "object") {
|
|
202
|
+
res.status(400).json({ error: "JSON body required" });
|
|
203
|
+
return;
|
|
204
|
+
}
|
|
205
|
+
delete updates.apiKey;
|
|
206
|
+
delete updates.serve;
|
|
207
|
+
saveConfig(updates);
|
|
208
|
+
state.config = loadConfig();
|
|
209
|
+
state.client = createClient(state.config);
|
|
210
|
+
state.systemPrompt = buildSystemPrompt(state.config.systemPrompt, state.config.skills);
|
|
211
|
+
res.json({ ok: true });
|
|
212
|
+
});
|
|
213
|
+
// Pairing
|
|
214
|
+
app.get("/pairing", (req, res) => {
|
|
215
|
+
if (!checkAuth(state, req.headers.authorization)) {
|
|
216
|
+
res.status(401).json({ error: "Unauthorized" });
|
|
217
|
+
return;
|
|
218
|
+
}
|
|
219
|
+
res.json(listPendingRequests());
|
|
220
|
+
});
|
|
221
|
+
app.post("/pairing/:code/approve", async (req, res) => {
|
|
222
|
+
if (!checkAuth(state, req.headers.authorization)) {
|
|
223
|
+
res.status(401).json({ error: "Unauthorized" });
|
|
224
|
+
return;
|
|
225
|
+
}
|
|
226
|
+
const request = approveRequest(req.params.code);
|
|
227
|
+
if (request) {
|
|
228
|
+
res.json({ ok: true, userId: request.userId, agentId: request.agentId });
|
|
229
|
+
// Notify the Telegram user they've been approved
|
|
230
|
+
try {
|
|
231
|
+
const { getActiveBots } = await import("../telegram.js");
|
|
232
|
+
await notifyUserApproved(request, getActiveBots());
|
|
233
|
+
}
|
|
234
|
+
catch { /* telegram may not be running */ }
|
|
235
|
+
}
|
|
236
|
+
else {
|
|
237
|
+
res.status(404).json({ error: "Request not found or expired" });
|
|
238
|
+
}
|
|
239
|
+
});
|
|
240
|
+
app.post("/pairing/:code/deny", async (req, res) => {
|
|
241
|
+
if (!checkAuth(state, req.headers.authorization)) {
|
|
242
|
+
res.status(401).json({ error: "Unauthorized" });
|
|
243
|
+
return;
|
|
244
|
+
}
|
|
245
|
+
const request = denyRequest(req.params.code);
|
|
246
|
+
if (request) {
|
|
247
|
+
res.json({ ok: true });
|
|
248
|
+
// Notify the Telegram user of denial
|
|
249
|
+
try {
|
|
250
|
+
const { getActiveBots } = await import("../telegram.js");
|
|
251
|
+
await notifyUserOfDenial(request, getActiveBots());
|
|
252
|
+
}
|
|
253
|
+
catch { /* telegram may not be running */ }
|
|
254
|
+
}
|
|
255
|
+
else {
|
|
256
|
+
res.status(404).json({ error: "Request not found or expired" });
|
|
257
|
+
}
|
|
258
|
+
});
|
|
259
|
+
// Bot approvals
|
|
260
|
+
app.get("/bot-approvals", (req, res) => {
|
|
261
|
+
if (!checkAuth(state, req.headers.authorization)) {
|
|
262
|
+
res.status(401).json({ error: "Unauthorized" });
|
|
263
|
+
return;
|
|
264
|
+
}
|
|
265
|
+
res.json(listPendingBotApprovals());
|
|
266
|
+
});
|
|
267
|
+
app.post("/bot-approvals/:agentId/approve", async (req, res) => {
|
|
268
|
+
if (!checkAuth(state, req.headers.authorization)) {
|
|
269
|
+
res.status(401).json({ error: "Unauthorized" });
|
|
270
|
+
return;
|
|
271
|
+
}
|
|
272
|
+
const approval = approveBotApproval(req.params.agentId);
|
|
273
|
+
if (!approval) {
|
|
274
|
+
res.status(404).json({ error: "Approval not found" });
|
|
275
|
+
return;
|
|
276
|
+
}
|
|
277
|
+
try {
|
|
278
|
+
const { startBot } = await import("../telegram.js");
|
|
279
|
+
await startBot(approval.agentId, approval.botToken, () => state.config, () => state.systemPrompt);
|
|
280
|
+
res.json({ ok: true, agentId: approval.agentId, botUsername: approval.botUsername });
|
|
281
|
+
}
|
|
282
|
+
catch (err) {
|
|
283
|
+
res.status(500).json({ error: errorMessage(err) });
|
|
284
|
+
}
|
|
285
|
+
});
|
|
286
|
+
app.post("/bot-approvals/:agentId/deny", (req, res) => {
|
|
287
|
+
if (!checkAuth(state, req.headers.authorization)) {
|
|
288
|
+
res.status(401).json({ error: "Unauthorized" });
|
|
289
|
+
return;
|
|
290
|
+
}
|
|
291
|
+
const approval = denyBotApproval(req.params.agentId);
|
|
292
|
+
if (approval) {
|
|
293
|
+
res.json({ ok: true });
|
|
294
|
+
}
|
|
295
|
+
else {
|
|
296
|
+
res.status(404).json({ error: "Approval not found" });
|
|
297
|
+
}
|
|
298
|
+
});
|
|
299
|
+
// Approvals
|
|
300
|
+
app.post("/approvals/:id/decide", (req, res) => {
|
|
301
|
+
if (!checkAuth(state, req.headers.authorization)) {
|
|
302
|
+
res.status(401).json({ error: "Unauthorized" });
|
|
303
|
+
return;
|
|
304
|
+
}
|
|
305
|
+
const { id } = req.params;
|
|
306
|
+
const { decision } = req.body;
|
|
307
|
+
if (!decision) {
|
|
308
|
+
res.status(400).json({ error: "decision is required (allow-once, allow-always, deny)" });
|
|
309
|
+
return;
|
|
310
|
+
}
|
|
311
|
+
const resolved = submitDecision(id, decision);
|
|
312
|
+
res.json({ ok: resolved });
|
|
313
|
+
});
|
|
314
|
+
}
|
|
315
|
+
//# sourceMappingURL=routes.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"routes.js","sourceRoot":"","sources":["../../src/gateway/routes.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
// Gateway shared state
|
|
2
|
+
import { createHash, timingSafeEqual } from "node:crypto";
|
|
3
|
+
import { loadMessages } from "../session.js";
|
|
4
|
+
import { compactHistory } from "../runtime/compact.js";
|
|
5
|
+
/** Timing-safe token comparison — prevents timing attacks that leak token info */
|
|
6
|
+
function safeEqual(a, b) {
|
|
7
|
+
const ha = createHash("sha256").update(a).digest();
|
|
8
|
+
const hb = createHash("sha256").update(b).digest();
|
|
9
|
+
return timingSafeEqual(ha, hb);
|
|
10
|
+
}
|
|
11
|
+
export function checkAuth(state, authHeader) {
|
|
12
|
+
if (!state.token)
|
|
13
|
+
return true;
|
|
14
|
+
if (!authHeader)
|
|
15
|
+
return false;
|
|
16
|
+
return safeEqual(authHeader, `Bearer ${state.token}`);
|
|
17
|
+
}
|
|
18
|
+
export function send(ws, data) {
|
|
19
|
+
if (ws.readyState === 1 /* WebSocket.OPEN */) {
|
|
20
|
+
ws.send(JSON.stringify(data));
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
export function broadcast(state, data) {
|
|
24
|
+
const json = JSON.stringify(data);
|
|
25
|
+
for (const ws of state.clients) {
|
|
26
|
+
if (ws.readyState === 1)
|
|
27
|
+
ws.send(json);
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
export function logMessage(state, channel, direction, sessionId, text) {
|
|
31
|
+
if (state.silent)
|
|
32
|
+
return;
|
|
33
|
+
const arrow = direction === "in" ? `\x1b[36m→\x1b[0m` : `\x1b[32m←\x1b[0m`;
|
|
34
|
+
const tag = `\x1b[90m[${channel}:${sessionId.slice(0, 16)}]\x1b[0m`;
|
|
35
|
+
const preview = text.slice(0, 160).replace(/\n/g, " ");
|
|
36
|
+
const suffix = text.length > 160 ? `\x1b[90m…\x1b[0m` : "";
|
|
37
|
+
console.log(` ${arrow} ${tag} ${preview}${suffix}`);
|
|
38
|
+
}
|
|
39
|
+
export function parseTokenFromUrl(url) {
|
|
40
|
+
if (!url)
|
|
41
|
+
return undefined;
|
|
42
|
+
const match = url.match(/[?&]token=([^&]+)/);
|
|
43
|
+
return match ? `Bearer ${match[1]}` : undefined;
|
|
44
|
+
}
|
|
45
|
+
/** Load history + run compaction for a session */
|
|
46
|
+
export async function prepareHistory(state, sid) {
|
|
47
|
+
let history = loadMessages(sid);
|
|
48
|
+
const result = await compactHistory(state.client, state.config.model, history, state.config.compaction);
|
|
49
|
+
if (result) {
|
|
50
|
+
return { history: result, compacted: true };
|
|
51
|
+
}
|
|
52
|
+
return { history, compacted: false };
|
|
53
|
+
}
|
|
54
|
+
//# sourceMappingURL=state.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"state.js","sourceRoot":"","sources":["../../src/gateway/state.ts"],"names":[],"mappings":"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"}
|