failproofai 0.0.6-beta.2 → 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 (119) hide show
  1. package/.next/standalone/.next/BUILD_ID +1 -1
  2. package/.next/standalone/.next/build-manifest.json +3 -3
  3. package/.next/standalone/.next/prerender-manifest.json +3 -3
  4. package/.next/standalone/.next/required-server-files.json +1 -1
  5. package/.next/standalone/.next/server/app/_global-error/page/server-reference-manifest.json +1 -1
  6. package/.next/standalone/.next/server/app/_global-error/page.js.nft.json +1 -1
  7. package/.next/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
  8. package/.next/standalone/.next/server/app/_global-error.html +1 -1
  9. package/.next/standalone/.next/server/app/_global-error.rsc +7 -7
  10. package/.next/standalone/.next/server/app/_global-error.segments/__PAGE__.segment.rsc +2 -2
  11. package/.next/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +7 -7
  12. package/.next/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +3 -3
  13. package/.next/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +3 -3
  14. package/.next/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  15. package/.next/standalone/.next/server/app/_not-found/page/server-reference-manifest.json +1 -1
  16. package/.next/standalone/.next/server/app/_not-found/page.js.nft.json +1 -1
  17. package/.next/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  18. package/.next/standalone/.next/server/app/_not-found.html +2 -2
  19. package/.next/standalone/.next/server/app/_not-found.rsc +15 -15
  20. package/.next/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +15 -15
  21. package/.next/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +4 -4
  22. package/.next/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +10 -10
  23. package/.next/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +2 -2
  24. package/.next/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +3 -3
  25. package/.next/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  26. package/.next/standalone/.next/server/app/index.html +1 -1
  27. package/.next/standalone/.next/server/app/index.rsc +15 -15
  28. package/.next/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +2 -2
  29. package/.next/standalone/.next/server/app/index.segments/_full.segment.rsc +15 -15
  30. package/.next/standalone/.next/server/app/index.segments/_head.segment.rsc +4 -4
  31. package/.next/standalone/.next/server/app/index.segments/_index.segment.rsc +10 -10
  32. package/.next/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  33. package/.next/standalone/.next/server/app/page/server-reference-manifest.json +1 -1
  34. package/.next/standalone/.next/server/app/page.js.nft.json +1 -1
  35. package/.next/standalone/.next/server/app/page_client-reference-manifest.js +1 -1
  36. package/.next/standalone/.next/server/app/policies/page/server-reference-manifest.json +8 -8
  37. package/.next/standalone/.next/server/app/policies/page.js.nft.json +1 -1
  38. package/.next/standalone/.next/server/app/policies/page_client-reference-manifest.js +1 -1
  39. package/.next/standalone/.next/server/app/project/[name]/page/server-reference-manifest.json +1 -1
  40. package/.next/standalone/.next/server/app/project/[name]/page.js.nft.json +1 -1
  41. package/.next/standalone/.next/server/app/project/[name]/page_client-reference-manifest.js +1 -1
  42. package/.next/standalone/.next/server/app/project/[name]/session/[sessionId]/page/react-loadable-manifest.json +2 -2
  43. package/.next/standalone/.next/server/app/project/[name]/session/[sessionId]/page/server-reference-manifest.json +2 -2
  44. package/.next/standalone/.next/server/app/project/[name]/session/[sessionId]/page.js.nft.json +1 -1
  45. package/.next/standalone/.next/server/app/project/[name]/session/[sessionId]/page_client-reference-manifest.js +1 -1
  46. package/.next/standalone/.next/server/app/projects/page/server-reference-manifest.json +1 -1
  47. package/.next/standalone/.next/server/app/projects/page.js.nft.json +1 -1
  48. package/.next/standalone/.next/server/app/projects/page_client-reference-manifest.js +1 -1
  49. package/.next/standalone/.next/server/chunks/[root-of-the-server]__0g72weg._.js +1 -1
  50. package/.next/standalone/.next/server/chunks/package_json_[json]_cjs_0z7w.hh._.js +1 -1
  51. package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__092s1ta._.js +2 -2
  52. package/.next/standalone/.next/server/chunks/ssr/{[root-of-the-server]__05akje6._.js → [root-of-the-server]__096k.db._.js} +2 -2
  53. package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__09icjsf._.js +2 -2
  54. package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0g.lg8b._.js +2 -2
  55. package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0h..k-e._.js +2 -2
  56. package/.next/standalone/.next/server/chunks/ssr/{[root-of-the-server]__0i5kvry._.js → [root-of-the-server]__0kyh86x._.js} +2 -2
  57. package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0okos0k._.js +2 -2
  58. package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0w6l33k._.js +2 -2
  59. package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__11pa2ra._.js +2 -2
  60. package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__12t-wym._.js +2 -2
  61. package/.next/standalone/.next/server/chunks/ssr/_10lm7or._.js +2 -2
  62. package/.next/standalone/.next/server/chunks/ssr/app_global-error_tsx_0xerkr6._.js +1 -1
  63. package/.next/standalone/.next/server/chunks/ssr/app_policies_hooks-client_tsx_0q-m0y-._.js +1 -1
  64. package/.next/standalone/.next/server/middleware-build-manifest.js +3 -3
  65. package/.next/standalone/.next/server/pages/404.html +2 -2
  66. package/.next/standalone/.next/server/pages/500.html +1 -1
  67. package/.next/standalone/.next/server/server-reference-manifest.js +1 -1
  68. package/.next/standalone/.next/server/server-reference-manifest.json +9 -9
  69. package/.next/standalone/.next/static/chunks/{1052sguyd-.ka.js → 0-dm_9a6nsc2l.js} +1 -1
  70. package/.next/standalone/.next/static/chunks/{05j1px0r8yzh6.js → 01pmw1-asbek~.js} +2 -2
  71. package/.next/standalone/.next/static/chunks/{14cl9poem30dq.js → 051m32nx~n5yr.js} +1 -1
  72. package/.next/standalone/.next/static/chunks/{0badv41uxa56..js → 0a-yctdwn368y.js} +1 -1
  73. package/.next/standalone/.next/static/chunks/{0xpl.oscrakvx.js → 0l-mu4okl-cj1.js} +1 -1
  74. package/.next/standalone/.next/static/chunks/{00j0rr7rh8ef8.js → 0mazj-p-~2kc6.js} +1 -1
  75. package/.next/standalone/.next/static/chunks/0qakntsrpc~1j.js +6 -0
  76. package/.next/standalone/.next/static/chunks/{0npb~873.wvg3.js → 156zca6aewyr-.js} +1 -1
  77. package/.next/standalone/CHANGELOG.md +7 -0
  78. package/.next/standalone/bin/failproofai.mjs +91 -4
  79. package/.next/standalone/dist/cli.mjs +1155 -54
  80. package/.next/standalone/docs/ar/built-in-policies.mdx +118 -118
  81. package/.next/standalone/docs/built-in-policies.mdx +2 -2
  82. package/.next/standalone/docs/de/built-in-policies.mdx +48 -48
  83. package/.next/standalone/docs/es/built-in-policies.mdx +82 -82
  84. package/.next/standalone/docs/fr/built-in-policies.mdx +72 -72
  85. package/.next/standalone/docs/he/built-in-policies.mdx +129 -128
  86. package/.next/standalone/docs/hi/built-in-policies.mdx +178 -182
  87. package/.next/standalone/docs/it/built-in-policies.mdx +64 -64
  88. package/.next/standalone/docs/ja/built-in-policies.mdx +128 -128
  89. package/.next/standalone/docs/ko/built-in-policies.mdx +111 -111
  90. package/.next/standalone/docs/pt-br/built-in-policies.mdx +65 -65
  91. package/.next/standalone/docs/ru/built-in-policies.mdx +72 -72
  92. package/.next/standalone/docs/tr/built-in-policies.mdx +99 -99
  93. package/.next/standalone/docs/vi/built-in-policies.mdx +69 -72
  94. package/.next/standalone/docs/zh/built-in-policies.mdx +76 -78
  95. package/.next/standalone/package.json +1 -1
  96. package/.next/standalone/server.js +1 -1
  97. package/.next/standalone/src/auth/login.ts +104 -0
  98. package/.next/standalone/src/auth/logout.ts +50 -0
  99. package/.next/standalone/src/auth/token-store.ts +64 -0
  100. package/.next/standalone/src/hooks/builtin-policies.ts +22 -20
  101. package/.next/standalone/src/hooks/handler.ts +35 -15
  102. package/.next/standalone/src/relay/daemon.ts +362 -0
  103. package/.next/standalone/src/relay/pid.ts +76 -0
  104. package/.next/standalone/src/relay/queue.ts +225 -0
  105. package/bin/failproofai.mjs +91 -4
  106. package/dist/cli.mjs +1155 -54
  107. package/package.json +1 -1
  108. package/src/auth/login.ts +104 -0
  109. package/src/auth/logout.ts +50 -0
  110. package/src/auth/token-store.ts +64 -0
  111. package/src/hooks/builtin-policies.ts +22 -20
  112. package/src/hooks/handler.ts +35 -15
  113. package/src/relay/daemon.ts +362 -0
  114. package/src/relay/pid.ts +76 -0
  115. package/src/relay/queue.ts +225 -0
  116. package/.next/standalone/.next/static/chunks/0ijk_kek9_wyx.js +0 -6
  117. /package/.next/standalone/.next/static/{A9pNTZdoYJTVyPAYwQMx5 → r-wX0MuAfCjbhJm3phQc8}/_buildManifest.js +0 -0
  118. /package/.next/standalone/.next/static/{A9pNTZdoYJTVyPAYwQMx5 → r-wX0MuAfCjbhJm3phQc8}/_clientMiddlewareManifest.js +0 -0
  119. /package/.next/standalone/.next/static/{A9pNTZdoYJTVyPAYwQMx5 → r-wX0MuAfCjbhJm3phQc8}/_ssgManifest.js +0 -0
