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.
Files changed (177) hide show
  1. package/.next/standalone/.failproofai/policies/review-policies.mjs +4 -3
  2. package/.next/standalone/.next/BUILD_ID +1 -1
  3. package/.next/standalone/.next/build-manifest.json +3 -3
  4. package/.next/standalone/.next/prerender-manifest.json +3 -3
  5. package/.next/standalone/.next/required-server-files.json +1 -1
  6. package/.next/standalone/.next/server/app/_global-error/page/server-reference-manifest.json +1 -1
  7. package/.next/standalone/.next/server/app/_global-error/page.js.nft.json +1 -1
  8. package/.next/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
  9. package/.next/standalone/.next/server/app/_global-error.html +1 -1
  10. package/.next/standalone/.next/server/app/_global-error.rsc +7 -7
  11. package/.next/standalone/.next/server/app/_global-error.segments/__PAGE__.segment.rsc +2 -2
  12. package/.next/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +7 -7
  13. package/.next/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +3 -3
  14. package/.next/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +3 -3
  15. package/.next/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  16. package/.next/standalone/.next/server/app/_not-found/page/server-reference-manifest.json +1 -1
  17. package/.next/standalone/.next/server/app/_not-found/page.js.nft.json +1 -1
  18. package/.next/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  19. package/.next/standalone/.next/server/app/_not-found.html +2 -2
  20. package/.next/standalone/.next/server/app/_not-found.rsc +15 -15
  21. package/.next/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +15 -15
  22. package/.next/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +4 -4
  23. package/.next/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +10 -10
  24. package/.next/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +2 -2
  25. package/.next/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +3 -3
  26. package/.next/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  27. package/.next/standalone/.next/server/app/index.html +1 -1
  28. package/.next/standalone/.next/server/app/index.rsc +15 -15
  29. package/.next/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +2 -2
  30. package/.next/standalone/.next/server/app/index.segments/_full.segment.rsc +15 -15
  31. package/.next/standalone/.next/server/app/index.segments/_head.segment.rsc +4 -4
  32. package/.next/standalone/.next/server/app/index.segments/_index.segment.rsc +10 -10
  33. package/.next/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  34. package/.next/standalone/.next/server/app/page/server-reference-manifest.json +1 -1
  35. package/.next/standalone/.next/server/app/page.js.nft.json +1 -1
  36. package/.next/standalone/.next/server/app/page_client-reference-manifest.js +1 -1
  37. package/.next/standalone/.next/server/app/policies/page/server-reference-manifest.json +8 -8
  38. package/.next/standalone/.next/server/app/policies/page.js.nft.json +1 -1
  39. package/.next/standalone/.next/server/app/policies/page_client-reference-manifest.js +1 -1
  40. package/.next/standalone/.next/server/app/project/[name]/page/server-reference-manifest.json +1 -1
  41. package/.next/standalone/.next/server/app/project/[name]/page.js.nft.json +1 -1
  42. package/.next/standalone/.next/server/app/project/[name]/page_client-reference-manifest.js +1 -1
  43. package/.next/standalone/.next/server/app/project/[name]/session/[sessionId]/page/react-loadable-manifest.json +2 -2
  44. package/.next/standalone/.next/server/app/project/[name]/session/[sessionId]/page/server-reference-manifest.json +2 -2
  45. package/.next/standalone/.next/server/app/project/[name]/session/[sessionId]/page.js.nft.json +1 -1
  46. package/.next/standalone/.next/server/app/project/[name]/session/[sessionId]/page_client-reference-manifest.js +1 -1
  47. package/.next/standalone/.next/server/app/projects/page/server-reference-manifest.json +1 -1
  48. package/.next/standalone/.next/server/app/projects/page.js.nft.json +1 -1
  49. package/.next/standalone/.next/server/app/projects/page_client-reference-manifest.js +1 -1
  50. package/.next/standalone/.next/server/chunks/[root-of-the-server]__0g72weg._.js +1 -1
  51. package/.next/standalone/.next/server/chunks/package_json_[json]_cjs_0z7w.hh._.js +1 -1
  52. package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__092s1ta._.js +2 -2
  53. package/.next/standalone/.next/server/chunks/ssr/{[root-of-the-server]__0~kmh8w._.js → [root-of-the-server]__096k.db._.js} +2 -2
  54. package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__09icjsf._.js +2 -2
  55. package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0g.lg8b._.js +2 -2
  56. package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0h..k-e._.js +2 -2
  57. package/.next/standalone/.next/server/chunks/ssr/{[root-of-the-server]__0rh.18_._.js → [root-of-the-server]__0kyh86x._.js} +2 -2
  58. package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0okos0k._.js +2 -2
  59. package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0w6l33k._.js +2 -2
  60. package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__11pa2ra._.js +2 -2
  61. package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__12t-wym._.js +2 -2
  62. package/.next/standalone/.next/server/chunks/ssr/_10lm7or._.js +2 -2
  63. package/.next/standalone/.next/server/chunks/ssr/app_global-error_tsx_0xerkr6._.js +1 -1
  64. package/.next/standalone/.next/server/chunks/ssr/app_policies_hooks-client_tsx_0q-m0y-._.js +1 -1
  65. package/.next/standalone/.next/server/middleware-build-manifest.js +3 -3
  66. package/.next/standalone/.next/server/pages/404.html +2 -2
  67. package/.next/standalone/.next/server/pages/500.html +1 -1
  68. package/.next/standalone/.next/server/server-reference-manifest.js +1 -1
  69. package/.next/standalone/.next/server/server-reference-manifest.json +9 -9
  70. package/.next/standalone/.next/static/chunks/{0gbf4cphy8ksq.js → 0-dm_9a6nsc2l.js} +1 -1
  71. package/.next/standalone/.next/static/chunks/{12~yi9oj8av8p.js → 01pmw1-asbek~.js} +2 -2
  72. package/.next/standalone/.next/static/chunks/{0v.yd0kg_ld3r.js → 051m32nx~n5yr.js} +1 -1
  73. package/.next/standalone/.next/static/chunks/{09_k80d~cq2wg.js → 0a-yctdwn368y.js} +1 -1
  74. package/.next/standalone/.next/static/chunks/{0bvhsa6zva2o..js → 0ksdlt_1hucdm.js} +1 -1
  75. package/.next/standalone/.next/static/chunks/{01b~z8f1ws0rk.js → 0l-mu4okl-cj1.js} +1 -1
  76. package/.next/standalone/.next/static/chunks/{08t08igdql9yt.js → 0mazj-p-~2kc6.js} +1 -1
  77. package/.next/standalone/.next/static/chunks/0qakntsrpc~1j.js +6 -0
  78. package/.next/standalone/.next/static/chunks/{03rz6ykw-a2xi.js → 156zca6aewyr-.js} +1 -1
  79. package/.next/standalone/CHANGELOG.md +18 -0
  80. package/.next/standalone/bin/failproofai.mjs +91 -4
  81. package/.next/standalone/dist/cli.mjs +1156 -55
  82. package/.next/standalone/docs/ar/built-in-policies.mdx +140 -103
  83. package/.next/standalone/docs/ar/custom-policies.mdx +72 -72
  84. package/.next/standalone/docs/ar/examples.mdx +86 -33
  85. package/.next/standalone/docs/ar/getting-started.mdx +82 -29
  86. package/.next/standalone/docs/built-in-policies.mdx +3 -3
  87. package/.next/standalone/docs/de/built-in-policies.mdx +97 -60
  88. package/.next/standalone/docs/de/custom-policies.mdx +56 -56
  89. package/.next/standalone/docs/de/examples.mdx +72 -18
  90. package/.next/standalone/docs/de/getting-started.mdx +72 -20
  91. package/.next/standalone/docs/es/built-in-policies.mdx +91 -54
  92. package/.next/standalone/docs/es/custom-policies.mdx +55 -55
  93. package/.next/standalone/docs/es/examples.mdx +73 -19
  94. package/.next/standalone/docs/es/getting-started.mdx +72 -20
  95. package/.next/standalone/docs/fr/built-in-policies.mdx +99 -62
  96. package/.next/standalone/docs/fr/custom-policies.mdx +51 -51
  97. package/.next/standalone/docs/fr/examples.mdx +78 -24
  98. package/.next/standalone/docs/fr/getting-started.mdx +65 -13
  99. package/.next/standalone/docs/he/built-in-policies.mdx +139 -99
  100. package/.next/standalone/docs/he/custom-policies.mdx +75 -75
  101. package/.next/standalone/docs/he/examples.mdx +87 -33
  102. package/.next/standalone/docs/he/getting-started.mdx +84 -33
  103. package/.next/standalone/docs/hi/built-in-policies.mdx +203 -166
  104. package/.next/standalone/docs/hi/custom-policies.mdx +71 -70
  105. package/.next/standalone/docs/hi/examples.mdx +90 -36
  106. package/.next/standalone/docs/hi/getting-started.mdx +80 -27
  107. package/.next/standalone/docs/i18n/README.ar.md +69 -69
  108. package/.next/standalone/docs/i18n/README.de.md +46 -46
  109. package/.next/standalone/docs/i18n/README.es.md +42 -42
  110. package/.next/standalone/docs/i18n/README.fr.md +39 -39
  111. package/.next/standalone/docs/i18n/README.he.md +83 -83
  112. package/.next/standalone/docs/i18n/README.hi.md +69 -69
  113. package/.next/standalone/docs/i18n/README.it.md +72 -72
  114. package/.next/standalone/docs/i18n/README.ja.md +71 -71
  115. package/.next/standalone/docs/i18n/README.ko.md +52 -52
  116. package/.next/standalone/docs/i18n/README.pt-br.md +44 -44
  117. package/.next/standalone/docs/i18n/README.ru.md +66 -66
  118. package/.next/standalone/docs/i18n/README.tr.md +82 -83
  119. package/.next/standalone/docs/i18n/README.vi.md +70 -71
  120. package/.next/standalone/docs/i18n/README.zh.md +51 -51
  121. package/.next/standalone/docs/it/built-in-policies.mdx +115 -78
  122. package/.next/standalone/docs/it/custom-policies.mdx +69 -69
  123. package/.next/standalone/docs/it/examples.mdx +93 -39
  124. package/.next/standalone/docs/it/getting-started.mdx +73 -21
  125. package/.next/standalone/docs/ja/built-in-policies.mdx +155 -118
  126. package/.next/standalone/docs/ja/custom-policies.mdx +71 -71
  127. package/.next/standalone/docs/ja/examples.mdx +76 -22
  128. package/.next/standalone/docs/ja/getting-started.mdx +65 -13
  129. package/.next/standalone/docs/ko/built-in-policies.mdx +103 -66
  130. package/.next/standalone/docs/ko/custom-policies.mdx +67 -67
  131. package/.next/standalone/docs/ko/examples.mdx +87 -33
  132. package/.next/standalone/docs/ko/getting-started.mdx +61 -9
  133. package/.next/standalone/docs/pt-br/built-in-policies.mdx +72 -35
  134. package/.next/standalone/docs/pt-br/custom-policies.mdx +56 -56
  135. package/.next/standalone/docs/pt-br/examples.mdx +78 -24
  136. package/.next/standalone/docs/pt-br/getting-started.mdx +64 -12
  137. package/.next/standalone/docs/ru/built-in-policies.mdx +135 -98
  138. package/.next/standalone/docs/ru/custom-policies.mdx +82 -81
  139. package/.next/standalone/docs/ru/examples.mdx +77 -22
  140. package/.next/standalone/docs/ru/getting-started.mdx +74 -22
  141. package/.next/standalone/docs/tr/built-in-policies.mdx +126 -89
  142. package/.next/standalone/docs/tr/custom-policies.mdx +59 -60
  143. package/.next/standalone/docs/tr/examples.mdx +97 -42
  144. package/.next/standalone/docs/tr/getting-started.mdx +75 -23
  145. package/.next/standalone/docs/vi/built-in-policies.mdx +116 -81
  146. package/.next/standalone/docs/vi/custom-policies.mdx +68 -68
  147. package/.next/standalone/docs/vi/examples.mdx +93 -38
  148. package/.next/standalone/docs/vi/getting-started.mdx +74 -22
  149. package/.next/standalone/docs/zh/built-in-policies.mdx +117 -82
  150. package/.next/standalone/docs/zh/custom-policies.mdx +49 -49
  151. package/.next/standalone/docs/zh/examples.mdx +90 -36
  152. package/.next/standalone/docs/zh/getting-started.mdx +73 -21
  153. package/.next/standalone/package.json +1 -1
  154. package/.next/standalone/server.js +1 -1
  155. package/.next/standalone/src/auth/login.ts +104 -0
  156. package/.next/standalone/src/auth/logout.ts +50 -0
  157. package/.next/standalone/src/auth/token-store.ts +64 -0
  158. package/.next/standalone/src/hooks/builtin-policies.ts +27 -21
  159. package/.next/standalone/src/hooks/handler.ts +35 -15
  160. package/.next/standalone/src/relay/daemon.ts +362 -0
  161. package/.next/standalone/src/relay/pid.ts +76 -0
  162. package/.next/standalone/src/relay/queue.ts +225 -0
  163. package/bin/failproofai.mjs +91 -4
  164. package/dist/cli.mjs +1156 -55
  165. package/package.json +1 -1
  166. package/src/auth/login.ts +104 -0
  167. package/src/auth/logout.ts +50 -0
  168. package/src/auth/token-store.ts +64 -0
  169. package/src/hooks/builtin-policies.ts +27 -21
  170. package/src/hooks/handler.ts +35 -15
  171. package/src/relay/daemon.ts +362 -0
  172. package/src/relay/pid.ts +76 -0
  173. package/src/relay/queue.ts +225 -0
  174. package/.next/standalone/.next/static/chunks/0wlyoif4_kj_t.js +0 -6
  175. /package/.next/standalone/.next/static/{CkmOT-ZvDN-sVULinGVKT → r-wX0MuAfCjbhJm3phQc8}/_buildManifest.js +0 -0
  176. /package/.next/standalone/.next/static/{CkmOT-ZvDN-sVULinGVKT → r-wX0MuAfCjbhJm3phQc8}/_clientMiddlewareManifest.js +0 -0
  177. /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
+ }