@vellumai/assistant 0.7.2 → 0.8.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.
Files changed (424) hide show
  1. package/ARCHITECTURE.md +45 -29
  2. package/Dockerfile +1 -0
  3. package/__tests__/permissions/gateway-threshold-reader.test.ts +236 -9
  4. package/bun.lock +3 -0
  5. package/docs/architecture/memory.md +5 -2
  6. package/knip.json +1 -0
  7. package/node_modules/@vellumai/gateway-client/src/ipc-client.ts +13 -4
  8. package/node_modules/@vellumai/ipc-server-utils/bun.lock +24 -0
  9. package/node_modules/@vellumai/ipc-server-utils/package.json +18 -0
  10. package/node_modules/@vellumai/ipc-server-utils/src/index.ts +6 -0
  11. package/node_modules/@vellumai/ipc-server-utils/src/socket-watchdog.test.ts +430 -0
  12. package/node_modules/@vellumai/ipc-server-utils/src/socket-watchdog.ts +221 -0
  13. package/node_modules/@vellumai/ipc-server-utils/tsconfig.json +20 -0
  14. package/node_modules/@vellumai/skill-host-contracts/src/assistant-event.ts +0 -9
  15. package/node_modules/@vellumai/slack-text/src/index.test.ts +18 -35
  16. package/node_modules/@vellumai/slack-text/src/index.ts +2 -48
  17. package/openapi.yaml +470 -25
  18. package/package.json +3 -1
  19. package/src/__tests__/annotate-risk-options.test.ts +291 -0
  20. package/src/__tests__/app-control-flow.test.ts +21 -11
  21. package/src/__tests__/approval-cascade.test.ts +8 -16
  22. package/src/__tests__/approval-routes-http.test.ts +6 -0
  23. package/src/__tests__/assistant-event-hub.test.ts +48 -0
  24. package/src/__tests__/assistant-event.test.ts +0 -10
  25. package/src/__tests__/assistant-events-sse-hardening.test.ts +2 -7
  26. package/src/__tests__/assistant-feature-flags-integration.test.ts +18 -0
  27. package/src/__tests__/auto-analysis-end-to-end.test.ts +48 -0
  28. package/src/__tests__/background-workers-disk-pressure.test.ts +268 -0
  29. package/src/__tests__/call-constants.test.ts +10 -1
  30. package/src/__tests__/call-controller.test.ts +127 -0
  31. package/src/__tests__/call-conversation-messages.test.ts +8 -2
  32. package/src/__tests__/channel-inbound-disk-pressure.test.ts +537 -0
  33. package/src/__tests__/channel-readiness-service.test.ts +4 -2
  34. package/src/__tests__/cli-memory-v2-reembed-skills.test.ts +58 -28
  35. package/src/__tests__/config-loader-backfill.test.ts +379 -0
  36. package/src/__tests__/config-loader-platform-defaults.test.ts +284 -1
  37. package/src/__tests__/config-schema.test.ts +1 -0
  38. package/src/__tests__/config-watcher-cleanup-throttle.test.ts +18 -9
  39. package/src/__tests__/config-watcher.test.ts +140 -69
  40. package/src/__tests__/context-search-agent-runner.test.ts +61 -3
  41. package/src/__tests__/context-search-conversations-source.test.ts +0 -24
  42. package/src/__tests__/context-search-fanout.test.ts +0 -1
  43. package/src/__tests__/context-search-memory-source.test.ts +6 -33
  44. package/src/__tests__/context-search-memory-v2-source.test.ts +0 -2
  45. package/src/__tests__/context-search-pkb-source.test.ts +12 -7
  46. package/src/__tests__/context-search-workspace-source.test.ts +0 -1
  47. package/src/__tests__/conversation-abort-tool-results.test.ts +1 -0
  48. package/src/__tests__/conversation-agent-loop-disk-pressure.test.ts +223 -0
  49. package/src/__tests__/conversation-agent-loop-inference-profile.test.ts +1 -1
  50. package/src/__tests__/conversation-agent-loop-overflow.test.ts +1 -1
  51. package/src/__tests__/conversation-agent-loop.test.ts +457 -8
  52. package/src/__tests__/conversation-confirmation-signals.test.ts +5 -13
  53. package/src/__tests__/conversation-error.test.ts +150 -3
  54. package/src/__tests__/conversation-init.benchmark.test.ts +1 -1
  55. package/src/__tests__/conversation-process-callsite.test.ts +38 -0
  56. package/src/__tests__/conversation-provider-retry-repair.test.ts +1 -0
  57. package/src/__tests__/conversation-runtime-assembly.test.ts +74 -0
  58. package/src/__tests__/conversation-slash-unknown.test.ts +1 -0
  59. package/src/__tests__/conversation-speed-override.test.ts +0 -3
  60. package/src/__tests__/conversation-store.test.ts +0 -18
  61. package/src/__tests__/conversation-surfaces-action-delivery.test.ts +170 -9
  62. package/src/__tests__/conversation-surfaces-app-control.test.ts +15 -4
  63. package/src/__tests__/conversation-surfaces-data-persist.test.ts +476 -0
  64. package/src/__tests__/conversation-tool-setup-app-refresh.test.ts +61 -5
  65. package/src/__tests__/conversation-workspace-injection.test.ts +1 -1
  66. package/src/__tests__/conversation-workspace-tool-tracking.test.ts +1 -1
  67. package/src/__tests__/credentials-cli.test.ts +7 -0
  68. package/src/__tests__/cu-unified-flow.test.ts +176 -10
  69. package/src/__tests__/date-context.test.ts +164 -2
  70. package/src/__tests__/disk-pressure-guard.test.ts +262 -0
  71. package/src/__tests__/disk-pressure-lifecycle.test.ts +168 -0
  72. package/src/__tests__/disk-pressure-policy.test.ts +241 -0
  73. package/src/__tests__/disk-pressure-routes.test.ts +379 -0
  74. package/src/__tests__/disk-pressure-tools.test.ts +277 -0
  75. package/src/__tests__/disk-usage.test.ts +150 -0
  76. package/src/__tests__/events-client-registration.test.ts +52 -0
  77. package/src/__tests__/events-dev-bypass-actor.test.ts +162 -0
  78. package/src/__tests__/file-write-tool.test.ts +4 -10
  79. package/src/__tests__/filing-service.test.ts +2 -20
  80. package/src/__tests__/handlers-skills-memory-v2-reseed.test.ts +10 -26
  81. package/src/__tests__/heartbeat-disk-pressure.test.ts +183 -0
  82. package/src/__tests__/heartbeat-service.test.ts +260 -11
  83. package/src/__tests__/host-app-control-proxy.test.ts +195 -25
  84. package/src/__tests__/host-bash-proxy.test.ts +227 -34
  85. package/src/__tests__/host-bash-routes.test.ts +178 -13
  86. package/src/__tests__/host-cu-proxy.test.ts +210 -3
  87. package/src/__tests__/host-cu-routes-targeted.test.ts +141 -12
  88. package/src/__tests__/host-file-proxy-targeted.test.ts +48 -9
  89. package/src/__tests__/host-file-proxy.test.ts +268 -6
  90. package/src/__tests__/host-file-routes-targeted.test.ts +175 -17
  91. package/src/__tests__/host-transfer-proxy-targeted.test.ts +408 -59
  92. package/src/__tests__/host-transfer-routes-targeted.test.ts +232 -17
  93. package/src/__tests__/http-user-message-parity.test.ts +107 -1
  94. package/src/__tests__/injector-chain.test.ts +36 -16
  95. package/src/__tests__/injector-disk-pressure.test.ts +224 -0
  96. package/src/__tests__/injector-pkb-v2-silenced.test.ts +10 -7
  97. package/src/__tests__/lifecycle-memory-v2-seed.test.ts +154 -67
  98. package/src/__tests__/managed-profile-guard.test.ts +18 -0
  99. package/src/__tests__/mcp-abort-signal.test.ts +130 -0
  100. package/src/__tests__/memory-admin-recall.test.ts +3 -11
  101. package/src/__tests__/memory-retrieval-pipeline.test.ts +22 -1
  102. package/src/__tests__/normalize-onboarding.test.ts +180 -0
  103. package/src/__tests__/notification-decision-fallback.test.ts +91 -0
  104. package/src/__tests__/notification-decision-strategy.test.ts +22 -0
  105. package/src/__tests__/oauth-cli.test.ts +121 -0
  106. package/src/__tests__/oauth-connect-routes.test.ts +316 -0
  107. package/src/__tests__/oauth-provider-seed-logos.test.ts +24 -2
  108. package/src/__tests__/onboarding-persona-write.test.ts +308 -0
  109. package/src/__tests__/openai-provider.test.ts +45 -8
  110. package/src/__tests__/persist-onboarding-artifacts.test.ts +44 -64
  111. package/src/__tests__/platform-callback-registration.test.ts +21 -4
  112. package/src/__tests__/platform.test.ts +2 -1
  113. package/src/__tests__/playbook-execution.test.ts +0 -43
  114. package/src/__tests__/plugin-tool-contribution.test.ts +47 -0
  115. package/src/__tests__/prechat-onboarding-contract.test.ts +214 -27
  116. package/src/__tests__/provider-tool-name.test.ts +23 -0
  117. package/src/__tests__/relay-server.test.ts +60 -5
  118. package/src/__tests__/runtime-events-sse.test.ts +4 -8
  119. package/src/__tests__/scheduler-disk-pressure.test.ts +148 -0
  120. package/src/__tests__/secret-ingress-http.test.ts +0 -1
  121. package/src/__tests__/secret-prompt-log-hygiene.test.ts +7 -5
  122. package/src/__tests__/secret-prompter-channel-fallback.test.ts +7 -5
  123. package/src/__tests__/secret-response-routing.test.ts +7 -5
  124. package/src/__tests__/server-history-render.test.ts +82 -0
  125. package/src/__tests__/skill-include-graph.test.ts +31 -0
  126. package/src/__tests__/skill-load-tool.test.ts +44 -16
  127. package/src/__tests__/skills.test.ts +39 -0
  128. package/src/__tests__/suggestion-routes.test.ts +46 -0
  129. package/src/__tests__/tool-execution-pipeline.benchmark.test.ts +0 -42
  130. package/src/__tests__/tool-executor.test.ts +155 -0
  131. package/src/__tests__/twilio-validation.test.ts +2 -2
  132. package/src/__tests__/voice-session-bridge.test.ts +3 -0
  133. package/src/__tests__/workspace-migration-065-bump-stale-heartbeat-interval.test.ts +122 -0
  134. package/src/__tests__/workspace-migration-066-seed-heartbeat-callsite-cost-default.test.ts +285 -0
  135. package/src/__tests__/workspace-migration-068-release-notes-local-timezone.test.ts +90 -0
  136. package/src/__tests__/workspace-migration-069-seed-onboarding-threads.test.ts +120 -0
  137. package/src/__tests__/workspace-migration-071-remove-safe-storage-release-note.test.ts +206 -0
  138. package/src/__tests__/workspace-migration-safe-storage-limits-release.test.ts +78 -0
  139. package/src/agent/loop.ts +11 -0
  140. package/src/approvals/guardian-request-resolvers.ts +3 -32
  141. package/src/backup/snapshot-lock.ts +2 -27
  142. package/src/bundler/compiler-tools.ts +3 -2
  143. package/src/calls/call-constants.ts +5 -8
  144. package/src/calls/call-controller.ts +130 -67
  145. package/src/calls/call-conversation-messages.ts +46 -10
  146. package/src/calls/relay-server.ts +7 -1
  147. package/src/calls/voice-session-bridge.ts +1 -1
  148. package/src/cli/commands/__tests__/webhooks.test.ts +0 -4
  149. package/src/cli/commands/bash.ts +35 -108
  150. package/src/cli/commands/contacts.ts +64 -25
  151. package/src/cli/commands/credentials.ts +56 -0
  152. package/src/cli/commands/memory-v2.ts +11 -10
  153. package/src/cli/commands/oauth/__tests__/connect.test.ts +401 -219
  154. package/src/cli/commands/oauth/connect.ts +124 -40
  155. package/src/cli/commands/platform/__tests__/callback-routes-list.test.ts +0 -3
  156. package/src/cli/commands/platform/__tests__/connect.test.ts +7 -1
  157. package/src/cli/commands/platform/__tests__/disconnect.test.ts +7 -1
  158. package/src/cli/commands/platform/__tests__/status.test.ts +103 -6
  159. package/src/cli/commands/platform/index.ts +16 -7
  160. package/src/cli/commands/status.ts +57 -0
  161. package/src/cli/program.ts +4 -2
  162. package/src/config/assistant-feature-flags.ts +13 -3
  163. package/src/config/bundled-skills/app-builder/SKILL.md +1 -3
  164. package/src/config/bundled-skills/messaging/tools/messaging-analyze-style.ts +4 -3
  165. package/src/config/bundled-skills/phone-calls/references/TROUBLESHOOTING.md +13 -7
  166. package/src/config/bundled-skills/playbooks/tools/playbook-create.ts +2 -2
  167. package/src/config/bundled-skills/playbooks/tools/playbook-delete.ts +2 -2
  168. package/src/config/bundled-skills/playbooks/tools/playbook-list.ts +2 -2
  169. package/src/config/bundled-skills/playbooks/tools/playbook-update.ts +2 -2
  170. package/src/config/env.ts +0 -8
  171. package/src/config/feature-flag-registry.json +13 -5
  172. package/src/config/loader.ts +199 -27
  173. package/src/config/schemas/__tests__/memory-v2.test.ts +10 -5
  174. package/src/config/schemas/call-site-catalog.ts +14 -0
  175. package/src/config/schemas/channels.ts +0 -5
  176. package/src/config/schemas/heartbeat.ts +1 -1
  177. package/src/config/schemas/llm.ts +2 -0
  178. package/src/config/schemas/memory-lifecycle.ts +13 -0
  179. package/src/config/schemas/memory-v2.ts +76 -12
  180. package/src/config/schemas/platform.ts +43 -3
  181. package/src/config/schemas/services.ts +28 -0
  182. package/src/config/seed-inference-profiles.ts +230 -33
  183. package/src/contacts/contact-store.ts +0 -25
  184. package/src/daemon/__tests__/conversation-lifecycle-auto-analyze.test.ts +32 -0
  185. package/src/daemon/__tests__/conversation-tool-setup.test.ts +86 -25
  186. package/src/daemon/assistant-attachments.ts +4 -4
  187. package/src/daemon/config-watcher.ts +85 -57
  188. package/src/daemon/conversation-agent-loop-handlers.ts +38 -0
  189. package/src/daemon/conversation-agent-loop.ts +183 -43
  190. package/src/daemon/conversation-error.ts +87 -15
  191. package/src/daemon/conversation-lifecycle.ts +22 -10
  192. package/src/daemon/conversation-process.ts +8 -0
  193. package/src/daemon/conversation-runtime-assembly.ts +26 -0
  194. package/src/daemon/conversation-store.ts +2 -2
  195. package/src/daemon/conversation-surfaces.ts +211 -29
  196. package/src/daemon/conversation-tool-setup.ts +66 -19
  197. package/src/daemon/conversation.ts +18 -23
  198. package/src/daemon/date-context.ts +71 -22
  199. package/src/daemon/disk-pressure-background-gate.ts +73 -0
  200. package/src/daemon/disk-pressure-guard.ts +343 -0
  201. package/src/daemon/disk-pressure-policy.ts +163 -0
  202. package/src/daemon/handlers/shared.ts +26 -1
  203. package/src/daemon/handlers/skills.ts +3 -4
  204. package/src/daemon/host-app-control-proxy.ts +137 -41
  205. package/src/daemon/host-bash-proxy.ts +47 -22
  206. package/src/daemon/host-browser-proxy.ts +1 -1
  207. package/src/daemon/host-cu-proxy.ts +50 -4
  208. package/src/daemon/host-file-proxy.ts +44 -8
  209. package/src/daemon/host-transfer-proxy.ts +97 -6
  210. package/src/daemon/lifecycle.ts +167 -101
  211. package/src/daemon/meet-host-supervisor.ts +4 -4
  212. package/src/daemon/meet-manifest-loader.ts +0 -1
  213. package/src/daemon/memory-v2-startup.ts +66 -15
  214. package/src/daemon/message-protocol.ts +3 -0
  215. package/src/daemon/message-types/conversations.ts +4 -0
  216. package/src/daemon/message-types/disk-pressure.ts +9 -0
  217. package/src/daemon/message-types/messages.ts +22 -1
  218. package/src/daemon/profiler-run-store.ts +5 -5
  219. package/src/daemon/tool-setup-types.ts +2 -2
  220. package/src/documents/document-store.ts +119 -0
  221. package/src/filing/filing-service.ts +29 -5
  222. package/src/heartbeat/__tests__/heartbeat-feed-event.test.ts +9 -16
  223. package/src/heartbeat/__tests__/heartbeat-run-store.test.ts +36 -0
  224. package/src/heartbeat/heartbeat-run-store.ts +13 -0
  225. package/src/heartbeat/heartbeat-service.ts +205 -31
  226. package/src/home/feed-scheduler.ts +18 -0
  227. package/src/inbound/platform-callback-registration.ts +8 -15
  228. package/src/ipc/__tests__/clients-list-ipc.test.ts +169 -0
  229. package/src/ipc/assistant-server.ts +149 -38
  230. package/src/ipc/gateway-client.ts +37 -3
  231. package/src/ipc/skill-server.ts +99 -42
  232. package/src/live-voice/live-voice-archive.ts +4 -4
  233. package/src/live-voice/protocol.ts +5 -7
  234. package/src/media/image-service.ts +1 -7
  235. package/src/memory/__tests__/fixtures/memory-v2-activation-fixtures.ts +21 -13
  236. package/src/memory/__tests__/jobs-worker-v2-schedule.test.ts +34 -51
  237. package/src/memory/__tests__/memory-v2-activation-log-store.test.ts +0 -6
  238. package/src/memory/__tests__/memory-v2-concept-frequency.test.ts +272 -0
  239. package/src/memory/admin.ts +5 -9
  240. package/src/memory/context-search/agent-runner.ts +19 -2
  241. package/src/memory/context-search/sources/conversations.ts +2 -11
  242. package/src/memory/context-search/sources/memory-v2.ts +1 -16
  243. package/src/memory/context-search/sources/memory.ts +2 -3
  244. package/src/memory/context-search/sources/pkb.ts +2 -3
  245. package/src/memory/context-search/types.ts +0 -1
  246. package/src/memory/conversation-crud.ts +4 -12
  247. package/src/memory/db-init.ts +2 -0
  248. package/src/memory/embedding-runtime-manager.ts +119 -5
  249. package/src/memory/graph/__tests__/conversation-graph-memory-v2-routing.test.ts +136 -82
  250. package/src/memory/graph/__tests__/handle-remember-v2.test.ts +11 -26
  251. package/src/memory/graph/conversation-graph-memory.ts +72 -61
  252. package/src/memory/graph/extraction.ts +1 -3
  253. package/src/memory/graph/graph-search.test.ts +11 -67
  254. package/src/memory/graph/graph-search.ts +4 -24
  255. package/src/memory/graph/retriever.test.ts +12 -1
  256. package/src/memory/graph/retriever.ts +10 -15
  257. package/src/memory/graph/tool-handlers.ts +3 -4
  258. package/src/memory/graph/tools.ts +4 -4
  259. package/src/memory/indexer.ts +53 -45
  260. package/src/memory/job-handlers/backfill.ts +2 -11
  261. package/src/memory/job-handlers/cleanup.ts +43 -0
  262. package/src/memory/job-handlers/embedding.ts +6 -8
  263. package/src/memory/job-handlers/summarization.ts +2 -7
  264. package/src/memory/jobs/__tests__/embed-concept-page.test.ts +116 -0
  265. package/src/memory/jobs/embed-concept-page.ts +223 -87
  266. package/src/memory/jobs-store.ts +48 -0
  267. package/src/memory/jobs-worker.ts +85 -43
  268. package/src/memory/memory-v2-activation-log-store.ts +32 -14
  269. package/src/memory/memory-v2-concept-frequency.ts +169 -0
  270. package/src/memory/migrations/239-trace-events-created-at-index.ts +18 -0
  271. package/src/memory/migrations/index.ts +1 -0
  272. package/src/memory/pkb/pkb-search.test.ts +7 -0
  273. package/src/memory/pkb/pkb-search.ts +4 -5
  274. package/src/memory/qdrant-client.ts +3 -13
  275. package/src/memory/rerank-local.ts +374 -0
  276. package/src/memory/search/semantic.ts +10 -72
  277. package/src/memory/trace-event-store.ts +1 -17
  278. package/src/memory/v2/__tests__/activation.test.ts +346 -255
  279. package/src/memory/v2/__tests__/consolidation-job.test.ts +61 -40
  280. package/src/memory/v2/__tests__/injection.test.ts +297 -190
  281. package/src/memory/v2/__tests__/prompts-consolidation.test.ts +61 -2
  282. package/src/memory/v2/__tests__/qdrant.test.ts +326 -9
  283. package/src/memory/v2/__tests__/reranker.test.ts +338 -0
  284. package/src/memory/v2/__tests__/sim.test.ts +113 -196
  285. package/src/memory/v2/__tests__/skill-store.test.ts +71 -65
  286. package/src/memory/v2/__tests__/static-context.test.ts +77 -14
  287. package/src/memory/v2/__tests__/sweep-job.test.ts +19 -33
  288. package/src/memory/v2/activation.ts +149 -156
  289. package/src/memory/v2/consolidation-job.ts +69 -20
  290. package/src/memory/v2/injection.ts +75 -68
  291. package/src/memory/v2/page-store.ts +39 -0
  292. package/src/memory/v2/prompts/consolidation.ts +41 -1
  293. package/src/memory/v2/qdrant.ts +306 -46
  294. package/src/memory/v2/reranker.ts +177 -0
  295. package/src/memory/v2/sim.ts +77 -110
  296. package/src/memory/v2/skill-content.ts +4 -3
  297. package/src/memory/v2/skill-store.ts +82 -59
  298. package/src/memory/v2/static-context.ts +26 -8
  299. package/src/memory/v2/sweep-job.ts +5 -6
  300. package/src/memory/v2/types.ts +17 -10
  301. package/src/notifications/copy-composer.ts +47 -0
  302. package/src/notifications/decision-engine.ts +46 -0
  303. package/src/notifications/signal.ts +4 -0
  304. package/src/oauth/AGENTS.md +3 -1
  305. package/src/oauth/__tests__/oauth-connect-state.test.ts +137 -0
  306. package/src/oauth/connect-orchestrator.ts +2 -0
  307. package/src/oauth/connection-resolver.test.ts +66 -1
  308. package/src/oauth/connection-resolver.ts +55 -1
  309. package/src/oauth/oauth-connect-state.ts +77 -0
  310. package/src/oauth/seed-providers.ts +58 -1
  311. package/src/permissions/gateway-threshold-reader.ts +116 -8
  312. package/src/permissions/prompter.ts +86 -96
  313. package/src/permissions/secret-prompter.ts +31 -31
  314. package/src/plugins/defaults/injectors.ts +36 -4
  315. package/src/plugins/defaults/memory-retrieval.ts +5 -6
  316. package/src/plugins/types.ts +7 -0
  317. package/src/proactive-artifact/aux-message-injector.ts +74 -0
  318. package/src/proactive-artifact/decision.test.ts +226 -0
  319. package/src/proactive-artifact/decision.ts +165 -0
  320. package/src/proactive-artifact/index.ts +7 -0
  321. package/src/proactive-artifact/job.test.ts +914 -0
  322. package/src/proactive-artifact/job.ts +366 -0
  323. package/src/proactive-artifact/message-copy.ts +58 -0
  324. package/src/proactive-artifact/trigger-state.test.ts +277 -0
  325. package/src/proactive-artifact/trigger-state.ts +119 -0
  326. package/src/prompts/normalize-onboarding.ts +80 -0
  327. package/src/prompts/persona-resolver.ts +101 -9
  328. package/src/prompts/system-prompt.ts +21 -7
  329. package/src/prompts/templates/BOOTSTRAP.md +13 -5
  330. package/src/prompts/templates/SOUL.md +13 -28
  331. package/src/providers/__tests__/retry-callsite.test.ts +222 -1
  332. package/src/providers/model-intents.ts +7 -0
  333. package/src/providers/openrouter/client.ts +8 -0
  334. package/src/providers/retry.ts +50 -0
  335. package/src/providers/types.ts +1 -0
  336. package/src/runtime/__tests__/agent-wake.test.ts +456 -3
  337. package/src/runtime/agent-wake.ts +238 -100
  338. package/src/runtime/assistant-event-hub.ts +36 -6
  339. package/src/runtime/assistant-event.ts +0 -1
  340. package/src/runtime/auth/__tests__/route-policy.test.ts +64 -0
  341. package/src/runtime/auth/route-policy.ts +15 -1
  342. package/src/runtime/auth/same-actor.ts +216 -0
  343. package/src/runtime/channel-approvals.ts +3 -2
  344. package/src/runtime/channel-retry-sweep.ts +65 -1
  345. package/src/runtime/local-actor-identity.ts +52 -11
  346. package/src/runtime/pending-interactions.ts +27 -15
  347. package/src/runtime/routes/__tests__/client-routes.test.ts +155 -0
  348. package/src/runtime/routes/__tests__/conversation-query-routes.test.ts +0 -5
  349. package/src/runtime/routes/__tests__/heartbeat-routes.test.ts +1 -1
  350. package/src/runtime/routes/__tests__/memory-v2-routes.test.ts +147 -0
  351. package/src/runtime/routes/approval-routes.ts +7 -3
  352. package/src/runtime/routes/client-routes.ts +20 -2
  353. package/src/runtime/routes/consolidation-routes.ts +8 -9
  354. package/src/runtime/routes/contact-routes.ts +0 -25
  355. package/src/runtime/routes/conversation-query-routes.ts +44 -1
  356. package/src/runtime/routes/conversation-routes.ts +35 -26
  357. package/src/runtime/routes/debug-bash-routes.ts +165 -0
  358. package/src/runtime/routes/disk-pressure-routes.ts +121 -0
  359. package/src/runtime/routes/document-pdf-renderer.ts +6 -2
  360. package/src/runtime/routes/documents-routes.ts +2 -75
  361. package/src/runtime/routes/events-routes.ts +41 -9
  362. package/src/runtime/routes/filing-routes.ts +2 -3
  363. package/src/runtime/routes/host-bash-routes.ts +23 -3
  364. package/src/runtime/routes/host-cu-routes.ts +33 -6
  365. package/src/runtime/routes/host-file-routes.ts +32 -6
  366. package/src/runtime/routes/host-transfer-routes.ts +79 -16
  367. package/src/runtime/routes/identity-routes.ts +7 -138
  368. package/src/runtime/routes/inbound-message-handler.ts +77 -12
  369. package/src/runtime/routes/index.ts +6 -0
  370. package/src/runtime/routes/memory-item-routes.test.ts +37 -17
  371. package/src/runtime/routes/memory-item-routes.ts +5 -6
  372. package/src/runtime/routes/memory-v2-routes.ts +136 -17
  373. package/src/runtime/routes/oauth-connect-routes.ts +153 -0
  374. package/src/runtime/verification-outbound-actions.ts +4 -4
  375. package/src/schedule/run-script.ts +37 -5
  376. package/src/schedule/scheduler.ts +20 -1
  377. package/src/security/encrypted-store.ts +2 -0
  378. package/src/security/secure-keys.ts +55 -0
  379. package/src/skills/include-graph.ts +35 -13
  380. package/src/skills/remote-skill-policy.ts +4 -10
  381. package/src/subagent/index.ts +1 -7
  382. package/src/subagent/manager.ts +1 -15
  383. package/src/tasks/task-runner.ts +0 -1
  384. package/src/tasks/task-store.ts +0 -3
  385. package/src/tools/background-tool-registry.ts +17 -3
  386. package/src/tools/document/document-tool.ts +20 -0
  387. package/src/tools/executor.ts +18 -2
  388. package/src/tools/host-filesystem/edit.test.ts +151 -0
  389. package/src/tools/host-filesystem/edit.ts +43 -1
  390. package/src/tools/host-filesystem/read.test.ts +129 -0
  391. package/src/tools/host-filesystem/read.ts +43 -1
  392. package/src/tools/host-filesystem/transfer.test.ts +127 -2
  393. package/src/tools/host-filesystem/transfer.ts +56 -11
  394. package/src/tools/host-filesystem/write.test.ts +134 -0
  395. package/src/tools/host-filesystem/write.ts +43 -1
  396. package/src/tools/host-terminal/host-shell.ts +13 -6
  397. package/src/tools/mcp/mcp-tool-factory.ts +2 -1
  398. package/src/tools/memory/register.test.ts +14 -9
  399. package/src/tools/memory/register.ts +1 -2
  400. package/src/tools/permission-checker.ts +15 -0
  401. package/src/tools/provider-tool-name.ts +28 -0
  402. package/src/tools/registry.ts +30 -9
  403. package/src/tools/skills/load.ts +24 -20
  404. package/src/tools/terminal/shell.ts +9 -1
  405. package/src/tools/tool-approval-handler.ts +31 -6
  406. package/src/tools/tool-name-aliases.ts +19 -0
  407. package/src/tools/types.ts +43 -3
  408. package/src/tts/provider-catalog.ts +3 -5
  409. package/src/util/disk-usage.ts +138 -0
  410. package/src/util/platform.ts +21 -11
  411. package/src/util/process-liveness.ts +26 -0
  412. package/src/workspace/heartbeat-service.ts +19 -0
  413. package/src/workspace/migrations/065-bump-stale-heartbeat-interval.ts +60 -0
  414. package/src/workspace/migrations/066-seed-heartbeat-callsite-cost-default.ts +146 -0
  415. package/src/workspace/migrations/067-release-notes-safe-storage-limits.ts +14 -0
  416. package/src/workspace/migrations/068-release-notes-local-timezone.ts +65 -0
  417. package/src/workspace/migrations/069-seed-onboarding-threads.ts +28 -0
  418. package/src/workspace/migrations/070-memory-v2-summary-schema-rebuild.ts +31 -0
  419. package/src/workspace/migrations/071-remove-safe-storage-release-note.ts +111 -0
  420. package/src/workspace/migrations/registry.ts +14 -0
  421. package/src/__tests__/conversation-tool-setup-memory-scope.test.ts +0 -167
  422. package/src/memory/v2/__tests__/skill-qdrant.test.ts +0 -657
  423. package/src/memory/v2/skill-qdrant.ts +0 -404
  424. package/src/signals/bash.ts +0 -198