@@ -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
+ }
@@ -0,0 +1,225 @@
1
+ import {
2
+ appendFileSync,
3
+ mkdirSync,
4
+ existsSync,
5
+ readFileSync,
6
+ statSync,
7
+ renameSync,
8
+ unlinkSync,
9
+ readdirSync,
10
+ chmodSync,
11
+ } from "node:fs";
12
+ import { join } from "node:path";
13
+ import { homedir } from "node:os";
14
+ import { createHash, randomUUID } from "node:crypto";
15
+ import { isLoggedIn } from "../auth/token-store";
16
+
17
+ const QUEUE_DIR = join(homedir(), ".failproofai", "cache", "server-queue");
18
+ const PENDING_FILE = join(QUEUE_DIR, "pending.jsonl");
19
+ const PROCESSING_PREFIX = "processing-";
20
+
21
+ // Cap — if the queue file exceeds this, `appendToServerQueue` is a no-op.
22
+ // Prevents unbounded growth when the daemon is down for a long time or the
23
+ // user installed the CLI but never logged in.
24
+ const MAX_QUEUE_BYTES = 50 * 1024 * 1024; // 50 MB
25
+
26
+ export interface RawEntry {
27
+ timestamp: number;
28
+ eventType: string;
29
+ toolName?: string | null;
30
+ policyName?: string | null;
31
+ policyNames?: string[];
32
+ decision: string;
33
+ reason?: string | null;
34
+ durationMs: number;
35
+ sessionId?: string | null;
36
+ transcriptPath?: string | null;
37
+ cwd?: string | null;
38
+ permissionMode?: string | null;
39
+ hookEventName?: string | null;
40
+ toolInput?: Record<string, unknown>;
41
+ }
42
+
43
+ /**
44
+ * What actually gets persisted and sent to the server. Intentionally a
45
+ * narrower shape than RawEntry — we drop / hash anything that could leak
46
+ * secrets or paths:
47
+ * - toolInput: dropped entirely (can contain credentials, file contents, commands)
48
+ * - cwd: replaced with cwd_hash (SHA-256) so the server can group by project
49
+ * - transcriptPath: dropped (local-only filesystem path)
50
+ * - reason: passed through a redactor for common credential patterns
51
+ */
52
+ export interface QueueEntry {
53
+ client_event_id: string;
54
+ timestamp: number;
55
+ event_type: string;
56
+ tool_name: string | null;
57
+ policy_name: string | null;
58
+ policy_names: string[];
59
+ decision: string;
60
+ reason: string | null;
61
+ duration_ms: number;
62
+ session_id: string | null;
63
+ cwd_hash: string | null;
64
+ permission_mode: string | null;
65
+ hook_event_name: string | null;
66
+ }
67
+
68
+ function hashCwd(cwd: string | null | undefined): string | null {
69
+ if (!cwd) return null;
70
+ return createHash("sha256").update(cwd).digest("hex");
71
+ }
72
+
73
+ function redactReason(reason: string | null | undefined): string | null {
74
+ if (!reason) return reason ?? null;
75
+ return reason
76
+ .replace(/AKIA[0-9A-Z]{16}/g, "[REDACTED-AWS-KEY]")
77
+ .replace(/eyJ[A-Za-z0-9_=-]+\.[A-Za-z0-9_=-]+\.[A-Za-z0-9_=-]+/g, "[REDACTED-JWT]")
78
+ .replace(/ghp_[A-Za-z0-9]{36,}/g, "[REDACTED-GH-TOKEN]")
79
+ .replace(/sk-[A-Za-z0-9]{20,}/g, "[REDACTED-API-KEY]")
80
+ .replace(/Bearer\s+[A-Za-z0-9_.=+-]+/gi, "Bearer [REDACTED]");
81
+ }
82
+
83
+ function sanitize(entry: RawEntry): QueueEntry {
84
+ return {
85
+ client_event_id: randomUUID(),
86
+ timestamp: entry.timestamp,
87
+ event_type: entry.eventType,
88
+ tool_name: entry.toolName ?? null,
89
+ policy_name: entry.policyName ?? null,
90
+ policy_names: entry.policyNames ?? [],
91
+ decision: entry.decision,
92
+ reason: redactReason(entry.reason),
93
+ duration_ms: entry.durationMs,
94
+ session_id: entry.sessionId ?? null,
95
+ cwd_hash: hashCwd(entry.cwd),
96
+ permission_mode: entry.permissionMode ?? null,
97
+ hook_event_name: entry.hookEventName ?? null,
98
+ };
99
+ }
100
+
101
+ function ensureDir(): void {
102
+ if (!existsSync(QUEUE_DIR)) {
103
+ mkdirSync(QUEUE_DIR, { recursive: true, mode: 0o700 });
104
+ }
105
+ }
106
+
107
+ /**
108
+ * Hook-side API — append one event to the pending queue.
109
+ *
110
+ * Uses `appendFileSync` (O_APPEND) which is atomic for small writes, so
111
+ * concurrent hook processes interleave lines correctly without clobbering
112
+ * each other. Sanitizes sensitive fields before persisting.
113
+ *
114
+ * No-op cases (keeps hook path fast and safe):
115
+ * - User not logged in (no auth.json exists)
116
+ * - Queue file already exceeds MAX_QUEUE_BYTES (prevents unbounded growth)
117
+ */
118
+ export function appendToServerQueue(entry: RawEntry): void {
119
+ if (!isLoggedIn()) return;
120
+ ensureDir();
121
+
122
+ try {
123
+ if (existsSync(PENDING_FILE) && statSync(PENDING_FILE).size > MAX_QUEUE_BYTES) {
124
+ return;
125
+ }
126
+ } catch {
127
+ // existsSync/statSync races are fine; proceed
128
+ }
129
+
130
+ const sanitized = sanitize(entry);
131
+ appendFileSync(PENDING_FILE, JSON.stringify(sanitized) + "\n", { mode: 0o600 });
132
+
133
+ // Tighten perms on first create in case the umask allowed wider access
134
+ try {
135
+ chmodSync(PENDING_FILE, 0o600);
136
+ } catch {
137
+ // Windows or non-critical; skip
138
+ }
139
+ }
140
+
141
+ export function queueSizeBytes(): number {
142
+ try {
143
+ return statSync(PENDING_FILE).size;
144
+ } catch {
145
+ return 0;
146
+ }
147
+ }
148
+
149
+ interface ClaimError extends Error {
150
+ code?: string;
151
+ }
152
+
153
+ /**
154
+ * Daemon-side API — atomically claim all pending events into a new
155
+ * processing file. Returns the processing file path, or null ONLY when
156
+ * there's nothing to claim (ENOENT). Other errors throw so we don't
157
+ * silently strand events.
158
+ */
159
+ export function claimPendingBatch(): string | null {
160
+ if (!existsSync(PENDING_FILE)) return null;
161
+ try {
162
+ const size = statSync(PENDING_FILE).size;
163
+ if (size === 0) return null;
164
+ } catch {
165
+ return null;
166
+ }
167
+
168
+ const seq = `${Date.now()}-${process.pid}`;
169
+ const processingFile = join(QUEUE_DIR, `${PROCESSING_PREFIX}${seq}.jsonl`);
170
+ try {
171
+ renameSync(PENDING_FILE, processingFile);
172
+ try {
173
+ chmodSync(processingFile, 0o600);
174
+ } catch {
175
+ // non-critical
176
+ }
177
+ return processingFile;
178
+ } catch (err) {
179
+ const e = err as ClaimError;
180
+ if (e?.code === "ENOENT") return null;
181
+ // Real failure (EACCES, EIO, etc.) — surface to caller; don't lose events
182
+ throw err;
183
+ }
184
+ }
185
+
186
+ export function findOrphanProcessingFiles(): string[] {
187
+ ensureDir();
188
+ try {
189
+ return readdirSync(QUEUE_DIR)
190
+ .filter((n) => n.startsWith(PROCESSING_PREFIX) && n.endsWith(".jsonl"))
191
+ .map((n) => join(QUEUE_DIR, n))
192
+ .sort();
193
+ } catch {
194
+ return [];
195
+ }
196
+ }
197
+
198
+ /**
199
+ * Parse a processing file into structured events, skipping (and logging
200
+ * via stderr) any malformed JSON lines so one bad entry doesn't wedge
201
+ * the entire file forever.
202
+ */
203
+ export function readProcessingFile(path: string): QueueEntry[] {
204
+ if (!existsSync(path)) return [];
205
+ const content = readFileSync(path, "utf8");
206
+ const out: QueueEntry[] = [];
207
+ for (const line of content.split("\n")) {
208
+ const trimmed = line.trim();
209
+ if (!trimmed) continue;
210
+ try {
211
+ out.push(JSON.parse(trimmed) as QueueEntry);
212
+ } catch {
213
+ // Skip malformed line — we can't recover it
214
+ }
215
+ }
216
+ return out;
217
+ }
218
+
219
+ export function deleteProcessingFile(path: string): void {
220
+ try {
221
+ unlinkSync(path);
222
+ } catch {
223
+ // best-effort; stale processing files are cleaned up on next run
224
+ }
225
+ }