@vellumai/assistant 0.5.0 → 0.5.2

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 (347) hide show
  1. package/ARCHITECTURE.md +54 -54
  2. package/docs/architecture/integrations.md +62 -67
  3. package/docs/credential-execution-service.md +3 -3
  4. package/package.json +1 -1
  5. package/src/__tests__/agent-loop.test.ts +111 -0
  6. package/src/__tests__/always-loaded-tools-guard.test.ts +3 -4
  7. package/src/__tests__/app-builder-tool-scripts.test.ts +13 -151
  8. package/src/__tests__/app-dir-path-guard.test.ts +78 -0
  9. package/src/__tests__/app-executors.test.ts +1 -291
  10. package/src/__tests__/app-git-history.test.ts +4 -4
  11. package/src/__tests__/app-routes-csp.test.ts +1 -0
  12. package/src/__tests__/app-store-dir-names.test.ts +426 -0
  13. package/src/__tests__/assistant-feature-flags-integration.test.ts +7 -9
  14. package/src/__tests__/attachments-store.test.ts +169 -21
  15. package/src/__tests__/attachments.test.ts +115 -1
  16. package/src/__tests__/btw-routes.test.ts +1 -0
  17. package/src/__tests__/canonical-guardian-store.test.ts +38 -0
  18. package/src/__tests__/channel-reply-delivery.test.ts +55 -0
  19. package/src/__tests__/checker.test.ts +54 -0
  20. package/src/__tests__/claude-code-skill-regression.test.ts +2 -0
  21. package/src/__tests__/claude-code-tool-profiles.test.ts +2 -0
  22. package/src/__tests__/compaction.benchmark.test.ts +2 -1
  23. package/src/__tests__/config-schema-cmd.test.ts +68 -21
  24. package/src/__tests__/config-schema.test.ts +1 -1
  25. package/src/__tests__/conversation-agent-loop-overflow.test.ts +149 -5
  26. package/src/__tests__/conversation-agent-loop.test.ts +290 -2
  27. package/src/__tests__/conversation-attachments.test.ts +17 -19
  28. package/src/__tests__/conversation-disk-view-integration.test.ts +277 -0
  29. package/src/__tests__/conversation-disk-view.test.ts +810 -0
  30. package/src/__tests__/conversation-error.test.ts +1 -1
  31. package/src/__tests__/conversation-fork-crud.test.ts +551 -0
  32. package/src/__tests__/conversation-fork-route.test.ts +386 -0
  33. package/src/__tests__/conversation-history-web-search.test.ts +1 -1
  34. package/src/__tests__/conversation-key-store-disk-view.test.ts +130 -0
  35. package/src/__tests__/conversation-media-retry.test.ts +8 -2
  36. package/src/__tests__/conversation-queue.test.ts +36 -1
  37. package/src/__tests__/conversation-routes-disk-view.test.ts +439 -0
  38. package/src/__tests__/conversation-routes-guardian-reply.test.ts +2 -2
  39. package/src/__tests__/conversation-routes-slash-commands.test.ts +2 -7
  40. package/src/__tests__/conversation-runtime-assembly.test.ts +17 -2
  41. package/src/__tests__/conversation-skill-tools.test.ts +4 -9
  42. package/src/__tests__/conversation-slash-commands.test.ts +149 -0
  43. package/src/__tests__/conversation-store.test.ts +24 -21
  44. package/src/__tests__/conversation-surfaces-state-update.test.ts +246 -0
  45. package/src/__tests__/conversation-surfaces-task-progress.test.ts +1 -0
  46. package/src/__tests__/conversation-title-service.test.ts +137 -0
  47. package/src/__tests__/conversation-tool-setup-app-refresh.test.ts +25 -315
  48. package/src/__tests__/conversation-tool-setup-memory-scope.test.ts +1 -0
  49. package/src/__tests__/conversation-tool-setup-side-effect-flag.test.ts +1 -0
  50. package/src/__tests__/conversation-workspace-cache-state.test.ts +44 -2
  51. package/src/__tests__/conversation-workspace-injection.test.ts +11 -0
  52. package/src/__tests__/credential-execution-feature-gates.test.ts +3 -3
  53. package/src/__tests__/credential-security-invariants.test.ts +3 -0
  54. package/src/__tests__/credential-vault-unit.test.ts +5 -10
  55. package/src/__tests__/cu-unified-flow.test.ts +1 -0
  56. package/src/__tests__/db-conversation-fork-lineage-migration.test.ts +241 -0
  57. package/src/__tests__/db-llm-request-log-provider-migration.test.ts +214 -0
  58. package/src/__tests__/diagnostics-export.test.ts +70 -1
  59. package/src/__tests__/filesystem-tools.test.ts +4 -2
  60. package/src/__tests__/first-greeting.test.ts +80 -0
  61. package/src/__tests__/gateway-only-guard.test.ts +1 -0
  62. package/src/__tests__/handlers-user-message-approval-consumption.test.ts +3 -7
  63. package/src/__tests__/history-repair.test.ts +103 -10
  64. package/src/__tests__/http-conversation-lineage.test.ts +251 -0
  65. package/src/__tests__/image-source-path-reinject.test.ts +136 -0
  66. package/src/__tests__/llm-context-normalization.test.ts +1116 -0
  67. package/src/__tests__/llm-context-route-provider.test.ts +217 -0
  68. package/src/__tests__/llm-request-log-turn-query.test.ts +270 -0
  69. package/src/__tests__/media-generate-image.test.ts +47 -94
  70. package/src/__tests__/memory-lifecycle-e2e.test.ts +3 -1
  71. package/src/__tests__/memory-recall-quality.test.ts +5 -5
  72. package/src/__tests__/migration-cross-version-compatibility.test.ts +4 -1
  73. package/src/__tests__/migration-export-http.test.ts +3 -1
  74. package/src/__tests__/migration-import-commit-http.test.ts +18 -4
  75. package/src/__tests__/migration-import-preflight-http.test.ts +1 -3
  76. package/src/__tests__/mime-builder.test.ts +3 -2
  77. package/src/__tests__/non-member-access-request.test.ts +12 -1
  78. package/src/__tests__/notification-decision-identity.test.ts +52 -0
  79. package/src/__tests__/oauth-apps-routes.test.ts +103 -0
  80. package/src/__tests__/oauth-store.test.ts +115 -0
  81. package/src/__tests__/provider-error-scenarios.test.ts +1 -3
  82. package/src/__tests__/provider-failover-actual-provider.test.ts +66 -0
  83. package/src/__tests__/recording-handler.test.ts +17 -0
  84. package/src/__tests__/registry.test.ts +3 -8
  85. package/src/__tests__/relay-server.test.ts +1 -1
  86. package/src/__tests__/runtime-attachment-metadata.test.ts +7 -3
  87. package/src/__tests__/schema-transforms.test.ts +165 -5
  88. package/src/__tests__/server-history-render.test.ts +2 -2
  89. package/src/__tests__/skill-feature-flags-integration.test.ts +18 -17
  90. package/src/__tests__/skill-feature-flags.test.ts +13 -13
  91. package/src/__tests__/skill-load-feature-flag.test.ts +4 -4
  92. package/src/__tests__/slack-app-setup-skill-regression.test.ts +3 -1
  93. package/src/__tests__/slack-inbound-verification.test.ts +2 -2
  94. package/src/__tests__/starter-task-flow.test.ts +1 -0
  95. package/src/__tests__/suggestion-routes.test.ts +443 -0
  96. package/src/__tests__/swarm-conversation-integration.test.ts +1 -0
  97. package/src/__tests__/swarm-recursion.test.ts +1 -0
  98. package/src/__tests__/swarm-tool.test.ts +1 -0
  99. package/src/__tests__/system-prompt.test.ts +8 -0
  100. package/src/__tests__/tool-execution-abort-cleanup.test.ts +1 -0
  101. package/src/__tests__/tool-preview-lifecycle.test.ts +32 -5
  102. package/src/__tests__/top-level-renderer.test.ts +22 -0
  103. package/src/__tests__/turn-boundary-resolution.test.ts +243 -0
  104. package/src/__tests__/web-fetch.test.ts +6 -2
  105. package/src/__tests__/workspace-migration-006-services-config.test.ts +335 -0
  106. package/src/__tests__/workspace-migration-007-web-search-provider-rename.test.ts +312 -0
  107. package/src/__tests__/workspace-migration-009-backfill-conversation-disk-view.test.ts +278 -0
  108. package/src/__tests__/workspace-migration-010-app-dir-rename.test.ts +275 -0
  109. package/src/__tests__/workspace-migration-012-rename-conversation-disk-view-dirs.test.ts +77 -0
  110. package/src/__tests__/workspace-migration-013-repair-conversation-disk-view.test.ts +401 -0
  111. package/src/__tests__/workspace-migration-backfill-installation-id.test.ts +328 -0
  112. package/src/__tests__/workspace-migration-seed-device-id.test.ts +6 -10
  113. package/src/agent/attachments.ts +27 -1
  114. package/src/agent/loop.ts +29 -1
  115. package/src/avatar/traits-png-sync.ts +80 -25
  116. package/src/bundler/app-bundler.ts +4 -4
  117. package/src/calls/call-domain.ts +1 -0
  118. package/src/calls/voice-session-bridge.ts +1 -0
  119. package/src/cli/commands/auth.ts +92 -0
  120. package/src/cli/commands/avatar.ts +7 -6
  121. package/src/cli/commands/config.ts +2 -0
  122. package/src/cli/commands/oauth/providers.ts +29 -0
  123. package/src/cli/program.ts +12 -0
  124. package/src/cli.ts +15 -48
  125. package/src/config/bundled-skills/app-builder/SKILL.md +103 -28
  126. package/src/config/bundled-skills/app-builder/TOOLS.json +5 -199
  127. package/src/config/bundled-skills/app-builder/tools/{app-query.ts → app-refresh.ts} +2 -2
  128. package/src/config/bundled-skills/contacts/tools/google-contacts.ts +2 -3
  129. package/src/config/bundled-skills/gmail/tools/gmail-archive.ts +6 -9
  130. package/src/config/bundled-skills/gmail/tools/gmail-attachments.ts +4 -6
  131. package/src/config/bundled-skills/gmail/tools/gmail-draft.ts +2 -3
  132. package/src/config/bundled-skills/gmail/tools/gmail-filters.ts +2 -3
  133. package/src/config/bundled-skills/gmail/tools/gmail-follow-up.ts +2 -3
  134. package/src/config/bundled-skills/gmail/tools/gmail-forward.ts +2 -3
  135. package/src/config/bundled-skills/gmail/tools/gmail-label.ts +4 -6
  136. package/src/config/bundled-skills/gmail/tools/gmail-outreach-scan.ts +2 -3
  137. package/src/config/bundled-skills/gmail/tools/gmail-send-draft.ts +2 -3
  138. package/src/config/bundled-skills/gmail/tools/gmail-sender-digest.ts +2 -3
  139. package/src/config/bundled-skills/gmail/tools/gmail-trash.ts +2 -3
  140. package/src/config/bundled-skills/gmail/tools/gmail-unsubscribe.ts +2 -3
  141. package/src/config/bundled-skills/gmail/tools/gmail-vacation.ts +2 -3
  142. package/src/config/bundled-skills/google-calendar/tools/shared.ts +1 -1
  143. package/src/config/bundled-skills/image-studio/SKILL.md +2 -2
  144. package/src/config/bundled-skills/image-studio/TOOLS.json +2 -2
  145. package/src/config/bundled-skills/image-studio/tools/media-generate-image.ts +45 -72
  146. package/src/config/bundled-skills/media-processing/tools/extract-keyframes.ts +2 -2
  147. package/src/config/bundled-skills/messaging/tools/shared.ts +1 -1
  148. package/src/config/bundled-skills/settings/tools/voice-config-update.ts +19 -3
  149. package/src/config/bundled-skills/skill-management/TOOLS.json +2 -2
  150. package/src/config/bundled-skills/slack/tools/shared.ts +19 -4
  151. package/src/config/bundled-skills/slack/tools/slack-scan-digest.ts +2 -3
  152. package/src/config/bundled-skills/transcribe/SKILL.md +1 -1
  153. package/src/config/bundled-skills/transcribe/TOOLS.json +2 -6
  154. package/src/config/bundled-skills/transcribe/tools/transcribe-media.ts +19 -83
  155. package/src/config/bundled-tool-registry.ts +2 -14
  156. package/src/config/feature-flag-registry.json +16 -0
  157. package/src/config/loader.ts +64 -0
  158. package/src/config/raw-config-utils.ts +30 -0
  159. package/src/config/schema-utils.ts +28 -7
  160. package/src/config/schema.ts +8 -0
  161. package/src/config/schemas/elevenlabs.ts +18 -0
  162. package/src/config/schemas/memory-lifecycle.ts +4 -2
  163. package/src/config/schemas/memory-storage.ts +1 -1
  164. package/src/config/schemas/services.ts +8 -6
  165. package/src/contacts/contact-store.ts +13 -6
  166. package/src/contacts/contacts-write.ts +0 -1
  167. package/src/context/window-manager.ts +13 -2
  168. package/src/daemon/conversation-agent-loop-handlers.ts +46 -42
  169. package/src/daemon/conversation-agent-loop.ts +56 -19
  170. package/src/daemon/conversation-attachments.ts +18 -36
  171. package/src/daemon/conversation-error.ts +2 -1
  172. package/src/daemon/conversation-history.ts +18 -4
  173. package/src/daemon/conversation-lifecycle.ts +39 -15
  174. package/src/daemon/conversation-messaging.ts +70 -26
  175. package/src/daemon/conversation-process.ts +58 -34
  176. package/src/daemon/conversation-runtime-assembly.ts +21 -38
  177. package/src/daemon/conversation-slash.ts +121 -256
  178. package/src/daemon/conversation-surfaces.ts +143 -20
  179. package/src/daemon/conversation-tool-setup.ts +0 -6
  180. package/src/daemon/conversation-workspace.ts +21 -1
  181. package/src/daemon/conversation.ts +51 -29
  182. package/src/daemon/first-greeting.ts +35 -0
  183. package/src/daemon/handlers/config-embeddings.ts +148 -0
  184. package/src/daemon/handlers/config-model.ts +71 -26
  185. package/src/daemon/handlers/conversations.ts +0 -23
  186. package/src/daemon/handlers/recording.ts +26 -21
  187. package/src/daemon/history-repair.ts +28 -8
  188. package/src/daemon/host-cu-proxy.ts +2 -2
  189. package/src/daemon/lifecycle.ts +106 -64
  190. package/src/daemon/message-protocol.ts +3 -0
  191. package/src/daemon/message-types/conversations.ts +19 -0
  192. package/src/daemon/message-types/messages.ts +1 -0
  193. package/src/daemon/message-types/shared.ts +2 -0
  194. package/src/daemon/message-types/surfaces.ts +2 -0
  195. package/src/daemon/message-types/upgrades.ts +23 -0
  196. package/src/daemon/server.ts +83 -12
  197. package/src/daemon/shutdown-handlers.ts +8 -5
  198. package/src/daemon/startup-error.ts +9 -0
  199. package/src/daemon/tool-side-effects.ts +11 -28
  200. package/src/events/tool-permission-telemetry-listener.ts +1 -3
  201. package/src/instrument.ts +0 -4
  202. package/src/media/app-icon-generator.ts +2 -2
  203. package/src/memory/app-git-service.ts +28 -16
  204. package/src/memory/app-store.ts +230 -41
  205. package/src/memory/attachments-store.ts +558 -130
  206. package/src/memory/conversation-attention-store.ts +70 -0
  207. package/src/memory/conversation-crud.ts +442 -3
  208. package/src/memory/conversation-directories.ts +125 -0
  209. package/src/memory/conversation-disk-view.ts +390 -0
  210. package/src/memory/conversation-key-store.ts +17 -5
  211. package/src/memory/conversation-queries.ts +5 -1
  212. package/src/memory/conversation-title-service.ts +21 -49
  213. package/src/memory/db-init.ts +28 -0
  214. package/src/memory/embedding-backend.ts +42 -53
  215. package/src/memory/embedding-gemini.test.ts +4 -4
  216. package/src/memory/embedding-local.ts +1 -3
  217. package/src/memory/embedding-ollama.ts +1 -3
  218. package/src/memory/embedding-openai.ts +1 -3
  219. package/src/memory/indexer.ts +9 -7
  220. package/src/memory/items-extractor.ts +42 -13
  221. package/src/memory/job-handlers/conversation-starters.ts +6 -1
  222. package/src/memory/job-handlers/embedding.test.ts +1 -4
  223. package/src/memory/llm-request-log-store.ts +100 -1
  224. package/src/memory/migrations/102-alter-table-columns.ts +5 -0
  225. package/src/memory/migrations/146-schedule-oneshot-routing.ts +3 -3
  226. package/src/memory/migrations/147-migrate-reminders-to-schedules.ts +66 -70
  227. package/src/memory/migrations/148-drop-reminders-table.ts +5 -9
  228. package/src/memory/migrations/160-drop-loopback-port-column.ts +1 -3
  229. package/src/memory/migrations/174-rename-thread-starters-table.ts +0 -7
  230. package/src/memory/migrations/178-oauth-providers-managed-service-config-key.ts +15 -0
  231. package/src/memory/migrations/179-llm-request-log-message-id.ts +16 -0
  232. package/src/memory/migrations/180-backfill-inline-attachments-to-disk.ts +66 -0
  233. package/src/memory/migrations/181-rename-thread-starters-checkpoints.ts +46 -0
  234. package/src/memory/migrations/182-oauth-providers-display-metadata.ts +20 -0
  235. package/src/memory/migrations/183-add-conversation-fork-lineage.ts +22 -0
  236. package/src/memory/migrations/184-llm-request-log-provider.ts +12 -0
  237. package/src/memory/migrations/index.ts +7 -0
  238. package/src/memory/migrations/registry.ts +13 -0
  239. package/src/memory/retriever.test.ts +601 -2
  240. package/src/memory/retriever.ts +85 -9
  241. package/src/memory/schema/conversations.ts +6 -0
  242. package/src/memory/schema/infrastructure.ts +13 -7
  243. package/src/memory/schema/oauth.ts +6 -0
  244. package/src/messaging/providers/gmail/mime-builder.ts +3 -1
  245. package/src/notifications/copy-composer.ts +26 -0
  246. package/src/notifications/decision-engine.ts +14 -1
  247. package/src/notifications/emit-signal.ts +1 -1
  248. package/src/notifications/signal.ts +36 -0
  249. package/src/oauth/byo-connection.test.ts +1 -45
  250. package/src/oauth/byo-connection.ts +2 -8
  251. package/src/oauth/connect-orchestrator.ts +15 -11
  252. package/src/oauth/connection-resolver.test.ts +191 -0
  253. package/src/oauth/connection-resolver.ts +66 -38
  254. package/src/oauth/connection.ts +0 -1
  255. package/src/oauth/oauth-store.ts +97 -47
  256. package/src/oauth/platform-connection.test.ts +0 -1
  257. package/src/oauth/platform-connection.ts +11 -3
  258. package/src/oauth/seed-providers.ts +78 -3
  259. package/src/oauth/token-persistence.ts +16 -10
  260. package/src/permissions/checker.ts +62 -19
  261. package/src/prompts/system-prompt.ts +2 -0
  262. package/src/prompts/templates/BOOTSTRAP.md +2 -0
  263. package/src/providers/anthropic/client.ts +8 -1
  264. package/src/providers/failover.ts +4 -1
  265. package/src/providers/gemini/client.ts +50 -0
  266. package/src/providers/model-catalog.ts +92 -0
  267. package/src/providers/model-intents.ts +29 -20
  268. package/src/providers/openai/client.ts +49 -0
  269. package/src/providers/types.ts +2 -0
  270. package/src/runtime/access-request-helper.ts +16 -7
  271. package/src/runtime/auth/credential-service.ts +3 -1
  272. package/src/runtime/auth/route-policy.ts +14 -1
  273. package/src/runtime/btw-sidechain.ts +101 -0
  274. package/src/runtime/channel-reply-delivery.ts +17 -1
  275. package/src/runtime/http-router.ts +3 -1
  276. package/src/runtime/http-server.ts +196 -141
  277. package/src/runtime/http-types.ts +1 -0
  278. package/src/runtime/migrations/vbundle-builder.ts +5 -1
  279. package/src/runtime/routes/access-request-decision.ts +41 -0
  280. package/src/runtime/routes/app-management-routes.ts +6 -3
  281. package/src/runtime/routes/app-routes.ts +7 -3
  282. package/src/runtime/routes/approval-routes.ts +1 -0
  283. package/src/runtime/routes/approval-strategies/guardian-callback-strategy.ts +34 -2
  284. package/src/runtime/routes/attachment-routes.ts +45 -15
  285. package/src/runtime/routes/btw-routes.ts +21 -61
  286. package/src/runtime/routes/conversation-management-routes.ts +68 -0
  287. package/src/runtime/routes/conversation-query-routes.ts +180 -10
  288. package/src/runtime/routes/conversation-routes.ts +222 -28
  289. package/src/runtime/routes/conversation-starter-routes.ts +9 -11
  290. package/src/runtime/routes/diagnostics-routes.ts +1 -0
  291. package/src/runtime/routes/inbound-stages/acl-enforcement.ts +2 -2
  292. package/src/runtime/routes/llm-context-normalization.ts +1199 -0
  293. package/src/runtime/routes/log-export-routes.ts +3 -0
  294. package/src/runtime/routes/memory-item-routes.test.ts +34 -0
  295. package/src/runtime/routes/memory-item-routes.ts +4 -0
  296. package/src/runtime/routes/migration-routes.ts +4 -1
  297. package/src/runtime/routes/oauth-apps.ts +291 -0
  298. package/src/runtime/routes/secret-routes.ts +28 -1
  299. package/src/runtime/routes/settings-routes.ts +14 -0
  300. package/src/runtime/routes/trace-event-routes.ts +4 -1
  301. package/src/schedule/schedule-store.ts +9 -21
  302. package/src/security/secure-keys.ts +21 -0
  303. package/src/signals/bash.ts +1 -1
  304. package/src/swarm/backend-claude-code.ts +3 -6
  305. package/src/telemetry/usage-telemetry-reporter.test.ts +3 -2
  306. package/src/telemetry/usage-telemetry-reporter.ts +3 -1
  307. package/src/tools/AGENTS.md +6 -10
  308. package/src/tools/apps/executors.ts +17 -232
  309. package/src/tools/claude-code/claude-code.ts +2 -3
  310. package/src/tools/credentials/vault.ts +7 -12
  311. package/src/tools/host-filesystem/read.ts +13 -10
  312. package/src/tools/network/__tests__/web-search.test.ts +4 -2
  313. package/src/tools/schedule/list.ts +2 -7
  314. package/src/tools/schema-transforms.ts +5 -0
  315. package/src/tools/shared/filesystem/format-diff.ts +4 -21
  316. package/src/tools/skills/execute.ts +1 -1
  317. package/src/tools/tool-manifest.ts +0 -6
  318. package/src/tools/ui-surface/definitions.ts +2 -2
  319. package/src/util/device-id.ts +28 -5
  320. package/src/util/platform.ts +6 -0
  321. package/src/util/pricing.ts +1 -0
  322. package/src/util/retry.ts +1 -3
  323. package/src/workspace/migrations/002-backfill-installation-id.ts +23 -12
  324. package/src/workspace/migrations/003-seed-device-id.ts +3 -4
  325. package/src/workspace/migrations/006-services-config.ts +5 -0
  326. package/src/workspace/migrations/008-voice-timeout-and-max-steps.ts +12 -0
  327. package/src/workspace/migrations/009-backfill-conversation-disk-view.ts +10 -0
  328. package/src/workspace/migrations/010-app-dir-rename.ts +223 -0
  329. package/src/workspace/migrations/012-rename-conversation-disk-view-dirs.ts +64 -0
  330. package/src/workspace/migrations/013-repair-conversation-disk-view.ts +11 -0
  331. package/src/workspace/migrations/rebuild-conversation-disk-view.ts +186 -0
  332. package/src/workspace/migrations/registry.ts +10 -0
  333. package/src/workspace/top-level-renderer.ts +12 -0
  334. package/src/__tests__/asset-materialize-tool.test.ts +0 -523
  335. package/src/__tests__/asset-search-tool.test.ts +0 -536
  336. package/src/__tests__/fixtures/media-reuse-fixtures.ts +0 -56
  337. package/src/__tests__/media-reuse-story.e2e.test.ts +0 -762
  338. package/src/__tests__/media-visibility-policy.test.ts +0 -190
  339. package/src/config/bundled-skills/app-builder/tools/app-file-edit.ts +0 -14
  340. package/src/config/bundled-skills/app-builder/tools/app-file-list.ts +0 -13
  341. package/src/config/bundled-skills/app-builder/tools/app-file-read.ts +0 -21
  342. package/src/config/bundled-skills/app-builder/tools/app-file-write.ts +0 -14
  343. package/src/config/bundled-skills/app-builder/tools/app-list.ts +0 -13
  344. package/src/config/bundled-skills/app-builder/tools/app-update.ts +0 -23
  345. package/src/daemon/media-visibility-policy.ts +0 -59
  346. package/src/tools/assets/materialize.ts +0 -248
  347. package/src/tools/assets/search.ts +0 -400