@@ -0,0 +1,430 @@
1
+ import {
2
+ afterAll,
3
+ afterEach,
4
+ beforeEach,
5
+ describe,
6
+ expect,
7
+ test,
8
+ } from "bun:test";
9
+ import {
10
+ existsSync,
11
+ mkdtempSync,
12
+ rmSync,
13
+ unlinkSync,
14
+ } from "node:fs";
15
+ import { createConnection, createServer, type Server, type Socket } from "node:net";
16
+ import { tmpdir } from "node:os";
17
+ import { join } from "node:path";
18
+
19
+ import { SocketWatchdog, type SocketWatchdogLogger } from "./socket-watchdog.js";
20
+
21
+ // macOS caps Unix-socket paths at sizeof(sun_path)-1 == 103 bytes, so the
22
+ // shared test-preload temp dir is too long. Mint a short path under tmpdir
23
+ // for these tests.
24
+ const shortRoot = mkdtempSync(join(tmpdir(), "vmw-"));
25
+ const socketPath = join(shortRoot, "g.sock");
26
+
27
+ afterAll(() => {
28
+ try {
29
+ rmSync(shortRoot, { recursive: true, force: true });
30
+ } catch {
31
+ // best-effort
32
+ }
33
+ });
34
+
35
+ interface TestHarness {
36
+ watchdog: SocketWatchdog;
37
+ /** Mutated by tests to simulate stop()/restart. */
38
+ serverRef: { current: Server | null };
39
+ /** Servers handed to onRebind, captured for assertions + cleanup. */
40
+ rebinds: Array<{ newServer: Server; oldServer: Server }>;
41
+ log: SocketWatchdogLogger;
42
+ loggedErrors: Array<{ obj: object; msg?: string }>;
43
+ /** Tracks every server the harness factory produced, for cleanup. */
44
+ spawnedServers: Server[];
45
+ }
46
+
47
+ interface BuildOptions {
48
+ intervalMs?: number;
49
+ createServerOverride?: () => Server;
50
+ /** Override `getServer` to simulate races. */
51
+ getServerOverride?: () => Server | null;
52
+ }
53
+
54
+ function buildHarness(opts: BuildOptions): TestHarness {
55
+ const serverRef: { current: Server | null } = { current: null };
56
+ const rebinds: Array<{ newServer: Server; oldServer: Server }> = [];
57
+ const loggedErrors: Array<{ obj: object; msg?: string }> = [];
58
+ const spawnedServers: Server[] = [];
59
+
60
+ const log: SocketWatchdogLogger = {
61
+ info: () => {},
62
+ warn: () => {},
63
+ error: (obj, msg) => {
64
+ loggedErrors.push({ obj, msg });
65
+ },
66
+ };
67
+
68
+ const defaultFactory = () => {
69
+ const s = createServer();
70
+ s.on("error", () => {
71
+ /* tests don't care; suppress */
72
+ });
73
+ spawnedServers.push(s);
74
+ return s;
75
+ };
76
+
77
+ const watchdog = new SocketWatchdog({
78
+ socketPath,
79
+ intervalMs: opts.intervalMs ?? 0,
80
+ getServer: opts.getServerOverride ?? (() => serverRef.current),
81
+ createServer: opts.createServerOverride ?? defaultFactory,
82
+ onRebind: (newServer, oldServer) => {
83
+ rebinds.push({ newServer, oldServer });
84
+ serverRef.current = newServer;
85
+ // Mirror gateway behavior: close old server gracefully so its
86
+ // accept-loop drains. Close errors are not the watchdog's concern.
87
+ oldServer.close(() => {
88
+ /* drained */
89
+ });
90
+ },
91
+ log,
92
+ });
93
+
94
+ return { watchdog, serverRef, rebinds, log, loggedErrors, spawnedServers };
95
+ }
96
+
97
+ /**
98
+ * Spin up a real listening server and install it into the harness. Returns
99
+ * once the kernel reports the socket file present on disk.
100
+ */
101
+ async function startInitialServer(harness: TestHarness): Promise<Server> {
102
+ const server = createServer();
103
+ server.on("error", () => {
104
+ /* ignore */
105
+ });
106
+ harness.spawnedServers.push(server);
107
+ await new Promise<void>((resolve, reject) => {
108
+ server.once("error", reject);
109
+ server.once("listening", () => resolve());
110
+ server.listen(socketPath);
111
+ });
112
+ harness.serverRef.current = server;
113
+ return server;
114
+ }
115
+
116
+ function connectClient(path: string): Promise<Socket> {
117
+ return new Promise<Socket>((resolve, reject) => {
118
+ const client: Socket = createConnection(path, () => resolve(client));
119
+ client.on("error", reject);
120
+ });
121
+ }
122
+
123
+ async function closeServer(server: Server): Promise<void> {
124
+ await new Promise<void>((resolve) => {
125
+ server.close(() => resolve());
126
+ });
127
+ }
128
+
129
+ describe("SocketWatchdog", () => {
130
+ let harness: TestHarness | undefined;
131
+ const sockets: Socket[] = [];
132
+
133
+ beforeEach(() => {
134
+ harness = undefined;
135
+ // Defensive: clean up any leftover socket file from a previous test
136
+ // whose afterEach didn't fully drain.
137
+ if (existsSync(socketPath)) {
138
+ try {
139
+ unlinkSync(socketPath);
140
+ } catch {
141
+ /* ignore */
142
+ }
143
+ }
144
+ });
145
+
146
+ afterEach(async () => {
147
+ for (const s of sockets) {
148
+ if (!s.destroyed) s.destroy();
149
+ }
150
+ sockets.length = 0;
151
+
152
+ if (harness) {
153
+ harness.watchdog.stop();
154
+ // Close every server the harness produced, regardless of how the
155
+ // test left things. Closing an already-closed server is a no-op.
156
+ for (const s of harness.spawnedServers) {
157
+ try {
158
+ await closeServer(s);
159
+ } catch {
160
+ /* already closed */
161
+ }
162
+ }
163
+ harness = undefined;
164
+ }
165
+
166
+ if (existsSync(socketPath)) {
167
+ try {
168
+ unlinkSync(socketPath);
169
+ } catch {
170
+ /* ignore */
171
+ }
172
+ }
173
+ });
174
+
175
+ test("rebindIfMissing is a no-op when the socket path exists", async () => {
176
+ harness = buildHarness({});
177
+ await startInitialServer(harness);
178
+
179
+ const rebound = await harness.watchdog.rebindIfMissing();
180
+ expect(rebound).toBe(false);
181
+ expect(harness.rebinds).toHaveLength(0);
182
+ expect(existsSync(socketPath)).toBe(true);
183
+ });
184
+
185
+ test("rebindIfMissing is a no-op when getServer returns null", async () => {
186
+ harness = buildHarness({});
187
+ // serverRef.current stays null.
188
+ const rebound = await harness.watchdog.rebindIfMissing();
189
+ expect(rebound).toBe(false);
190
+ expect(harness.rebinds).toHaveLength(0);
191
+ });
192
+
193
+ test("rebindIfMissing recreates the listener when the path is gone", async () => {
194
+ harness = buildHarness({});
195
+ const initial = await startInitialServer(harness);
196
+ expect(existsSync(socketPath)).toBe(true);
197
+
198
+ // Simulate the cleanup that wipes /run/* — unlink the path while the
199
+ // listener fd is still alive in the kernel.
200
+ unlinkSync(socketPath);
201
+ expect(existsSync(socketPath)).toBe(false);
202
+
203
+ const rebound = await harness.watchdog.rebindIfMissing();
204
+ expect(rebound).toBe(true);
205
+ expect(existsSync(socketPath)).toBe(true);
206
+ expect(harness.rebinds).toHaveLength(1);
207
+ expect(harness.rebinds[0]!.oldServer).toBe(initial);
208
+ expect(harness.serverRef.current).toBe(harness.rebinds[0]!.newServer);
209
+
210
+ // A fresh client can connect to the re-bound listener.
211
+ const client = await connectClient(socketPath);
212
+ sockets.push(client);
213
+ expect(client.destroyed).toBe(false);
214
+ });
215
+
216
+ test("connected clients survive a rebind", async () => {
217
+ harness = buildHarness({});
218
+ await startInitialServer(harness);
219
+
220
+ const survivor = await connectClient(socketPath);
221
+ sockets.push(survivor);
222
+ expect(survivor.destroyed).toBe(false);
223
+
224
+ unlinkSync(socketPath);
225
+ const rebound = await harness.watchdog.rebindIfMissing();
226
+ expect(rebound).toBe(true);
227
+
228
+ // Give the close-callback a moment to settle without churning the EL.
229
+ await new Promise((r) => setTimeout(r, 10));
230
+ expect(survivor.destroyed).toBe(false);
231
+ });
232
+
233
+ test("rebindIfMissing aborts when getServer changes mid-listen (shutdown race)", async () => {
234
+ // Drive the race deterministically by mutating what getServer returns
235
+ // between its first call (precondition check) and its second call
236
+ // (post-listen race guard).
237
+ const initial = createServer();
238
+ initial.on("error", () => {});
239
+ await new Promise<void>((r) => {
240
+ initial.once("listening", () => r());
241
+ initial.listen(socketPath);
242
+ });
243
+
244
+ let getServerCalls = 0;
245
+ const rebinds: Array<{ newServer: Server; oldServer: Server }> = [];
246
+ const spawnedNewServers: Server[] = [];
247
+
248
+ const watchdog = new SocketWatchdog({
249
+ socketPath,
250
+ intervalMs: 0,
251
+ getServer: () => {
252
+ getServerCalls++;
253
+ // First call: precondition — initialServer is still around.
254
+ // Subsequent calls (race guard): null, simulating stop().
255
+ return getServerCalls === 1 ? initial : null;
256
+ },
257
+ createServer: () => {
258
+ const s = createServer();
259
+ s.on("error", () => {});
260
+ spawnedNewServers.push(s);
261
+ return s;
262
+ },
263
+ onRebind: (n, o) => {
264
+ rebinds.push({ newServer: n, oldServer: o });
265
+ },
266
+ log: { info: () => {}, warn: () => {}, error: () => {} },
267
+ });
268
+
269
+ unlinkSync(socketPath);
270
+ expect(existsSync(socketPath)).toBe(false);
271
+
272
+ const rebound = await watchdog.rebindIfMissing();
273
+ expect(rebound).toBe(false);
274
+ expect(rebinds).toHaveLength(0);
275
+ // The race guard should have unlinked the path the discarded server
276
+ // recreated, so a future start() doesn't see a phantom listener.
277
+ expect(existsSync(socketPath)).toBe(false);
278
+ // getServer was called at least twice — once for precondition, once
279
+ // for the race guard.
280
+ expect(getServerCalls).toBeGreaterThanOrEqual(2);
281
+
282
+ // Cleanup: initial is still listening on the unlinked path; close it.
283
+ await closeServer(initial);
284
+ for (const s of spawnedNewServers) {
285
+ try {
286
+ await closeServer(s);
287
+ } catch {
288
+ /* already closed by race guard */
289
+ }
290
+ }
291
+ });
292
+
293
+ test("rebindIfMissing returns false and logs when listen() rejects", async () => {
294
+ // Provide a factory whose listen() always errors, so rebindIfMissing
295
+ // hits the catch branch.
296
+ const initial = createServer();
297
+ initial.on("error", () => {});
298
+ await new Promise<void>((r) => {
299
+ initial.once("listening", () => r());
300
+ initial.listen(socketPath);
301
+ });
302
+
303
+ const rebinds: Array<{ newServer: Server; oldServer: Server }> = [];
304
+ const loggedErrors: Array<{ obj: object; msg?: string }> = [];
305
+ const failingFactory = () => {
306
+ const s = createServer();
307
+ s.on("error", () => {});
308
+ // Replace listen to immediately error.
309
+ const realListen = s.listen.bind(s);
310
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
311
+ (s as any).listen = (_path: string) => {
312
+ queueMicrotask(() => s.emit("error", new Error("simulated EADDRINUSE")));
313
+ return s;
314
+ };
315
+ // Keep realListen reference alive so TS doesn't complain
316
+ void realListen;
317
+ return s;
318
+ };
319
+
320
+ const watchdog = new SocketWatchdog({
321
+ socketPath,
322
+ intervalMs: 0,
323
+ getServer: () => initial,
324
+ createServer: failingFactory,
325
+ onRebind: (n, o) => rebinds.push({ newServer: n, oldServer: o }),
326
+ log: {
327
+ info: () => {},
328
+ warn: () => {},
329
+ error: (obj, msg) => loggedErrors.push({ obj, msg }),
330
+ },
331
+ });
332
+
333
+ unlinkSync(socketPath);
334
+ const rebound = await watchdog.rebindIfMissing();
335
+ expect(rebound).toBe(false);
336
+ expect(rebinds).toHaveLength(0);
337
+ expect(loggedErrors.length).toBeGreaterThan(0);
338
+
339
+ await closeServer(initial);
340
+ });
341
+
342
+ test("watchdog timer catches synchronous rebind errors so unhandled rejections don't escape", async () => {
343
+ // createServer factory throws synchronously — simulates EACCES on
344
+ // mkdir / a broken factory dependency.
345
+ const throwingFactory = () => {
346
+ throw new Error("boom — synchronous factory failure");
347
+ };
348
+
349
+ const initial = createServer();
350
+ initial.on("error", () => {});
351
+ await new Promise<void>((r) => {
352
+ initial.once("listening", () => r());
353
+ initial.listen(socketPath);
354
+ });
355
+
356
+ const loggedErrors: Array<{ obj: object; msg?: string }> = [];
357
+ const watchdog = new SocketWatchdog({
358
+ socketPath,
359
+ intervalMs: 5,
360
+ getServer: () => initial,
361
+ createServer: throwingFactory,
362
+ onRebind: () => {},
363
+ log: {
364
+ info: () => {},
365
+ warn: () => {},
366
+ error: (obj, msg) => loggedErrors.push({ obj, msg }),
367
+ },
368
+ });
369
+
370
+ unlinkSync(socketPath);
371
+
372
+ const seenRejections: unknown[] = [];
373
+ const onRejection = (reason: unknown) => seenRejections.push(reason);
374
+ process.on("unhandledRejection", onRejection);
375
+
376
+ try {
377
+ watchdog.start();
378
+ // Let the timer fire several times.
379
+ await new Promise((r) => setTimeout(r, 30));
380
+ watchdog.stop();
381
+ } finally {
382
+ process.off("unhandledRejection", onRejection);
383
+ }
384
+
385
+ expect(seenRejections).toHaveLength(0);
386
+ expect(loggedErrors.length).toBeGreaterThan(0);
387
+
388
+ await closeServer(initial);
389
+ });
390
+
391
+ test("start() polls and rebinds without manual ticking", async () => {
392
+ harness = buildHarness({ intervalMs: 10 });
393
+ await startInitialServer(harness);
394
+ harness.watchdog.start();
395
+
396
+ unlinkSync(socketPath);
397
+
398
+ // Wait up to 500ms for the timer to recover.
399
+ const deadline = Date.now() + 500;
400
+ while (harness.rebinds.length === 0 && Date.now() < deadline) {
401
+ await new Promise((r) => setTimeout(r, 5));
402
+ }
403
+
404
+ expect(harness.rebinds).toHaveLength(1);
405
+ expect(existsSync(socketPath)).toBe(true);
406
+ });
407
+
408
+ test("stop() prevents future rebinds from firing", async () => {
409
+ harness = buildHarness({ intervalMs: 10 });
410
+ await startInitialServer(harness);
411
+ harness.watchdog.start();
412
+
413
+ // First recovery cycle.
414
+ unlinkSync(socketPath);
415
+ let deadline = Date.now() + 500;
416
+ while (harness.rebinds.length < 1 && Date.now() < deadline) {
417
+ await new Promise((r) => setTimeout(r, 5));
418
+ }
419
+ expect(harness.rebinds).toHaveLength(1);
420
+
421
+ harness.watchdog.stop();
422
+ const stoppedAt = harness.rebinds.length;
423
+
424
+ // Unlink again. Wait three intervals; no new rebind should appear.
425
+ unlinkSync(socketPath);
426
+ await new Promise((r) => setTimeout(r, 50));
427
+ expect(harness.rebinds).toHaveLength(stoppedAt);
428
+ expect(existsSync(socketPath)).toBe(false);
429
+ });
430
+ });
@@ -0,0 +1,221 @@
1
+ /**
2
+ * Resilience helper for Unix-domain-socket IPC servers: re-binds the
3
+ * listening socket when its on-disk path entry has been removed (e.g. by a
4
+ * tmpfs sweep or rogue cleanup of `/run/*`).
5
+ *
6
+ * Existing connected sockets survive the re-bind because the kernel keeps
7
+ * connection inodes alive independently of the listener path; only new
8
+ * `connect()` calls require the path to exist.
9
+ *
10
+ * Consumers wire their `Server` reference into the watchdog via callbacks
11
+ * rather than passing the server directly so the watchdog can guard against
12
+ * shutdown/restart races mid-rebind.
13
+ */
14
+
15
+ import { existsSync, mkdirSync, unlinkSync } from "node:fs";
16
+ import type { Server } from "node:net";
17
+ import { dirname } from "node:path";
18
+
19
+ /**
20
+ * Minimal logger surface (pino-compatible). Each method receives a context
21
+ * object plus an optional human-readable message.
22
+ */
23
+ export interface SocketWatchdogLogger {
24
+ info(obj: object, msg?: string): void;
25
+ warn(obj: object, msg?: string): void;
26
+ error(obj: object, msg?: string): void;
27
+ }
28
+
29
+ export interface SocketWatchdogOptions {
30
+ /** Absolute path to the Unix socket file the consumer is listening on. */
31
+ socketPath: string;
32
+ /**
33
+ * How often to stat the socket path. Set to `0` to disable. Defaults to
34
+ * 5000ms.
35
+ */
36
+ intervalMs?: number;
37
+ /**
38
+ * Returns the consumer's current listening server. The watchdog uses this
39
+ * both as a precondition (no rebind when null) and as a generation marker
40
+ * to detect shutdown/restart races mid-rebind.
41
+ */
42
+ getServer: () => Server | null;
43
+ /**
44
+ * Factory for a fresh listening Server. Called by the watchdog when a
45
+ * rebind is needed; the watchdog drives `.listen(socketPath)` and waits
46
+ * for the `listening` event before installing.
47
+ */
48
+ createServer: () => Server;
49
+ /**
50
+ * Invoked when a rebind succeeds. The consumer is responsible for
51
+ * swapping its primary server reference to `newServer` and disposing of
52
+ * `oldServer` (typically by tracking it as a legacy listener while
53
+ * in-flight clients drain, then closing it).
54
+ */
55
+ onRebind: (newServer: Server, oldServer: Server) => void;
56
+ /** Pino-compatible logger. */
57
+ log: SocketWatchdogLogger;
58
+ }
59
+
60
+ const DEFAULT_INTERVAL_MS = 5000;
61
+
62
+ /**
63
+ * Ensure the directory containing `socketPath` exists. Created with mode
64
+ * `0o700` so a freshly-spawned dir on a tmpfs mount doesn't leak the IPC
65
+ * surface to other UIDs. Existing directories keep their permissions —
66
+ * `mkdir` only applies the mode to directories it creates.
67
+ */
68
+ export function ensureSocketDir(socketPath: string): void {
69
+ const socketDir = dirname(socketPath);
70
+ if (!existsSync(socketDir)) {
71
+ mkdirSync(socketDir, { recursive: true, mode: 0o700 });
72
+ }
73
+ }
74
+
75
+ /**
76
+ * Watchdog that periodically stats a Unix socket file and re-binds the
77
+ * listener when the path has been removed.
78
+ *
79
+ * Lifecycle:
80
+ * - Construct with the consumer's callbacks.
81
+ * - Call {@link start} after the consumer's initial `listen()` succeeds.
82
+ * - Call {@link stop} during shutdown (before closing the underlying
83
+ * server) so an in-flight rebind doesn't resurrect the listener.
84
+ *
85
+ * The watchdog timer is `unref`-ed so it never keeps the event loop alive
86
+ * on its own.
87
+ */
88
+ export class SocketWatchdog {
89
+ private readonly socketPath: string;
90
+ private readonly intervalMs: number;
91
+ private readonly getServer: () => Server | null;
92
+ private readonly createServer: () => Server;
93
+ private readonly onRebind: (newServer: Server, oldServer: Server) => void;
94
+ private readonly log: SocketWatchdogLogger;
95
+
96
+ private handle: ReturnType<typeof setInterval> | null = null;
97
+
98
+ constructor(options: SocketWatchdogOptions) {
99
+ this.socketPath = options.socketPath;
100
+ this.intervalMs = options.intervalMs ?? DEFAULT_INTERVAL_MS;
101
+ this.getServer = options.getServer;
102
+ this.createServer = options.createServer;
103
+ this.onRebind = options.onRebind;
104
+ this.log = options.log;
105
+ }
106
+
107
+ /**
108
+ * Begin polling the socket path. No-op if `intervalMs <= 0` or the
109
+ * watchdog is already running.
110
+ */
111
+ start(): void {
112
+ if (this.intervalMs <= 0 || this.handle !== null) return;
113
+ this.handle = setInterval(() => {
114
+ // The async entry path of rebindIfMissing performs filesystem work
115
+ // (ensureSocketDir, createServer) before its inner try/catch, so a
116
+ // synchronous throw — e.g. EACCES on a read-only fs — would surface
117
+ // as an unhandled rejection on every tick. Catch here so the timer
118
+ // stays quiet on persistent failure modes.
119
+ this.rebindIfMissing().catch((err) => {
120
+ this.log.error(
121
+ { err, path: this.socketPath },
122
+ "Watchdog rebind failed unexpectedly",
123
+ );
124
+ });
125
+ }, this.intervalMs);
126
+ this.handle.unref?.();
127
+ }
128
+
129
+ /** Stop the polling timer. Safe to call multiple times. */
130
+ stop(): void {
131
+ if (this.handle !== null) {
132
+ clearInterval(this.handle);
133
+ this.handle = null;
134
+ }
135
+ }
136
+
137
+ /**
138
+ * Re-bind the listening socket if its path entry is missing on disk.
139
+ *
140
+ * Public for tests so the watchdog can be exercised deterministically
141
+ * without waiting for the interval. Returns `true` when a re-bind was
142
+ * performed, `false` when the socket was already healthy, the consumer
143
+ * is not running, or a shutdown/restart raced the rebind.
144
+ */
145
+ async rebindIfMissing(): Promise<boolean> {
146
+ const initialServer = this.getServer();
147
+ if (initialServer === null) return false;
148
+ if (existsSync(this.socketPath)) return false;
149
+
150
+ this.log.warn(
151
+ { path: this.socketPath },
152
+ "IPC socket path missing on disk — re-binding listener",
153
+ );
154
+
155
+ ensureSocketDir(this.socketPath);
156
+
157
+ const newServer = this.createServer();
158
+ try {
159
+ await new Promise<void>((resolve, reject) => {
160
+ const onError = (err: unknown) => {
161
+ newServer.off("listening", onListening);
162
+ reject(err);
163
+ };
164
+ const onListening = () => {
165
+ newServer.off("error", onError);
166
+ resolve();
167
+ };
168
+ newServer.once("error", onError);
169
+ newServer.once("listening", onListening);
170
+ newServer.listen(this.socketPath);
171
+ });
172
+ } catch (err) {
173
+ this.log.error(
174
+ { err, path: this.socketPath },
175
+ "Failed to re-bind IPC socket — will retry on next watchdog tick",
176
+ );
177
+ try {
178
+ newServer.close();
179
+ } catch {
180
+ /* ignore */
181
+ }
182
+ return false;
183
+ }
184
+
185
+ // Race guard: while we were awaiting listen(), the consumer may have
186
+ // stopped, restarted, or otherwise replaced its server reference.
187
+ // Installing newServer would resurrect a listener after shutdown
188
+ // (keeping the process alive and accepting IPC again). Discard the
189
+ // new server instead.
190
+ if (this.getServer() !== initialServer) {
191
+ try {
192
+ newServer.close();
193
+ } catch {
194
+ /* ignore */
195
+ }
196
+ // newServer.listen() recreated the path on disk. If our listen won
197
+ // the race, the file is sitting there — clean it up so it doesn't
198
+ // shadow a future start().
199
+ if (existsSync(this.socketPath)) {
200
+ try {
201
+ unlinkSync(this.socketPath);
202
+ } catch {
203
+ /* ignore */
204
+ }
205
+ }
206
+ this.log.warn(
207
+ { path: this.socketPath },
208
+ "IPC server state changed during rebind — discarded new listener",
209
+ );
210
+ return false;
211
+ }
212
+
213
+ this.onRebind(newServer, initialServer);
214
+
215
+ this.log.info(
216
+ { path: this.socketPath },
217
+ "IPC socket re-bound after path loss",
218
+ );
219
+ return true;
220
+ }
221
+ }
@@ -0,0 +1,20 @@
1
+ {
2
+ "compilerOptions": {
3
+ "target": "ES2022",
4
+ "module": "NodeNext",
5
+ "moduleResolution": "NodeNext",
6
+ "strict": true,
7
+ "esModuleInterop": true,
8
+ "skipLibCheck": true,
9
+ "forceConsistentCasingInFileNames": true,
10
+ "resolveJsonModule": true,
11
+ "declaration": true,
12
+ "declarationMap": true,
13
+ "sourceMap": true,
14
+ "outDir": "./dist",
15
+ "rootDir": "./src",
16
+ "types": ["bun-types"]
17
+ },
18
+ "include": ["src/**/*"],
19
+ "exclude": ["node_modules", "dist"]
20
+ }
@@ -84,12 +84,3 @@ export function formatSseFrame(event: AssistantEvent): string {
84
84
  export function formatSseHeartbeat(): string {
85
85
  return ": heartbeat\n\n";
86
86
  }
87
-
88
- /**
89
- * Format a keep-alive as both an SSE comment (for proxy keepalive) and a
90
- * data-bearing event (so fetch-based SSE clients that cannot observe comment
91
- * lines can still detect heartbeats for disconnect watchdogs).
92
- */
93
- export function formatSseHeartbeatWithData(): string {
94
- return `${formatSseHeartbeat()}event: assistant_event\ndata: ${JSON.stringify({ type: "heartbeat" })}\n\n`;
95
- }