failproofai 0.0.6-beta.1 → 0.0.6-beta.3
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/.next/standalone/.failproofai/policies/review-policies.mjs +4 -3
- package/.next/standalone/.next/BUILD_ID +1 -1
- package/.next/standalone/.next/build-manifest.json +3 -3
- package/.next/standalone/.next/prerender-manifest.json +3 -3
- package/.next/standalone/.next/required-server-files.json +1 -1
- package/.next/standalone/.next/server/app/_global-error/page/server-reference-manifest.json +1 -1
- package/.next/standalone/.next/server/app/_global-error/page.js.nft.json +1 -1
- package/.next/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
- package/.next/standalone/.next/server/app/_global-error.html +1 -1
- package/.next/standalone/.next/server/app/_global-error.rsc +7 -7
- package/.next/standalone/.next/server/app/_global-error.segments/__PAGE__.segment.rsc +2 -2
- package/.next/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +7 -7
- package/.next/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +3 -3
- package/.next/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +3 -3
- package/.next/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
- package/.next/standalone/.next/server/app/_not-found/page/server-reference-manifest.json +1 -1
- package/.next/standalone/.next/server/app/_not-found/page.js.nft.json +1 -1
- package/.next/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
- package/.next/standalone/.next/server/app/_not-found.html +2 -2
- package/.next/standalone/.next/server/app/_not-found.rsc +15 -15
- package/.next/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +15 -15
- package/.next/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +4 -4
- package/.next/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +10 -10
- package/.next/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +2 -2
- package/.next/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +3 -3
- package/.next/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
- package/.next/standalone/.next/server/app/index.html +1 -1
- package/.next/standalone/.next/server/app/index.rsc +15 -15
- package/.next/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +2 -2
- package/.next/standalone/.next/server/app/index.segments/_full.segment.rsc +15 -15
- package/.next/standalone/.next/server/app/index.segments/_head.segment.rsc +4 -4
- package/.next/standalone/.next/server/app/index.segments/_index.segment.rsc +10 -10
- package/.next/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
- package/.next/standalone/.next/server/app/page/server-reference-manifest.json +1 -1
- package/.next/standalone/.next/server/app/page.js.nft.json +1 -1
- package/.next/standalone/.next/server/app/page_client-reference-manifest.js +1 -1
- package/.next/standalone/.next/server/app/policies/page/server-reference-manifest.json +8 -8
- package/.next/standalone/.next/server/app/policies/page.js.nft.json +1 -1
- package/.next/standalone/.next/server/app/policies/page_client-reference-manifest.js +1 -1
- package/.next/standalone/.next/server/app/project/[name]/page/server-reference-manifest.json +1 -1
- package/.next/standalone/.next/server/app/project/[name]/page.js.nft.json +1 -1
- package/.next/standalone/.next/server/app/project/[name]/page_client-reference-manifest.js +1 -1
- package/.next/standalone/.next/server/app/project/[name]/session/[sessionId]/page/react-loadable-manifest.json +2 -2
- package/.next/standalone/.next/server/app/project/[name]/session/[sessionId]/page/server-reference-manifest.json +2 -2
- package/.next/standalone/.next/server/app/project/[name]/session/[sessionId]/page.js.nft.json +1 -1
- package/.next/standalone/.next/server/app/project/[name]/session/[sessionId]/page_client-reference-manifest.js +1 -1
- package/.next/standalone/.next/server/app/projects/page/server-reference-manifest.json +1 -1
- package/.next/standalone/.next/server/app/projects/page.js.nft.json +1 -1
- package/.next/standalone/.next/server/app/projects/page_client-reference-manifest.js +1 -1
- package/.next/standalone/.next/server/chunks/[root-of-the-server]__0g72weg._.js +1 -1
- package/.next/standalone/.next/server/chunks/package_json_[json]_cjs_0z7w.hh._.js +1 -1
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__092s1ta._.js +2 -2
- package/.next/standalone/.next/server/chunks/ssr/{[root-of-the-server]__0~kmh8w._.js → [root-of-the-server]__096k.db._.js} +2 -2
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__09icjsf._.js +2 -2
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0g.lg8b._.js +2 -2
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0h..k-e._.js +2 -2
- package/.next/standalone/.next/server/chunks/ssr/{[root-of-the-server]__0rh.18_._.js → [root-of-the-server]__0kyh86x._.js} +2 -2
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0okos0k._.js +2 -2
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0w6l33k._.js +2 -2
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__11pa2ra._.js +2 -2
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__12t-wym._.js +2 -2
- package/.next/standalone/.next/server/chunks/ssr/_10lm7or._.js +2 -2
- package/.next/standalone/.next/server/chunks/ssr/app_global-error_tsx_0xerkr6._.js +1 -1
- package/.next/standalone/.next/server/chunks/ssr/app_policies_hooks-client_tsx_0q-m0y-._.js +1 -1
- package/.next/standalone/.next/server/middleware-build-manifest.js +3 -3
- package/.next/standalone/.next/server/pages/404.html +2 -2
- package/.next/standalone/.next/server/pages/500.html +1 -1
- package/.next/standalone/.next/server/server-reference-manifest.js +1 -1
- package/.next/standalone/.next/server/server-reference-manifest.json +9 -9
- package/.next/standalone/.next/static/chunks/{0gbf4cphy8ksq.js → 0-dm_9a6nsc2l.js} +1 -1
- package/.next/standalone/.next/static/chunks/{12~yi9oj8av8p.js → 01pmw1-asbek~.js} +2 -2
- package/.next/standalone/.next/static/chunks/{0v.yd0kg_ld3r.js → 051m32nx~n5yr.js} +1 -1
- package/.next/standalone/.next/static/chunks/{09_k80d~cq2wg.js → 0a-yctdwn368y.js} +1 -1
- package/.next/standalone/.next/static/chunks/{0bvhsa6zva2o..js → 0ksdlt_1hucdm.js} +1 -1
- package/.next/standalone/.next/static/chunks/{01b~z8f1ws0rk.js → 0l-mu4okl-cj1.js} +1 -1
- package/.next/standalone/.next/static/chunks/{08t08igdql9yt.js → 0mazj-p-~2kc6.js} +1 -1
- package/.next/standalone/.next/static/chunks/0qakntsrpc~1j.js +6 -0
- package/.next/standalone/.next/static/chunks/{03rz6ykw-a2xi.js → 156zca6aewyr-.js} +1 -1
- package/.next/standalone/CHANGELOG.md +18 -0
- package/.next/standalone/bin/failproofai.mjs +91 -4
- package/.next/standalone/dist/cli.mjs +1156 -55
- package/.next/standalone/docs/ar/built-in-policies.mdx +140 -103
- package/.next/standalone/docs/ar/custom-policies.mdx +72 -72
- package/.next/standalone/docs/ar/examples.mdx +86 -33
- package/.next/standalone/docs/ar/getting-started.mdx +82 -29
- package/.next/standalone/docs/built-in-policies.mdx +3 -3
- package/.next/standalone/docs/de/built-in-policies.mdx +97 -60
- package/.next/standalone/docs/de/custom-policies.mdx +56 -56
- package/.next/standalone/docs/de/examples.mdx +72 -18
- package/.next/standalone/docs/de/getting-started.mdx +72 -20
- package/.next/standalone/docs/es/built-in-policies.mdx +91 -54
- package/.next/standalone/docs/es/custom-policies.mdx +55 -55
- package/.next/standalone/docs/es/examples.mdx +73 -19
- package/.next/standalone/docs/es/getting-started.mdx +72 -20
- package/.next/standalone/docs/fr/built-in-policies.mdx +99 -62
- package/.next/standalone/docs/fr/custom-policies.mdx +51 -51
- package/.next/standalone/docs/fr/examples.mdx +78 -24
- package/.next/standalone/docs/fr/getting-started.mdx +65 -13
- package/.next/standalone/docs/he/built-in-policies.mdx +139 -99
- package/.next/standalone/docs/he/custom-policies.mdx +75 -75
- package/.next/standalone/docs/he/examples.mdx +87 -33
- package/.next/standalone/docs/he/getting-started.mdx +84 -33
- package/.next/standalone/docs/hi/built-in-policies.mdx +203 -166
- package/.next/standalone/docs/hi/custom-policies.mdx +71 -70
- package/.next/standalone/docs/hi/examples.mdx +90 -36
- package/.next/standalone/docs/hi/getting-started.mdx +80 -27
- package/.next/standalone/docs/i18n/README.ar.md +69 -69
- package/.next/standalone/docs/i18n/README.de.md +46 -46
- package/.next/standalone/docs/i18n/README.es.md +42 -42
- package/.next/standalone/docs/i18n/README.fr.md +39 -39
- package/.next/standalone/docs/i18n/README.he.md +83 -83
- package/.next/standalone/docs/i18n/README.hi.md +69 -69
- package/.next/standalone/docs/i18n/README.it.md +72 -72
- package/.next/standalone/docs/i18n/README.ja.md +71 -71
- package/.next/standalone/docs/i18n/README.ko.md +52 -52
- package/.next/standalone/docs/i18n/README.pt-br.md +44 -44
- package/.next/standalone/docs/i18n/README.ru.md +66 -66
- package/.next/standalone/docs/i18n/README.tr.md +82 -83
- package/.next/standalone/docs/i18n/README.vi.md +70 -71
- package/.next/standalone/docs/i18n/README.zh.md +51 -51
- package/.next/standalone/docs/it/built-in-policies.mdx +115 -78
- package/.next/standalone/docs/it/custom-policies.mdx +69 -69
- package/.next/standalone/docs/it/examples.mdx +93 -39
- package/.next/standalone/docs/it/getting-started.mdx +73 -21
- package/.next/standalone/docs/ja/built-in-policies.mdx +155 -118
- package/.next/standalone/docs/ja/custom-policies.mdx +71 -71
- package/.next/standalone/docs/ja/examples.mdx +76 -22
- package/.next/standalone/docs/ja/getting-started.mdx +65 -13
- package/.next/standalone/docs/ko/built-in-policies.mdx +103 -66
- package/.next/standalone/docs/ko/custom-policies.mdx +67 -67
- package/.next/standalone/docs/ko/examples.mdx +87 -33
- package/.next/standalone/docs/ko/getting-started.mdx +61 -9
- package/.next/standalone/docs/pt-br/built-in-policies.mdx +72 -35
- package/.next/standalone/docs/pt-br/custom-policies.mdx +56 -56
- package/.next/standalone/docs/pt-br/examples.mdx +78 -24
- package/.next/standalone/docs/pt-br/getting-started.mdx +64 -12
- package/.next/standalone/docs/ru/built-in-policies.mdx +135 -98
- package/.next/standalone/docs/ru/custom-policies.mdx +82 -81
- package/.next/standalone/docs/ru/examples.mdx +77 -22
- package/.next/standalone/docs/ru/getting-started.mdx +74 -22
- package/.next/standalone/docs/tr/built-in-policies.mdx +126 -89
- package/.next/standalone/docs/tr/custom-policies.mdx +59 -60
- package/.next/standalone/docs/tr/examples.mdx +97 -42
- package/.next/standalone/docs/tr/getting-started.mdx +75 -23
- package/.next/standalone/docs/vi/built-in-policies.mdx +116 -81
- package/.next/standalone/docs/vi/custom-policies.mdx +68 -68
- package/.next/standalone/docs/vi/examples.mdx +93 -38
- package/.next/standalone/docs/vi/getting-started.mdx +74 -22
- package/.next/standalone/docs/zh/built-in-policies.mdx +117 -82
- package/.next/standalone/docs/zh/custom-policies.mdx +49 -49
- package/.next/standalone/docs/zh/examples.mdx +90 -36
- package/.next/standalone/docs/zh/getting-started.mdx +73 -21
- package/.next/standalone/package.json +1 -1
- package/.next/standalone/server.js +1 -1
- package/.next/standalone/src/auth/login.ts +104 -0
- package/.next/standalone/src/auth/logout.ts +50 -0
- package/.next/standalone/src/auth/token-store.ts +64 -0
- package/.next/standalone/src/hooks/builtin-policies.ts +27 -21
- package/.next/standalone/src/hooks/handler.ts +35 -15
- package/.next/standalone/src/relay/daemon.ts +362 -0
- package/.next/standalone/src/relay/pid.ts +76 -0
- package/.next/standalone/src/relay/queue.ts +225 -0
- package/bin/failproofai.mjs +91 -4
- package/dist/cli.mjs +1156 -55
- package/package.json +1 -1
- package/src/auth/login.ts +104 -0
- package/src/auth/logout.ts +50 -0
- package/src/auth/token-store.ts +64 -0
- package/src/hooks/builtin-policies.ts +27 -21
- package/src/hooks/handler.ts +35 -15
- package/src/relay/daemon.ts +362 -0
- package/src/relay/pid.ts +76 -0
- package/src/relay/queue.ts +225 -0
- package/.next/standalone/.next/static/chunks/0wlyoif4_kj_t.js +0 -6
- /package/.next/standalone/.next/static/{CkmOT-ZvDN-sVULinGVKT → r-wX0MuAfCjbhJm3phQc8}/_buildManifest.js +0 -0
- /package/.next/standalone/.next/static/{CkmOT-ZvDN-sVULinGVKT → r-wX0MuAfCjbhJm3phQc8}/_clientMiddlewareManifest.js +0 -0
- /package/.next/standalone/.next/static/{CkmOT-ZvDN-sVULinGVKT → r-wX0MuAfCjbhJm3phQc8}/_ssgManifest.js +0 -0
|
@@ -148,26 +148,46 @@ export async function handleHookEvent(eventType: string): Promise<number> {
|
|
|
148
148
|
}
|
|
149
149
|
|
|
150
150
|
// Persist activity to disk (visible in /policies activity tab)
|
|
151
|
+
const activityEntry = {
|
|
152
|
+
timestamp: Date.now(),
|
|
153
|
+
eventType,
|
|
154
|
+
toolName: (parsed.tool_name as string) ?? null,
|
|
155
|
+
policyName: result.policyName,
|
|
156
|
+
policyNames: result.policyNames,
|
|
157
|
+
decision: result.decision,
|
|
158
|
+
reason: result.reason,
|
|
159
|
+
durationMs,
|
|
160
|
+
sessionId: session.sessionId,
|
|
161
|
+
transcriptPath: session.transcriptPath,
|
|
162
|
+
cwd: session.cwd,
|
|
163
|
+
permissionMode: session.permissionMode,
|
|
164
|
+
hookEventName: session.hookEventName,
|
|
165
|
+
};
|
|
151
166
|
try {
|
|
152
|
-
persistHookActivity(
|
|
153
|
-
timestamp: Date.now(),
|
|
154
|
-
eventType,
|
|
155
|
-
toolName: (parsed.tool_name as string) ?? null,
|
|
156
|
-
policyName: result.policyName,
|
|
157
|
-
policyNames: result.policyNames,
|
|
158
|
-
decision: result.decision,
|
|
159
|
-
reason: result.reason,
|
|
160
|
-
durationMs,
|
|
161
|
-
sessionId: session.sessionId,
|
|
162
|
-
transcriptPath: session.transcriptPath,
|
|
163
|
-
cwd: session.cwd,
|
|
164
|
-
permissionMode: session.permissionMode,
|
|
165
|
-
hookEventName: session.hookEventName,
|
|
166
|
-
});
|
|
167
|
+
persistHookActivity(activityEntry);
|
|
167
168
|
} catch {
|
|
168
169
|
hookLogWarn("activity persistence failed");
|
|
169
170
|
}
|
|
170
171
|
|
|
172
|
+
// Enqueue for server relay — fire-and-forget, never blocks hook.
|
|
173
|
+
// queue.ts is a no-op if the user is not logged in (no auth.json), and
|
|
174
|
+
// sanitizes the entry before persisting (drops toolInput/transcriptPath,
|
|
175
|
+
// hashes cwd, redacts known secret patterns in `reason`).
|
|
176
|
+
try {
|
|
177
|
+
const { appendToServerQueue } = await import("../relay/queue");
|
|
178
|
+
appendToServerQueue(activityEntry);
|
|
179
|
+
} catch {
|
|
180
|
+
// Server queue is best-effort; fail-open
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
// Lazy-start relay daemon if user is logged in — ~1ms when already running
|
|
184
|
+
try {
|
|
185
|
+
const { ensureRelayRunning } = await import("../relay/daemon");
|
|
186
|
+
ensureRelayRunning();
|
|
187
|
+
} catch {
|
|
188
|
+
// Relay is best-effort; hook must succeed regardless
|
|
189
|
+
}
|
|
190
|
+
|
|
171
191
|
// Fire PostHog telemetry for decisions that affect Claude's behavior
|
|
172
192
|
if (result.decision === "deny" || result.decision === "instruct") {
|
|
173
193
|
try {
|
|
@@ -0,0 +1,362 @@
|
|
|
1
|
+
import { spawn } from "node:child_process";
|
|
2
|
+
import { existsSync } from "node:fs";
|
|
3
|
+
import { join } from "node:path";
|
|
4
|
+
import { homedir } from "node:os";
|
|
5
|
+
import { randomUUID } from "node:crypto";
|
|
6
|
+
import { readTokens, writeTokens, isLoggedIn } from "../auth/token-store";
|
|
7
|
+
import { readPid, writePid, clearPid, isProcessAlive } from "./pid";
|
|
8
|
+
import {
|
|
9
|
+
claimPendingBatch,
|
|
10
|
+
readProcessingFile,
|
|
11
|
+
deleteProcessingFile,
|
|
12
|
+
findOrphanProcessingFiles,
|
|
13
|
+
type QueueEntry,
|
|
14
|
+
} from "./queue";
|
|
15
|
+
|
|
16
|
+
const QUEUE_DIR = join(homedir(), ".failproofai", "cache", "server-queue");
|
|
17
|
+
const BATCH_SIZE = 100;
|
|
18
|
+
const FLUSH_INTERVAL_MS = 2000;
|
|
19
|
+
const RECONNECT_BASE_MS = 1000;
|
|
20
|
+
const RECONNECT_MAX_MS = 60_000;
|
|
21
|
+
const HTTP_TIMEOUT_MS = 10_000;
|
|
22
|
+
const WS_CONNECT_TIMEOUT_MS = 15_000;
|
|
23
|
+
const ACK_TIMEOUT_MS = 30_000;
|
|
24
|
+
|
|
25
|
+
/**
|
|
26
|
+
* Lazy-start check: call on every hook invocation. Near-zero cost when daemon
|
|
27
|
+
* is already running (~1ms PID check); spawns daemon once after reboots.
|
|
28
|
+
*/
|
|
29
|
+
export function ensureRelayRunning(): void {
|
|
30
|
+
if (!isLoggedIn()) return;
|
|
31
|
+
|
|
32
|
+
const pid = readPid();
|
|
33
|
+
if (pid !== null && isProcessAlive(pid)) return;
|
|
34
|
+
|
|
35
|
+
if (pid !== null) clearPid();
|
|
36
|
+
spawnDaemon();
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
function spawnDaemon(): void {
|
|
40
|
+
const entrypoint = process.env.FAILPROOFAI_RELAY_ENTRYPOINT ?? process.argv[1];
|
|
41
|
+
if (!entrypoint) return;
|
|
42
|
+
|
|
43
|
+
const child = spawn(process.execPath, [entrypoint, "--relay-daemon"], {
|
|
44
|
+
detached: true,
|
|
45
|
+
stdio: "ignore",
|
|
46
|
+
env: { ...process.env, FAILPROOFAI_DAEMON: "1" },
|
|
47
|
+
});
|
|
48
|
+
child.unref();
|
|
49
|
+
|
|
50
|
+
if (typeof child.pid === "number") {
|
|
51
|
+
writePid(child.pid);
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
/**
|
|
56
|
+
* Block until the spawned daemon has been observed running, or until the
|
|
57
|
+
* timeout elapses. Used by `relay start` so we don't falsely report
|
|
58
|
+
* "Failed to start daemon" in the split-second window before the child
|
|
59
|
+
* has finished exec-ing.
|
|
60
|
+
*/
|
|
61
|
+
export async function waitForRelayAlive(timeoutMs = 2_000): Promise<boolean> {
|
|
62
|
+
const deadline = Date.now() + timeoutMs;
|
|
63
|
+
while (Date.now() < deadline) {
|
|
64
|
+
const pid = readPid();
|
|
65
|
+
if (pid !== null && isProcessAlive(pid)) return true;
|
|
66
|
+
await new Promise((r) => setTimeout(r, 50));
|
|
67
|
+
}
|
|
68
|
+
return false;
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
async function refreshTokenIfNeeded(): Promise<string | null> {
|
|
72
|
+
const tokens = readTokens();
|
|
73
|
+
if (!tokens) return null;
|
|
74
|
+
|
|
75
|
+
const nowSec = Math.floor(Date.now() / 1000);
|
|
76
|
+
if (tokens.expires_at - nowSec > 300) {
|
|
77
|
+
return tokens.access_token;
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
try {
|
|
81
|
+
const resp = await fetch(`${tokens.server_url}/api/v1/auth/refresh`, {
|
|
82
|
+
method: "POST",
|
|
83
|
+
headers: { "Content-Type": "application/json" },
|
|
84
|
+
body: JSON.stringify({ refresh_token: tokens.refresh_token }),
|
|
85
|
+
signal: AbortSignal.timeout(HTTP_TIMEOUT_MS),
|
|
86
|
+
});
|
|
87
|
+
if (!resp.ok) return tokens.access_token;
|
|
88
|
+
const refreshed = (await resp.json()) as {
|
|
89
|
+
access_token: string;
|
|
90
|
+
refresh_token: string;
|
|
91
|
+
expires_in: number;
|
|
92
|
+
};
|
|
93
|
+
writeTokens({
|
|
94
|
+
...tokens,
|
|
95
|
+
access_token: refreshed.access_token,
|
|
96
|
+
refresh_token: refreshed.refresh_token,
|
|
97
|
+
expires_at: nowSec + refreshed.expires_in,
|
|
98
|
+
});
|
|
99
|
+
return refreshed.access_token;
|
|
100
|
+
} catch {
|
|
101
|
+
return tokens.access_token;
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
type WebSocketLike = {
|
|
106
|
+
send(data: string): void;
|
|
107
|
+
close(): void;
|
|
108
|
+
readyState: number;
|
|
109
|
+
onopen: (() => void) | null;
|
|
110
|
+
onmessage: ((ev: { data: string }) => void) | null;
|
|
111
|
+
onerror: ((ev: unknown) => void) | null;
|
|
112
|
+
onclose: (() => void) | null;
|
|
113
|
+
};
|
|
114
|
+
|
|
115
|
+
class Relay {
|
|
116
|
+
private readonly ws: WebSocketLike;
|
|
117
|
+
private readonly pendingAcks = new Map<string, (ok: boolean) => void>();
|
|
118
|
+
private closed = false;
|
|
119
|
+
|
|
120
|
+
constructor(ws: WebSocketLike) {
|
|
121
|
+
this.ws = ws;
|
|
122
|
+
ws.onmessage = (ev) => this.handleMessage(ev.data);
|
|
123
|
+
ws.onclose = () => this.handleClose();
|
|
124
|
+
ws.onerror = () => this.handleClose();
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
private handleMessage(data: string): void {
|
|
128
|
+
try {
|
|
129
|
+
const msg = JSON.parse(data) as { ack?: string; error?: string };
|
|
130
|
+
if (msg.ack && this.pendingAcks.has(msg.ack)) {
|
|
131
|
+
const resolve = this.pendingAcks.get(msg.ack)!;
|
|
132
|
+
this.pendingAcks.delete(msg.ack);
|
|
133
|
+
resolve(true);
|
|
134
|
+
}
|
|
135
|
+
} catch {
|
|
136
|
+
// Ignore unparseable server messages
|
|
137
|
+
}
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
private handleClose(): void {
|
|
141
|
+
this.closed = true;
|
|
142
|
+
// Reject all outstanding acks so callers can retry
|
|
143
|
+
for (const [, resolve] of this.pendingAcks) {
|
|
144
|
+
resolve(false);
|
|
145
|
+
}
|
|
146
|
+
this.pendingAcks.clear();
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
isClosed(): boolean {
|
|
150
|
+
return this.closed;
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
close(): void {
|
|
154
|
+
try {
|
|
155
|
+
this.ws.close();
|
|
156
|
+
} catch {
|
|
157
|
+
// ignore
|
|
158
|
+
}
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
/**
|
|
162
|
+
* Send a batch and wait for the server's ack (keyed on batch_id).
|
|
163
|
+
* Returns true only when the server confirms the insert.
|
|
164
|
+
*/
|
|
165
|
+
async sendBatchAndWaitAck(events: QueueEntry[]): Promise<boolean> {
|
|
166
|
+
if (this.closed) return false;
|
|
167
|
+
const batchId = randomUUID();
|
|
168
|
+
|
|
169
|
+
const ackPromise = new Promise<boolean>((resolve) => {
|
|
170
|
+
this.pendingAcks.set(batchId, resolve);
|
|
171
|
+
setTimeout(() => {
|
|
172
|
+
if (this.pendingAcks.delete(batchId)) resolve(false);
|
|
173
|
+
}, ACK_TIMEOUT_MS);
|
|
174
|
+
});
|
|
175
|
+
|
|
176
|
+
try {
|
|
177
|
+
this.ws.send(JSON.stringify({ batch_id: batchId, events }));
|
|
178
|
+
} catch {
|
|
179
|
+
this.pendingAcks.delete(batchId);
|
|
180
|
+
return false;
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
return ackPromise;
|
|
184
|
+
}
|
|
185
|
+
}
|
|
186
|
+
|
|
187
|
+
async function connect(wsUrl: string, token: string): Promise<WebSocketLike> {
|
|
188
|
+
const WSCtor: any = (globalThis as any).WebSocket;
|
|
189
|
+
if (!WSCtor) {
|
|
190
|
+
throw new Error("WebSocket not available in this Node version. Requires Node 22+.");
|
|
191
|
+
}
|
|
192
|
+
const ws: WebSocketLike = new WSCtor(wsUrl);
|
|
193
|
+
|
|
194
|
+
await new Promise<void>((resolve, reject) => {
|
|
195
|
+
let settled = false;
|
|
196
|
+
const timeout = setTimeout(() => {
|
|
197
|
+
if (settled) return;
|
|
198
|
+
settled = true;
|
|
199
|
+
try {
|
|
200
|
+
ws.close();
|
|
201
|
+
} catch {
|
|
202
|
+
// ignore
|
|
203
|
+
}
|
|
204
|
+
reject(new Error("WebSocket connect timeout"));
|
|
205
|
+
}, WS_CONNECT_TIMEOUT_MS);
|
|
206
|
+
|
|
207
|
+
ws.onopen = () => {
|
|
208
|
+
if (settled) return;
|
|
209
|
+
settled = true;
|
|
210
|
+
clearTimeout(timeout);
|
|
211
|
+
try {
|
|
212
|
+
ws.send(token);
|
|
213
|
+
resolve();
|
|
214
|
+
} catch (e) {
|
|
215
|
+
reject(e);
|
|
216
|
+
}
|
|
217
|
+
};
|
|
218
|
+
ws.onerror = (e) => {
|
|
219
|
+
if (settled) return;
|
|
220
|
+
settled = true;
|
|
221
|
+
clearTimeout(timeout);
|
|
222
|
+
reject(e);
|
|
223
|
+
};
|
|
224
|
+
ws.onclose = () => {
|
|
225
|
+
if (settled) return;
|
|
226
|
+
settled = true;
|
|
227
|
+
clearTimeout(timeout);
|
|
228
|
+
reject(new Error("WebSocket closed before opening"));
|
|
229
|
+
};
|
|
230
|
+
});
|
|
231
|
+
|
|
232
|
+
return ws;
|
|
233
|
+
}
|
|
234
|
+
|
|
235
|
+
/**
|
|
236
|
+
* Send all events from a processing file and wait for server acks on every
|
|
237
|
+
* batch. Returns true only when every batch was acknowledged — in that
|
|
238
|
+
* case the caller may delete the processing file.
|
|
239
|
+
*/
|
|
240
|
+
async function sendProcessingFile(relay: Relay, path: string): Promise<boolean> {
|
|
241
|
+
const events = readProcessingFile(path);
|
|
242
|
+
if (events.length === 0) return true;
|
|
243
|
+
|
|
244
|
+
for (let i = 0; i < events.length; i += BATCH_SIZE) {
|
|
245
|
+
const batch = events.slice(i, i + BATCH_SIZE);
|
|
246
|
+
const ok = await relay.sendBatchAndWaitAck(batch);
|
|
247
|
+
if (!ok) return false;
|
|
248
|
+
}
|
|
249
|
+
return true;
|
|
250
|
+
}
|
|
251
|
+
|
|
252
|
+
export async function runDaemon(): Promise<void> {
|
|
253
|
+
let reconnectDelay = RECONNECT_BASE_MS;
|
|
254
|
+
|
|
255
|
+
while (true) {
|
|
256
|
+
const token = await refreshTokenIfNeeded();
|
|
257
|
+
const tokens = readTokens();
|
|
258
|
+
if (!token || !tokens) {
|
|
259
|
+
await new Promise((r) => setTimeout(r, 30_000));
|
|
260
|
+
continue;
|
|
261
|
+
}
|
|
262
|
+
|
|
263
|
+
const wsUrl = `${tokens.server_url.replace(/^http/, "ws")}/ws/events/ingest`;
|
|
264
|
+
|
|
265
|
+
try {
|
|
266
|
+
const ws = await connect(wsUrl, token);
|
|
267
|
+
const relay = new Relay(ws);
|
|
268
|
+
reconnectDelay = RECONNECT_BASE_MS;
|
|
269
|
+
|
|
270
|
+
// Drain any orphaned processing files from a prior crash first
|
|
271
|
+
for (const orphan of findOrphanProcessingFiles()) {
|
|
272
|
+
if (relay.isClosed()) break;
|
|
273
|
+
const ok = await sendProcessingFile(relay, orphan);
|
|
274
|
+
if (ok) deleteProcessingFile(orphan);
|
|
275
|
+
}
|
|
276
|
+
|
|
277
|
+
while (!relay.isClosed()) {
|
|
278
|
+
let processingFile: string | null = null;
|
|
279
|
+
try {
|
|
280
|
+
processingFile = claimPendingBatch();
|
|
281
|
+
} catch {
|
|
282
|
+
// Transient FS error — retry on next tick
|
|
283
|
+
}
|
|
284
|
+
|
|
285
|
+
if (processingFile) {
|
|
286
|
+
const ok = await sendProcessingFile(relay, processingFile);
|
|
287
|
+
if (ok) {
|
|
288
|
+
deleteProcessingFile(processingFile);
|
|
289
|
+
} else {
|
|
290
|
+
// Ack failed or connection dropped — leave file for retry
|
|
291
|
+
break;
|
|
292
|
+
}
|
|
293
|
+
}
|
|
294
|
+
await new Promise((r) => setTimeout(r, FLUSH_INTERVAL_MS));
|
|
295
|
+
}
|
|
296
|
+
|
|
297
|
+
relay.close();
|
|
298
|
+
} catch {
|
|
299
|
+
// Connection failed — wait and retry with backoff
|
|
300
|
+
}
|
|
301
|
+
|
|
302
|
+
if (existsSync(QUEUE_DIR)) {
|
|
303
|
+
// noop; QUEUE_DIR referenced to preserve import when tree-shaking
|
|
304
|
+
}
|
|
305
|
+
|
|
306
|
+
await new Promise((r) => setTimeout(r, reconnectDelay));
|
|
307
|
+
reconnectDelay = Math.min(reconnectDelay * 2, RECONNECT_MAX_MS);
|
|
308
|
+
}
|
|
309
|
+
}
|
|
310
|
+
|
|
311
|
+
/**
|
|
312
|
+
* One-shot: POST all pending events to the server via REST batch endpoint.
|
|
313
|
+
* Used by `failproofai sync` — same rotate-then-delete pattern, but with
|
|
314
|
+
* HTTP response status as the ack mechanism.
|
|
315
|
+
*/
|
|
316
|
+
export async function runOneShotSync(): Promise<number> {
|
|
317
|
+
const token = await refreshTokenIfNeeded();
|
|
318
|
+
const tokens = readTokens();
|
|
319
|
+
if (!token || !tokens) {
|
|
320
|
+
throw new Error("Not logged in. Run `failproofai login` first.");
|
|
321
|
+
}
|
|
322
|
+
|
|
323
|
+
let total = 0;
|
|
324
|
+
|
|
325
|
+
async function postBatch(events: QueueEntry[]): Promise<void> {
|
|
326
|
+
const resp = await fetch(`${tokens!.server_url}/api/v1/events/batch`, {
|
|
327
|
+
method: "POST",
|
|
328
|
+
headers: {
|
|
329
|
+
"Content-Type": "application/json",
|
|
330
|
+
Authorization: `Bearer ${token}`,
|
|
331
|
+
},
|
|
332
|
+
body: JSON.stringify({ events }),
|
|
333
|
+
signal: AbortSignal.timeout(HTTP_TIMEOUT_MS),
|
|
334
|
+
});
|
|
335
|
+
if (!resp.ok) {
|
|
336
|
+
throw new Error(`Sync failed: ${resp.status} ${resp.statusText}`);
|
|
337
|
+
}
|
|
338
|
+
}
|
|
339
|
+
|
|
340
|
+
// Drain orphans first
|
|
341
|
+
for (const orphan of findOrphanProcessingFiles()) {
|
|
342
|
+
const events = readProcessingFile(orphan);
|
|
343
|
+
if (events.length > 0) {
|
|
344
|
+
await postBatch(events);
|
|
345
|
+
total += events.length;
|
|
346
|
+
}
|
|
347
|
+
deleteProcessingFile(orphan);
|
|
348
|
+
}
|
|
349
|
+
|
|
350
|
+
// Drain fresh pending batch
|
|
351
|
+
const processingFile = claimPendingBatch();
|
|
352
|
+
if (processingFile) {
|
|
353
|
+
const events = readProcessingFile(processingFile);
|
|
354
|
+
if (events.length > 0) {
|
|
355
|
+
await postBatch(events);
|
|
356
|
+
total += events.length;
|
|
357
|
+
}
|
|
358
|
+
deleteProcessingFile(processingFile);
|
|
359
|
+
}
|
|
360
|
+
|
|
361
|
+
return total;
|
|
362
|
+
}
|
|
@@ -0,0 +1,76 @@
|
|
|
1
|
+
import { readFileSync, writeFileSync, existsSync, unlinkSync, mkdirSync } from "node:fs";
|
|
2
|
+
import { join, dirname } from "node:path";
|
|
3
|
+
import { homedir } from "node:os";
|
|
4
|
+
|
|
5
|
+
const PID_FILE = join(homedir(), ".failproofai", "relay.pid");
|
|
6
|
+
|
|
7
|
+
export function readPid(): number | null {
|
|
8
|
+
if (!existsSync(PID_FILE)) return null;
|
|
9
|
+
try {
|
|
10
|
+
const raw = readFileSync(PID_FILE, "utf8").trim();
|
|
11
|
+
const pid = parseInt(raw, 10);
|
|
12
|
+
if (Number.isNaN(pid) || pid <= 0) return null;
|
|
13
|
+
return pid;
|
|
14
|
+
} catch {
|
|
15
|
+
return null;
|
|
16
|
+
}
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
export function writePid(pid: number): void {
|
|
20
|
+
const dir = dirname(PID_FILE);
|
|
21
|
+
if (!existsSync(dir)) mkdirSync(dir, { recursive: true, mode: 0o700 });
|
|
22
|
+
writeFileSync(PID_FILE, String(pid));
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
export function clearPid(): void {
|
|
26
|
+
if (existsSync(PID_FILE)) unlinkSync(PID_FILE);
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
interface ErrnoError extends Error {
|
|
30
|
+
code?: string;
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
/**
|
|
34
|
+
* `process.kill(pid, 0)` sends signal 0 as an existence probe.
|
|
35
|
+
*
|
|
36
|
+
* no throw → PID exists and we can signal it
|
|
37
|
+
* ESRCH → PID doesn't exist (process is gone)
|
|
38
|
+
* EPERM → PID exists but belongs to a different user — still ALIVE,
|
|
39
|
+
* just unsignalable by us. Treating this as "dead" would cause
|
|
40
|
+
* us to clear the PID file and spawn a second daemon while
|
|
41
|
+
* the first keeps running.
|
|
42
|
+
*/
|
|
43
|
+
export function isProcessAlive(pid: number): boolean {
|
|
44
|
+
try {
|
|
45
|
+
process.kill(pid, 0);
|
|
46
|
+
return true;
|
|
47
|
+
} catch (err) {
|
|
48
|
+
const e = err as ErrnoError;
|
|
49
|
+
// EPERM means the process exists but we can't signal it — still alive
|
|
50
|
+
if (e?.code === "EPERM") return true;
|
|
51
|
+
// ESRCH or anything else → treat as dead
|
|
52
|
+
return false;
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
export function stopRelay(): boolean {
|
|
57
|
+
const pid = readPid();
|
|
58
|
+
if (pid === null) return false;
|
|
59
|
+
if (!isProcessAlive(pid)) {
|
|
60
|
+
clearPid();
|
|
61
|
+
return false;
|
|
62
|
+
}
|
|
63
|
+
try {
|
|
64
|
+
process.kill(pid, "SIGTERM");
|
|
65
|
+
clearPid();
|
|
66
|
+
return true;
|
|
67
|
+
} catch {
|
|
68
|
+
return false;
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
export function relayStatus(): { running: boolean; pid: number | null } {
|
|
73
|
+
const pid = readPid();
|
|
74
|
+
if (pid === null) return { running: false, pid: null };
|
|
75
|
+
return { running: isProcessAlive(pid), pid };
|
|
76
|
+
}
|