@@ -0,0 +1,810 @@
1
+ import {
2
+ existsSync,
3
+ mkdirSync,
4
+ mkdtempSync,
5
+ readFileSync,
6
+ rmSync,
7
+ writeFileSync,
8
+ } from "node:fs";
9
+ import { tmpdir } from "node:os";
10
+ import { join } from "node:path";
11
+ import { afterAll, beforeEach, describe, expect, mock, test } from "bun:test";
12
+
13
+ // ---------------------------------------------------------------------------
14
+ // Mocks — must come before any imports that depend on them
15
+ // ---------------------------------------------------------------------------
16
+
17
+ const testDir = mkdtempSync(join(tmpdir(), "conv-disk-view-test-"));
18
+ const workspaceDir = join(testDir, "workspace");
19
+ const conversationsDir = join(workspaceDir, "conversations");
20
+ mkdirSync(conversationsDir, { recursive: true });
21
+
22
+ mock.module("../util/platform.js", () => ({
23
+ getDataDir: () => join(workspaceDir, "data"),
24
+ getWorkspaceDir: () => workspaceDir,
25
+ getConversationsDir: () => conversationsDir,
26
+ isMacOS: () => process.platform === "darwin",
27
+ isLinux: () => process.platform === "linux",
28
+ isWindows: () => process.platform === "win32",
29
+ getPidPath: () => join(testDir, "test.pid"),
30
+ getDbPath: () => join(testDir, "test.db"),
31
+ getLogPath: () => join(testDir, "test.log"),
32
+ ensureDataDir: () => {},
33
+ getRootDir: () => testDir,
34
+ }));
35
+
36
+ mock.module("../util/logger.js", () => ({
37
+ getLogger: () =>
38
+ new Proxy({} as Record<string, unknown>, {
39
+ get: () => () => {},
40
+ }),
41
+ }));
42
+
43
+ mock.module("../config/loader.js", () => ({
44
+ getConfig: () => ({
45
+ ui: {},
46
+ model: "test",
47
+ provider: "test",
48
+ memory: { enabled: false },
49
+ rateLimit: { maxRequestsPerMinute: 0 },
50
+ }),
51
+ }));
52
+
53
+ // ---------------------------------------------------------------------------
54
+ // Imports — after mocks
55
+ // ---------------------------------------------------------------------------
56
+
57
+ import {
58
+ linkAttachmentToMessage,
59
+ uploadAttachment,
60
+ } from "../memory/attachments-store.js";
61
+ import {
62
+ addMessage,
63
+ createConversation,
64
+ deleteMessageById,
65
+ relinkAttachments,
66
+ updateMessageContent,
67
+ } from "../memory/conversation-crud.js";
68
+ import {
69
+ flattenContentBlocks,
70
+ getConversationDirName,
71
+ getConversationDirPath,
72
+ initConversationDir,
73
+ rebuildConversationDiskViewFromDbState,
74
+ removeConversationDir,
75
+ resolveUniqueFilename,
76
+ syncMessageToDisk,
77
+ updateMetaFile,
78
+ } from "../memory/conversation-disk-view.js";
79
+ import { getDb, initializeDb, rawRun, resetDb } from "../memory/db.js";
80
+
81
+ initializeDb();
82
+
83
+ afterAll(() => {
84
+ resetDb();
85
+ try {
86
+ rmSync(testDir, { recursive: true });
87
+ } catch {
88
+ /* best effort */
89
+ }
90
+ });
91
+
92
+ function resetTables() {
93
+ const db = getDb();
94
+ db.run("DELETE FROM message_attachments");
95
+ db.run("DELETE FROM attachments");
96
+ db.run("DELETE FROM messages");
97
+ db.run("DELETE FROM conversations");
98
+ }
99
+
100
+ function getLegacyConversationDirName(id: string, createdAtMs: number): string {
101
+ return `${id}_${new Date(createdAtMs).toISOString().replace(/:/g, "-")}`;
102
+ }
103
+
104
+ // ---------------------------------------------------------------------------
105
+ // getConversationDirName
106
+ // ---------------------------------------------------------------------------
107
+
108
+ describe("getConversationDirName", () => {
109
+ test("produces filesystem-safe name with colons replaced by hyphens", () => {
110
+ // 2026-03-18T14:23:00.000Z
111
+ const ts = new Date("2026-03-18T14:23:00.000Z").getTime();
112
+ const name = getConversationDirName("abc123", ts);
113
+ expect(name).toBe("2026-03-18T14-23-00.000Z_abc123");
114
+ // No colons in the name (safe for Windows/macOS/Linux)
115
+ expect(name).not.toContain(":");
116
+ });
117
+
118
+ test("handles epoch zero", () => {
119
+ const name = getConversationDirName("conv0", 0);
120
+ expect(name).toBe("1970-01-01T00-00-00.000Z_conv0");
121
+ });
122
+ });
123
+
124
+ // ---------------------------------------------------------------------------
125
+ // getConversationDirPath
126
+ // ---------------------------------------------------------------------------
127
+
128
+ describe("getConversationDirPath", () => {
129
+ test("returns absolute path under conversations dir", () => {
130
+ const ts = Date.now();
131
+ const dirPath = getConversationDirPath("test-id", ts);
132
+ expect(dirPath.startsWith(conversationsDir)).toBe(true);
133
+ expect(dirPath).toContain("_test-id");
134
+ });
135
+ });
136
+
137
+ // ---------------------------------------------------------------------------
138
+ // initConversationDir
139
+ // ---------------------------------------------------------------------------
140
+
141
+ describe("initConversationDir", () => {
142
+ beforeEach(resetTables);
143
+
144
+ test("creates directory and writes valid meta.json", () => {
145
+ const now = Date.now();
146
+ initConversationDir({
147
+ id: "conv-init-1",
148
+ title: "Test Conversation",
149
+ createdAt: now,
150
+ conversationType: "standard",
151
+ originChannel: "desktop",
152
+ });
153
+
154
+ const dirPath = getConversationDirPath("conv-init-1", now);
155
+ expect(existsSync(dirPath)).toBe(true);
156
+
157
+ const metaPath = join(dirPath, "meta.json");
158
+ expect(existsSync(metaPath)).toBe(true);
159
+
160
+ const meta = JSON.parse(readFileSync(metaPath, "utf-8"));
161
+ expect(meta.id).toBe("conv-init-1");
162
+ expect(meta.title).toBe("Test Conversation");
163
+ expect(meta.type).toBe("standard");
164
+ expect(meta.channel).toBe("desktop");
165
+ expect(meta.createdAt).toBe(new Date(now).toISOString());
166
+ expect(meta.updatedAt).toBe(new Date(now).toISOString());
167
+
168
+ // Cleanup
169
+ rmSync(dirPath, { recursive: true, force: true });
170
+ });
171
+
172
+ test("handles null title and null originChannel", () => {
173
+ const now = Date.now();
174
+ initConversationDir({
175
+ id: "conv-init-null",
176
+ title: null,
177
+ createdAt: now,
178
+ conversationType: "private",
179
+ originChannel: null,
180
+ });
181
+
182
+ const dirPath = getConversationDirPath("conv-init-null", now);
183
+ const meta = JSON.parse(readFileSync(join(dirPath, "meta.json"), "utf-8"));
184
+ expect(meta.title).toBeNull();
185
+ expect(meta.channel).toBeNull();
186
+ expect(meta.type).toBe("private");
187
+
188
+ rmSync(dirPath, { recursive: true, force: true });
189
+ });
190
+ });
191
+
192
+ // ---------------------------------------------------------------------------
193
+ // updateMetaFile
194
+ // ---------------------------------------------------------------------------
195
+
196
+ describe("updateMetaFile", () => {
197
+ beforeEach(resetTables);
198
+
199
+ test("rewrites meta.json with updated fields", () => {
200
+ const created = Date.now();
201
+ const updated = created + 5000;
202
+
203
+ initConversationDir({
204
+ id: "conv-update",
205
+ title: "Original",
206
+ createdAt: created,
207
+ conversationType: "standard",
208
+ originChannel: null,
209
+ });
210
+
211
+ updateMetaFile({
212
+ id: "conv-update",
213
+ title: "Updated Title",
214
+ createdAt: created,
215
+ updatedAt: updated,
216
+ conversationType: "standard",
217
+ originChannel: "telegram",
218
+ });
219
+
220
+ const dirPath = getConversationDirPath("conv-update", created);
221
+ const meta = JSON.parse(readFileSync(join(dirPath, "meta.json"), "utf-8"));
222
+ expect(meta.title).toBe("Updated Title");
223
+ expect(meta.channel).toBe("telegram");
224
+ expect(meta.updatedAt).toBe(new Date(updated).toISOString());
225
+
226
+ rmSync(dirPath, { recursive: true, force: true });
227
+ });
228
+
229
+ test("reuses legacy directory names when the new directory does not exist", () => {
230
+ const created = Date.now();
231
+ const legacyDirName = getLegacyConversationDirName("conv-legacy", created);
232
+ const legacyDirPath = join(conversationsDir, legacyDirName);
233
+ mkdirSync(legacyDirPath, { recursive: true });
234
+
235
+ updateMetaFile({
236
+ id: "conv-legacy",
237
+ title: "Legacy",
238
+ createdAt: created,
239
+ updatedAt: created + 1234,
240
+ conversationType: "standard",
241
+ originChannel: "desktop",
242
+ });
243
+
244
+ expect(existsSync(join(legacyDirPath, "meta.json"))).toBe(true);
245
+ expect(existsSync(getConversationDirPath("conv-legacy", created))).toBe(
246
+ false,
247
+ );
248
+
249
+ rmSync(legacyDirPath, { recursive: true, force: true });
250
+ });
251
+
252
+ test("recreates a missing directory before rewriting meta.json", () => {
253
+ const created = Date.now();
254
+ const updated = created + 2500;
255
+ initConversationDir({
256
+ id: "conv-update-recreate",
257
+ title: "Original",
258
+ createdAt: created,
259
+ conversationType: "standard",
260
+ originChannel: null,
261
+ });
262
+
263
+ const dirPath = getConversationDirPath("conv-update-recreate", created);
264
+ rmSync(dirPath, { recursive: true, force: true });
265
+
266
+ updateMetaFile({
267
+ id: "conv-update-recreate",
268
+ title: "Recreated",
269
+ createdAt: created,
270
+ updatedAt: updated,
271
+ conversationType: "standard",
272
+ originChannel: "desktop",
273
+ });
274
+
275
+ expect(existsSync(dirPath)).toBe(true);
276
+ expect(existsSync(join(dirPath, "meta.json"))).toBe(true);
277
+
278
+ const meta = JSON.parse(readFileSync(join(dirPath, "meta.json"), "utf-8"));
279
+ expect(meta.title).toBe("Recreated");
280
+ expect(meta.channel).toBe("desktop");
281
+ expect(meta.updatedAt).toBe(new Date(updated).toISOString());
282
+
283
+ rmSync(dirPath, { recursive: true, force: true });
284
+ });
285
+ });
286
+
287
+ // ---------------------------------------------------------------------------
288
+ // flattenContentBlocks
289
+ // ---------------------------------------------------------------------------
290
+
291
+ describe("flattenContentBlocks", () => {
292
+ test("extracts text from text blocks", () => {
293
+ const blocks = JSON.stringify([
294
+ { type: "text", text: "Hello" },
295
+ { type: "text", text: "World" },
296
+ ]);
297
+ const result = flattenContentBlocks(blocks);
298
+ expect(result.content).toBe("Hello\nWorld");
299
+ expect(result.toolCalls).toEqual([]);
300
+ expect(result.toolResults).toEqual([]);
301
+ });
302
+
303
+ test("extracts tool_use blocks", () => {
304
+ const blocks = JSON.stringify([
305
+ { type: "tool_use", name: "image_resize", input: { width: 800 } },
306
+ ]);
307
+ const result = flattenContentBlocks(blocks);
308
+ expect(result.toolCalls).toEqual([
309
+ { name: "image_resize", input: { width: 800 } },
310
+ ]);
311
+ });
312
+
313
+ test("extracts tool_result blocks", () => {
314
+ const blocks = JSON.stringify([{ type: "tool_result", content: "Done!" }]);
315
+ const result = flattenContentBlocks(blocks);
316
+ expect(result.toolResults).toEqual([{ content: "Done!" }]);
317
+ });
318
+
319
+ test("skips image and file blocks", () => {
320
+ const blocks = JSON.stringify([
321
+ { type: "text", text: "Here is an image" },
322
+ { type: "image", source: { data: "base64..." } },
323
+ { type: "file", path: "/tmp/test.txt" },
324
+ ]);
325
+ const result = flattenContentBlocks(blocks);
326
+ expect(result.content).toBe("Here is an image");
327
+ expect(result.toolCalls).toEqual([]);
328
+ expect(result.toolResults).toEqual([]);
329
+ });
330
+
331
+ test("handles plain text (non-JSON) content", () => {
332
+ const result = flattenContentBlocks("Just a string message");
333
+ expect(result.content).toBe("Just a string message");
334
+ });
335
+
336
+ test("handles non-array JSON gracefully", () => {
337
+ const result = flattenContentBlocks(
338
+ JSON.stringify({ text: "not an array" }),
339
+ );
340
+ expect(result.content).toBe(JSON.stringify({ text: "not an array" }));
341
+ });
342
+
343
+ test("handles mixed block types", () => {
344
+ const blocks = JSON.stringify([
345
+ { type: "text", text: "Can you resize this?" },
346
+ { type: "image", source: { data: "abc" } },
347
+ { type: "tool_use", name: "image_resize", input: { width: 800 } },
348
+ { type: "tool_result", content: "Resized to 800x600" },
349
+ { type: "text", text: "Done." },
350
+ ]);
351
+ const result = flattenContentBlocks(blocks);
352
+ expect(result.content).toBe("Can you resize this?\nDone.");
353
+ expect(result.toolCalls).toHaveLength(1);
354
+ expect(result.toolResults).toHaveLength(1);
355
+ });
356
+ });
357
+
358
+ // ---------------------------------------------------------------------------
359
+ // resolveUniqueFilename
360
+ // ---------------------------------------------------------------------------
361
+
362
+ describe("resolveUniqueFilename", () => {
363
+ test("returns original filename when no collision", () => {
364
+ const dir = mkdtempSync(join(tmpdir(), "unique-fn-"));
365
+ expect(resolveUniqueFilename(dir, "photo.png")).toBe("photo.png");
366
+ rmSync(dir, { recursive: true });
367
+ });
368
+
369
+ test("appends -2, -3 on collision", () => {
370
+ const dir = mkdtempSync(join(tmpdir(), "unique-fn-"));
371
+ writeFileSync(join(dir, "photo.png"), "");
372
+ expect(resolveUniqueFilename(dir, "photo.png")).toBe("photo-2.png");
373
+
374
+ writeFileSync(join(dir, "photo-2.png"), "");
375
+ expect(resolveUniqueFilename(dir, "photo.png")).toBe("photo-3.png");
376
+
377
+ rmSync(dir, { recursive: true });
378
+ });
379
+
380
+ test("handles files without extension", () => {
381
+ const dir = mkdtempSync(join(tmpdir(), "unique-fn-"));
382
+ writeFileSync(join(dir, "README"), "");
383
+ expect(resolveUniqueFilename(dir, "README")).toBe("README-2");
384
+ rmSync(dir, { recursive: true });
385
+ });
386
+
387
+ test("strips path traversal sequences from filename", () => {
388
+ const dir = mkdtempSync(join(tmpdir(), "unique-fn-"));
389
+ expect(resolveUniqueFilename(dir, "../../evil.txt")).toBe("evil.txt");
390
+ expect(resolveUniqueFilename(dir, "../secret.png")).toBe("secret.png");
391
+ expect(resolveUniqueFilename(dir, "foo/bar/baz.txt")).toBe("baz.txt");
392
+ rmSync(dir, { recursive: true });
393
+ });
394
+ });
395
+
396
+ // ---------------------------------------------------------------------------
397
+ // syncMessageToDisk
398
+ // ---------------------------------------------------------------------------
399
+
400
+ describe("syncMessageToDisk", () => {
401
+ beforeEach(resetTables);
402
+
403
+ test("appends correct JSONL for text-only message", async () => {
404
+ const conv = createConversation("Test");
405
+ initConversationDir({
406
+ id: conv.id,
407
+ title: conv.title,
408
+ createdAt: conv.createdAt,
409
+ conversationType: conv.conversationType,
410
+ originChannel: null,
411
+ });
412
+
413
+ const msg = await addMessage(
414
+ conv.id,
415
+ "user",
416
+ "Hello, assistant!",
417
+ undefined,
418
+ { skipIndexing: true },
419
+ );
420
+
421
+ syncMessageToDisk(conv.id, msg.id, conv.createdAt);
422
+
423
+ const dirPath = getConversationDirPath(conv.id, conv.createdAt);
424
+ const jsonlPath = join(dirPath, "messages.jsonl");
425
+ expect(existsSync(jsonlPath)).toBe(true);
426
+
427
+ const lines = readFileSync(jsonlPath, "utf-8").trim().split("\n");
428
+ expect(lines).toHaveLength(1);
429
+
430
+ const record = JSON.parse(lines[0]);
431
+ expect(record.role).toBe("user");
432
+ expect(record.content).toBe("Hello, assistant!");
433
+ expect(record.ts).toBeDefined();
434
+ expect(record.toolCalls).toBeUndefined();
435
+ expect(record.attachments).toBeUndefined();
436
+
437
+ rmSync(dirPath, { recursive: true, force: true });
438
+ });
439
+
440
+ test("appends correct JSONL for message with tool calls", async () => {
441
+ const conv = createConversation("Tool Test");
442
+ initConversationDir({
443
+ id: conv.id,
444
+ title: conv.title,
445
+ createdAt: conv.createdAt,
446
+ conversationType: conv.conversationType,
447
+ originChannel: null,
448
+ });
449
+
450
+ const content = JSON.stringify([
451
+ { type: "text", text: "Resizing image..." },
452
+ { type: "tool_use", name: "image_resize", input: { width: 800 } },
453
+ ]);
454
+
455
+ const msg = await addMessage(conv.id, "assistant", content, undefined, {
456
+ skipIndexing: true,
457
+ });
458
+
459
+ syncMessageToDisk(conv.id, msg.id, conv.createdAt);
460
+
461
+ const dirPath = getConversationDirPath(conv.id, conv.createdAt);
462
+ const lines = readFileSync(join(dirPath, "messages.jsonl"), "utf-8")
463
+ .trim()
464
+ .split("\n");
465
+ const record = JSON.parse(lines[0]);
466
+ expect(record.content).toBe("Resizing image...");
467
+ expect(record.toolCalls).toEqual([
468
+ { name: "image_resize", input: { width: 800 } },
469
+ ]);
470
+
471
+ rmSync(dirPath, { recursive: true, force: true });
472
+ });
473
+
474
+ test("copies attachments and includes filenames in JSONL", async () => {
475
+ const conv = createConversation("Attach Test");
476
+ initConversationDir({
477
+ id: conv.id,
478
+ title: conv.title,
479
+ createdAt: conv.createdAt,
480
+ conversationType: conv.conversationType,
481
+ originChannel: null,
482
+ });
483
+
484
+ const msg = await addMessage(conv.id, "user", "See attached", undefined, {
485
+ skipIndexing: true,
486
+ });
487
+
488
+ // Upload an attachment and link to the message
489
+ const att = uploadAttachment("photo.png", "image/png", "iVBORw0K");
490
+ linkAttachmentToMessage(msg.id, att.id, 0);
491
+
492
+ syncMessageToDisk(conv.id, msg.id, conv.createdAt);
493
+
494
+ const dirPath = getConversationDirPath(conv.id, conv.createdAt);
495
+ const attachDir = join(dirPath, "attachments");
496
+ expect(existsSync(join(attachDir, "photo.png"))).toBe(true);
497
+
498
+ const lines = readFileSync(join(dirPath, "messages.jsonl"), "utf-8")
499
+ .trim()
500
+ .split("\n");
501
+ const record = JSON.parse(lines[0]);
502
+ expect(record.attachments).toEqual(["photo.png"]);
503
+
504
+ rmSync(dirPath, { recursive: true, force: true });
505
+ });
506
+
507
+ test("appends multiple messages sequentially", async () => {
508
+ const conv = createConversation("Multi");
509
+ initConversationDir({
510
+ id: conv.id,
511
+ title: conv.title,
512
+ createdAt: conv.createdAt,
513
+ conversationType: conv.conversationType,
514
+ originChannel: null,
515
+ });
516
+
517
+ const msg1 = await addMessage(conv.id, "user", "First", undefined, {
518
+ skipIndexing: true,
519
+ });
520
+ const msg2 = await addMessage(conv.id, "assistant", "Second", undefined, {
521
+ skipIndexing: true,
522
+ });
523
+
524
+ syncMessageToDisk(conv.id, msg1.id, conv.createdAt);
525
+ syncMessageToDisk(conv.id, msg2.id, conv.createdAt);
526
+
527
+ const dirPath = getConversationDirPath(conv.id, conv.createdAt);
528
+ const lines = readFileSync(join(dirPath, "messages.jsonl"), "utf-8")
529
+ .trim()
530
+ .split("\n");
531
+ expect(lines).toHaveLength(2);
532
+ expect(JSON.parse(lines[0]).role).toBe("user");
533
+ expect(JSON.parse(lines[1]).role).toBe("assistant");
534
+
535
+ rmSync(dirPath, { recursive: true, force: true });
536
+ });
537
+
538
+ test("appends to a legacy directory when that is the only existing path", async () => {
539
+ const conv = createConversation("Legacy Attach Test");
540
+ const createdAt = conv.createdAt;
541
+ const newDirPath = getConversationDirPath(conv.id, createdAt);
542
+ rmSync(newDirPath, { recursive: true, force: true });
543
+ const legacyDirPath = join(
544
+ conversationsDir,
545
+ getLegacyConversationDirName(conv.id, createdAt),
546
+ );
547
+ mkdirSync(legacyDirPath, { recursive: true });
548
+
549
+ const msg = await addMessage(conv.id, "user", "Legacy path", undefined, {
550
+ skipIndexing: true,
551
+ });
552
+
553
+ const att = uploadAttachment("legacy.png", "image/png", "iVBORw0K");
554
+ linkAttachmentToMessage(msg.id, att.id, 0);
555
+
556
+ syncMessageToDisk(conv.id, msg.id, createdAt);
557
+
558
+ expect(
559
+ existsSync(join(legacyDirPath, "messages.jsonl")),
560
+ ).toBe(true);
561
+ expect(existsSync(join(newDirPath, "messages.jsonl"))).toBe(false);
562
+ expect(existsSync(join(legacyDirPath, "attachments", "legacy.png"))).toBe(
563
+ true,
564
+ );
565
+
566
+ rmSync(legacyDirPath, { recursive: true, force: true });
567
+ });
568
+
569
+ test("recreates a missing directory before appending messages and attachments", async () => {
570
+ const conv = createConversation("Recreate Sync");
571
+ initConversationDir({
572
+ id: conv.id,
573
+ title: conv.title,
574
+ createdAt: conv.createdAt,
575
+ conversationType: conv.conversationType,
576
+ originChannel: null,
577
+ });
578
+
579
+ const msg = await addMessage(conv.id, "user", "Disk repair", undefined, {
580
+ skipIndexing: true,
581
+ });
582
+ const att = uploadAttachment("repair.png", "image/png", "iVBORw0K");
583
+ rawRun(
584
+ `INSERT INTO message_attachments (id, message_id, attachment_id, position, created_at)
585
+ VALUES (?, ?, ?, ?, ?)`,
586
+ `manual-link-${msg.id}`,
587
+ msg.id,
588
+ att.id,
589
+ 0,
590
+ Date.now(),
591
+ );
592
+
593
+ const dirPath = getConversationDirPath(conv.id, conv.createdAt);
594
+ const legacyDirPath = join(
595
+ conversationsDir,
596
+ getLegacyConversationDirName(conv.id, conv.createdAt),
597
+ );
598
+ rmSync(dirPath, { recursive: true, force: true });
599
+ rmSync(legacyDirPath, { recursive: true, force: true });
600
+
601
+ syncMessageToDisk(conv.id, msg.id, conv.createdAt);
602
+
603
+ expect(existsSync(dirPath)).toBe(true);
604
+ expect(existsSync(join(dirPath, "messages.jsonl"))).toBe(true);
605
+ expect(existsSync(join(dirPath, "attachments", "repair.png"))).toBe(true);
606
+
607
+ const lines = readFileSync(join(dirPath, "messages.jsonl"), "utf-8")
608
+ .trim()
609
+ .split("\n");
610
+ expect(lines).toHaveLength(1);
611
+ const record = JSON.parse(lines[0]);
612
+ expect(record.content).toBe("Disk repair");
613
+ expect(record.attachments).toHaveLength(1);
614
+ expect(existsSync(join(dirPath, "attachments", record.attachments[0]))).toBe(
615
+ true,
616
+ );
617
+
618
+ rmSync(dirPath, { recursive: true, force: true });
619
+ });
620
+ });
621
+
622
+ describe("rebuildConversationDiskViewFromDbState", () => {
623
+ beforeEach(resetTables);
624
+
625
+ test("rewrites stale pre-consolidation disk view with final DB state", async () => {
626
+ const conv = createConversation("Consolidation Repair");
627
+ initConversationDir({
628
+ id: conv.id,
629
+ title: conv.title,
630
+ createdAt: conv.createdAt,
631
+ conversationType: conv.conversationType,
632
+ originChannel: null,
633
+ });
634
+
635
+ const userMsg = await addMessage(conv.id, "user", "find docs", undefined, {
636
+ skipIndexing: true,
637
+ });
638
+ const assistantPart1 = await addMessage(
639
+ conv.id,
640
+ "assistant",
641
+ JSON.stringify([{ type: "text", text: "Searching..." }]),
642
+ undefined,
643
+ { skipIndexing: true },
644
+ );
645
+ const internalToolResult = await addMessage(
646
+ conv.id,
647
+ "user",
648
+ JSON.stringify([
649
+ { type: "tool_result", tool_use_id: "tool-1", content: "done" },
650
+ ]),
651
+ undefined,
652
+ { skipIndexing: true },
653
+ );
654
+ const assistantPart2 = await addMessage(
655
+ conv.id,
656
+ "assistant",
657
+ JSON.stringify([{ type: "text", text: "Found it." }]),
658
+ undefined,
659
+ { skipIndexing: true },
660
+ );
661
+
662
+ const att = uploadAttachment("result.txt", "text/plain", "ok");
663
+ linkAttachmentToMessage(assistantPart2.id, att.id, 0);
664
+
665
+ // Simulate stale disk view generated before consolidation.
666
+ syncMessageToDisk(conv.id, userMsg.id, conv.createdAt);
667
+ syncMessageToDisk(conv.id, assistantPart1.id, conv.createdAt);
668
+ syncMessageToDisk(conv.id, internalToolResult.id, conv.createdAt);
669
+ syncMessageToDisk(conv.id, assistantPart2.id, conv.createdAt);
670
+
671
+ // Simulate DB mutations performed by consolidation.
672
+ updateMessageContent(
673
+ assistantPart1.id,
674
+ JSON.stringify([
675
+ { type: "text", text: "Searching..." },
676
+ { type: "tool_result", tool_use_id: "tool-1", content: "done" },
677
+ { type: "text", text: "Found it." },
678
+ ]),
679
+ );
680
+ relinkAttachments([assistantPart2.id], assistantPart1.id);
681
+ deleteMessageById(internalToolResult.id);
682
+ deleteMessageById(assistantPart2.id);
683
+
684
+ rebuildConversationDiskViewFromDbState(conv.id);
685
+
686
+ const dirPath = getConversationDirPath(conv.id, conv.createdAt);
687
+ const lines = readFileSync(join(dirPath, "messages.jsonl"), "utf-8")
688
+ .trim()
689
+ .split("\n");
690
+
691
+ expect(lines).toHaveLength(2);
692
+
693
+ const rebuiltUser = JSON.parse(lines[0]);
694
+ const rebuiltAssistant = JSON.parse(lines[1]);
695
+
696
+ expect(rebuiltUser.role).toBe("user");
697
+ expect(rebuiltAssistant.role).toBe("assistant");
698
+ expect(rebuiltAssistant.content).toBe("Searching...\nFound it.");
699
+ expect(rebuiltAssistant.toolResults).toEqual([{ content: "done" }]);
700
+ expect(rebuiltAssistant.attachments).toHaveLength(1);
701
+ expect(
702
+ existsSync(join(dirPath, "attachments", rebuiltAssistant.attachments[0])),
703
+ ).toBe(true);
704
+
705
+ rmSync(dirPath, { recursive: true, force: true });
706
+ });
707
+ });
708
+
709
+ // ---------------------------------------------------------------------------
710
+ // removeConversationDir
711
+ // ---------------------------------------------------------------------------
712
+
713
+ describe("removeConversationDir", () => {
714
+ test("removes the directory and its contents", () => {
715
+ const now = Date.now();
716
+ initConversationDir({
717
+ id: "conv-remove",
718
+ title: "To be removed",
719
+ createdAt: now,
720
+ conversationType: "standard",
721
+ originChannel: null,
722
+ });
723
+
724
+ const dirPath = getConversationDirPath("conv-remove", now);
725
+ expect(existsSync(dirPath)).toBe(true);
726
+
727
+ removeConversationDir("conv-remove", now);
728
+ expect(existsSync(dirPath)).toBe(false);
729
+ });
730
+
731
+ test("handles non-existent directory gracefully", () => {
732
+ // Should not throw
733
+ removeConversationDir("nonexistent", Date.now());
734
+ });
735
+
736
+ test("removes both new-format and legacy directories when both exist", () => {
737
+ const created = Date.now();
738
+ const newDirPath = getConversationDirPath("conv-both", created);
739
+ const legacyDirPath = join(
740
+ conversationsDir,
741
+ getLegacyConversationDirName("conv-both", created),
742
+ );
743
+ mkdirSync(newDirPath, { recursive: true });
744
+ mkdirSync(legacyDirPath, { recursive: true });
745
+
746
+ removeConversationDir("conv-both", created);
747
+
748
+ expect(existsSync(newDirPath)).toBe(false);
749
+ expect(existsSync(legacyDirPath)).toBe(false);
750
+ });
751
+ });
752
+
753
+ // ---------------------------------------------------------------------------
754
+ // Error resilience
755
+ // ---------------------------------------------------------------------------
756
+
757
+ describe("error resilience", () => {
758
+ test("initConversationDir does not throw on write failure", () => {
759
+ // Create a file at the path where a directory would be created, so
760
+ // mkdirSync fails with EEXIST. This triggers the try/catch in
761
+ // initConversationDir. The function should swallow the error.
762
+ const badConvId = "conv-fail-write";
763
+ const now = Date.now();
764
+ const dirPath = getConversationDirPath(badConvId, now);
765
+
766
+ mkdirSync(conversationsDir, { recursive: true });
767
+ writeFileSync(dirPath, "blocker");
768
+
769
+ try {
770
+ // Should not throw despite the internal failure
771
+ expect(() => {
772
+ initConversationDir({
773
+ id: badConvId,
774
+ title: "Test",
775
+ createdAt: now,
776
+ conversationType: "standard",
777
+ originChannel: null,
778
+ });
779
+ }).not.toThrow();
780
+ } finally {
781
+ rmSync(dirPath, { force: true });
782
+ }
783
+ });
784
+
785
+ test("updateMetaFile does not throw when directory does not exist", () => {
786
+ expect(() => {
787
+ updateMetaFile({
788
+ id: "nonexistent",
789
+ title: "X",
790
+ createdAt: 1000,
791
+ updatedAt: 2000,
792
+ conversationType: "standard",
793
+ originChannel: null,
794
+ });
795
+ }).not.toThrow();
796
+ });
797
+
798
+ test("syncMessageToDisk does not throw when message is not found", () => {
799
+ // Should not throw — logs a warning instead
800
+ expect(() => {
801
+ syncMessageToDisk("missing-conv", "missing-msg", Date.now());
802
+ }).not.toThrow();
803
+ });
804
+
805
+ test("removeConversationDir does not throw on missing directory", () => {
806
+ expect(() => {
807
+ removeConversationDir("nonexistent-id", 0);
808
+ }).not.toThrow();
809
+ });
810
+ });