ruflo 3.10.46 → 3.12.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 (498) hide show
  1. package/README.md +412 -412
  2. package/bin/ruflo.js +77 -77
  3. package/package.json +118 -113
  4. package/src/chat-ui/Dockerfile +25 -25
  5. package/src/chat-ui/patch-mcp-url-safety.sh +28 -28
  6. package/src/config/config.example.json +76 -76
  7. package/src/mcp-bridge/Dockerfile +45 -45
  8. package/src/mcp-bridge/index.js +1692 -1692
  9. package/src/mcp-bridge/mcp-stdio-kernel.js +159 -159
  10. package/src/mcp-bridge/package.json +17 -17
  11. package/src/mcp-bridge/test-harness.js +470 -470
  12. package/src/nginx/Dockerfile +10 -10
  13. package/src/nginx/nginx.conf +67 -67
  14. package/src/nginx/static/favicon-dark.svg +4 -4
  15. package/src/nginx/static/favicon.svg +4 -4
  16. package/src/nginx/static/icon.svg +5 -5
  17. package/src/nginx/static/logo.svg +9 -9
  18. package/src/nginx/static/manifest.json +22 -22
  19. package/src/nginx/static/welcome.js +184 -184
  20. package/src/ruvocal/.claude/skills/add-model-descriptions/SKILL.md +73 -73
  21. package/src/ruvocal/.claude-flow/daemon-state.json +135 -0
  22. package/src/ruvocal/.claude-flow/data/pending-insights.jsonl +0 -0
  23. package/src/ruvocal/.claude-flow/data/ranked-context.json +5 -0
  24. package/src/ruvocal/.claude-flow/logs/daemon.log +31 -0
  25. package/src/ruvocal/.claude-flow/logs/headless/audit_1777949411822_juxau0_prompt.log +989 -0
  26. package/src/ruvocal/.claude-flow/logs/headless/audit_1777949411822_juxau0_result.log +67 -0
  27. package/src/ruvocal/.claude-flow/logs/headless/audit_1777950042278_jvj5xq_prompt.log +989 -0
  28. package/src/ruvocal/.claude-flow/logs/headless/audit_1777950042278_jvj5xq_result.log +93 -0
  29. package/src/ruvocal/.claude-flow/logs/headless/optimize_1777949531823_yt5yc2_prompt.log +1498 -0
  30. package/src/ruvocal/.claude-flow/logs/headless/optimize_1777949531823_yt5yc2_result.log +93 -0
  31. package/src/ruvocal/.claude-flow/logs/headless/testgaps_1777949771821_elw1j4_prompt.log +1498 -0
  32. package/src/ruvocal/.claude-flow/logs/headless/testgaps_1777949771821_elw1j4_result.log +100 -0
  33. package/src/ruvocal/.claude-flow/metrics/codebase-map.json +11 -0
  34. package/src/ruvocal/.claude-flow/metrics/consolidation.json +6 -0
  35. package/src/ruvocal/.claude-flow/neural/stats.json +6 -0
  36. package/src/ruvocal/.claude-flow/sessions/current.json +13 -0
  37. package/src/ruvocal/.devcontainer/Dockerfile +9 -9
  38. package/src/ruvocal/.devcontainer/devcontainer.json +36 -36
  39. package/src/ruvocal/.dockerignore +16 -16
  40. package/src/ruvocal/.eslintignore +13 -13
  41. package/src/ruvocal/.eslintrc.cjs +45 -45
  42. package/src/ruvocal/.gcloudignore +18 -18
  43. package/src/ruvocal/.github/ISSUE_TEMPLATE/bug-report--chat-ui-.md +43 -43
  44. package/src/ruvocal/.github/ISSUE_TEMPLATE/config-support.md +9 -9
  45. package/src/ruvocal/.github/ISSUE_TEMPLATE/feature-request--chat-ui-.md +17 -17
  46. package/src/ruvocal/.github/ISSUE_TEMPLATE/huggingchat.md +11 -11
  47. package/src/ruvocal/.github/release.yml +16 -16
  48. package/src/ruvocal/.github/workflows/build-docs.yml +18 -18
  49. package/src/ruvocal/.github/workflows/build-image.yml +142 -142
  50. package/src/ruvocal/.github/workflows/build-pr-docs.yml +20 -20
  51. package/src/ruvocal/.github/workflows/deploy-dev.yml +63 -63
  52. package/src/ruvocal/.github/workflows/deploy-prod.yml +78 -78
  53. package/src/ruvocal/.github/workflows/lint-and-test.yml +84 -84
  54. package/src/ruvocal/.github/workflows/slugify.yaml +72 -72
  55. package/src/ruvocal/.github/workflows/trufflehog.yml +17 -17
  56. package/src/ruvocal/.github/workflows/upload-pr-documentation.yml +16 -16
  57. package/src/ruvocal/.husky/lint-stage-config.js +4 -4
  58. package/src/ruvocal/.husky/pre-commit +2 -2
  59. package/src/ruvocal/.prettierignore +14 -14
  60. package/src/ruvocal/.prettierrc +7 -7
  61. package/src/ruvocal/.swarm/attestation.db +0 -0
  62. package/src/ruvocal/.swarm/hnsw.index +0 -0
  63. package/src/ruvocal/.swarm/hnsw.metadata.json +1 -0
  64. package/src/ruvocal/.swarm/memory.db +0 -0
  65. package/src/ruvocal/.swarm/schema.sql +305 -0
  66. package/src/ruvocal/CLAUDE.md +126 -126
  67. package/src/ruvocal/Dockerfile +96 -96
  68. package/src/ruvocal/LICENSE +202 -202
  69. package/src/ruvocal/PRIVACY.md +41 -41
  70. package/src/ruvocal/README.md +164 -164
  71. package/src/ruvocal/chart/Chart.yaml +5 -5
  72. package/src/ruvocal/chart/env/dev.yaml +260 -260
  73. package/src/ruvocal/chart/env/prod.yaml +273 -273
  74. package/src/ruvocal/chart/templates/_helpers.tpl +22 -22
  75. package/src/ruvocal/chart/templates/config.yaml +10 -10
  76. package/src/ruvocal/chart/templates/deployment.yaml +81 -81
  77. package/src/ruvocal/chart/templates/hpa.yaml +45 -45
  78. package/src/ruvocal/chart/templates/infisical.yaml +24 -24
  79. package/src/ruvocal/chart/templates/ingress-internal.yaml +32 -32
  80. package/src/ruvocal/chart/templates/ingress.yaml +32 -32
  81. package/src/ruvocal/chart/templates/network-policy.yaml +36 -36
  82. package/src/ruvocal/chart/templates/service-account.yaml +13 -13
  83. package/src/ruvocal/chart/templates/service-monitor.yaml +17 -17
  84. package/src/ruvocal/chart/templates/service.yaml +21 -21
  85. package/src/ruvocal/chart/values.yaml +73 -73
  86. package/src/ruvocal/cloudbuild.yaml +68 -68
  87. package/src/ruvocal/config/branding.env.example +19 -19
  88. package/src/ruvocal/docker-compose.yml +21 -21
  89. package/src/ruvocal/docs/adr/ADR-029-HUGGINGFACE-CHAT-UI-CLOUD-RUN.md +1236 -1236
  90. package/src/ruvocal/docs/adr/ADR-033-RUVECTOR-RUFLO-MCP-INTEGRATION.md +111 -111
  91. package/src/ruvocal/docs/adr/ADR-034-OPTIONAL-MCP-BACKENDS.md +117 -117
  92. package/src/ruvocal/docs/adr/ADR-035-MCP-TOOL-GROUPS.md +186 -186
  93. package/src/ruvocal/docs/adr/ADR-037-AUTOPILOT-CHAT-MODE.md +1500 -1500
  94. package/src/ruvocal/docs/adr/ADR-038-RUVOCAL-FORK.md +286 -286
  95. package/src/ruvocal/docs/source/_toctree.yml +30 -30
  96. package/src/ruvocal/docs/source/configuration/common-issues.md +38 -38
  97. package/src/ruvocal/docs/source/configuration/llm-router.md +105 -105
  98. package/src/ruvocal/docs/source/configuration/mcp-tools.md +84 -84
  99. package/src/ruvocal/docs/source/configuration/metrics.md +9 -9
  100. package/src/ruvocal/docs/source/configuration/open-id.md +57 -57
  101. package/src/ruvocal/docs/source/configuration/overview.md +89 -89
  102. package/src/ruvocal/docs/source/configuration/theming.md +20 -20
  103. package/src/ruvocal/docs/source/developing/architecture.md +48 -48
  104. package/src/ruvocal/docs/source/index.md +53 -53
  105. package/src/ruvocal/docs/source/installation/docker.md +43 -43
  106. package/src/ruvocal/docs/source/installation/helm.md +43 -43
  107. package/src/ruvocal/docs/source/installation/local.md +62 -62
  108. package/src/ruvocal/entrypoint.sh +18 -18
  109. package/src/ruvocal/mcp-bridge/Dockerfile +45 -45
  110. package/src/ruvocal/mcp-bridge/cloudbuild.yaml +49 -49
  111. package/src/ruvocal/mcp-bridge/index.js +1902 -1902
  112. package/src/ruvocal/mcp-bridge/mcp-stdio-kernel.js +159 -159
  113. package/src/ruvocal/mcp-bridge/package-lock.json +762 -762
  114. package/src/ruvocal/mcp-bridge/package.json +17 -17
  115. package/src/ruvocal/mcp-bridge/test-harness.js +470 -470
  116. package/src/ruvocal/package-lock.json +11741 -11741
  117. package/src/ruvocal/package.json +121 -121
  118. package/src/ruvocal/postcss.config.js +6 -6
  119. package/src/ruvocal/rvf.manifest.json +204 -204
  120. package/src/ruvocal/scripts/config.ts +64 -64
  121. package/src/ruvocal/scripts/generate-welcome.mjs +181 -181
  122. package/src/ruvocal/scripts/populate.ts +288 -288
  123. package/src/ruvocal/scripts/samples.txt +194 -194
  124. package/src/ruvocal/scripts/setups/vitest-setup-server.ts +44 -44
  125. package/src/ruvocal/scripts/updateLocalEnv.ts +48 -48
  126. package/src/ruvocal/src/ambient.d.ts +7 -7
  127. package/src/ruvocal/src/app.d.ts +29 -29
  128. package/src/ruvocal/src/app.html +53 -53
  129. package/src/ruvocal/src/hooks.server.ts +32 -32
  130. package/src/ruvocal/src/hooks.ts +6 -6
  131. package/src/ruvocal/src/lib/APIClient.ts +148 -148
  132. package/src/ruvocal/src/lib/actions/clickOutside.ts +18 -18
  133. package/src/ruvocal/src/lib/actions/snapScrollToBottom.ts +346 -346
  134. package/src/ruvocal/src/lib/buildPrompt.ts +33 -33
  135. package/src/ruvocal/src/lib/components/AnnouncementBanner.svelte +20 -20
  136. package/src/ruvocal/src/lib/components/BackgroundGenerationPoller.svelte +168 -168
  137. package/src/ruvocal/src/lib/components/CodeBlock.svelte +73 -73
  138. package/src/ruvocal/src/lib/components/CopyToClipBoardBtn.svelte +92 -92
  139. package/src/ruvocal/src/lib/components/DeleteConversationModal.svelte +75 -75
  140. package/src/ruvocal/src/lib/components/EditConversationModal.svelte +100 -100
  141. package/src/ruvocal/src/lib/components/ExpandNavigation.svelte +22 -22
  142. package/src/ruvocal/src/lib/components/FoundationBackground.svelte +242 -242
  143. package/src/ruvocal/src/lib/components/HoverTooltip.svelte +44 -44
  144. package/src/ruvocal/src/lib/components/HtmlPreviewModal.svelte +143 -143
  145. package/src/ruvocal/src/lib/components/InfiniteScroll.svelte +50 -50
  146. package/src/ruvocal/src/lib/components/MobileNav.svelte +300 -300
  147. package/src/ruvocal/src/lib/components/Modal.svelte +115 -115
  148. package/src/ruvocal/src/lib/components/ModelCardMetadata.svelte +71 -71
  149. package/src/ruvocal/src/lib/components/NavConversationItem.svelte +151 -151
  150. package/src/ruvocal/src/lib/components/NavMenu.svelte +313 -313
  151. package/src/ruvocal/src/lib/components/Pagination.svelte +97 -97
  152. package/src/ruvocal/src/lib/components/PaginationArrow.svelte +27 -27
  153. package/src/ruvocal/src/lib/components/Portal.svelte +24 -24
  154. package/src/ruvocal/src/lib/components/RetryBtn.svelte +18 -18
  155. package/src/ruvocal/src/lib/components/RuFloUniverse.svelte +185 -185
  156. package/src/ruvocal/src/lib/components/RufloHelpModal.svelte +411 -411
  157. package/src/ruvocal/src/lib/components/ScrollToBottomBtn.svelte +47 -47
  158. package/src/ruvocal/src/lib/components/ScrollToPreviousBtn.svelte +77 -77
  159. package/src/ruvocal/src/lib/components/ShareConversationModal.svelte +182 -182
  160. package/src/ruvocal/src/lib/components/StopGeneratingBtn.svelte +69 -69
  161. package/src/ruvocal/src/lib/components/SubscribeModal.svelte +87 -87
  162. package/src/ruvocal/src/lib/components/Switch.svelte +36 -36
  163. package/src/ruvocal/src/lib/components/SystemPromptModal.svelte +44 -44
  164. package/src/ruvocal/src/lib/components/Toast.svelte +27 -27
  165. package/src/ruvocal/src/lib/components/Tooltip.svelte +30 -30
  166. package/src/ruvocal/src/lib/components/WelcomeModal.svelte +46 -46
  167. package/src/ruvocal/src/lib/components/chat/Alternatives.svelte +77 -77
  168. package/src/ruvocal/src/lib/components/chat/BlockWrapper.svelte +72 -72
  169. package/src/ruvocal/src/lib/components/chat/ChatInput.svelte +490 -490
  170. package/src/ruvocal/src/lib/components/chat/ChatIntroduction.svelte +123 -123
  171. package/src/ruvocal/src/lib/components/chat/ChatMessage.svelte +548 -548
  172. package/src/ruvocal/src/lib/components/chat/ChatWindow.svelte +1057 -1057
  173. package/src/ruvocal/src/lib/components/chat/FileDropzone.svelte +92 -92
  174. package/src/ruvocal/src/lib/components/chat/ImageLightbox.svelte +66 -66
  175. package/src/ruvocal/src/lib/components/chat/MarkdownBlock.svelte +23 -23
  176. package/src/ruvocal/src/lib/components/chat/MarkdownRenderer.svelte +69 -69
  177. package/src/ruvocal/src/lib/components/chat/MarkdownRenderer.svelte.test.ts +58 -58
  178. package/src/ruvocal/src/lib/components/chat/MessageAvatar.svelte +103 -103
  179. package/src/ruvocal/src/lib/components/chat/ModelSwitch.svelte +64 -64
  180. package/src/ruvocal/src/lib/components/chat/OpenReasoningResults.svelte +81 -81
  181. package/src/ruvocal/src/lib/components/chat/TaskGroup.svelte +88 -88
  182. package/src/ruvocal/src/lib/components/chat/ToolUpdate.svelte +273 -273
  183. package/src/ruvocal/src/lib/components/chat/UploadedFile.svelte +253 -253
  184. package/src/ruvocal/src/lib/components/chat/UrlFetchModal.svelte +203 -203
  185. package/src/ruvocal/src/lib/components/chat/VoiceRecorder.svelte +214 -214
  186. package/src/ruvocal/src/lib/components/icons/IconBurger.svelte +20 -20
  187. package/src/ruvocal/src/lib/components/icons/IconCheap.svelte +20 -20
  188. package/src/ruvocal/src/lib/components/icons/IconChevron.svelte +24 -24
  189. package/src/ruvocal/src/lib/components/icons/IconDazzled.svelte +40 -40
  190. package/src/ruvocal/src/lib/components/icons/IconFast.svelte +20 -20
  191. package/src/ruvocal/src/lib/components/icons/IconLoading.svelte +22 -22
  192. package/src/ruvocal/src/lib/components/icons/IconMCP.svelte +28 -28
  193. package/src/ruvocal/src/lib/components/icons/IconMoon.svelte +21 -21
  194. package/src/ruvocal/src/lib/components/icons/IconNew.svelte +20 -20
  195. package/src/ruvocal/src/lib/components/icons/IconOmni.svelte +90 -90
  196. package/src/ruvocal/src/lib/components/icons/IconPaperclip.svelte +24 -24
  197. package/src/ruvocal/src/lib/components/icons/IconPro.svelte +37 -37
  198. package/src/ruvocal/src/lib/components/icons/IconShare.svelte +21 -21
  199. package/src/ruvocal/src/lib/components/icons/IconSun.svelte +93 -93
  200. package/src/ruvocal/src/lib/components/icons/Logo.svelte +68 -68
  201. package/src/ruvocal/src/lib/components/icons/LogoHuggingFaceBorderless.svelte +54 -54
  202. package/src/ruvocal/src/lib/components/mcp/AddServerForm.svelte +250 -250
  203. package/src/ruvocal/src/lib/components/mcp/MCPServerManager.svelte +185 -185
  204. package/src/ruvocal/src/lib/components/mcp/ServerCard.svelte +203 -203
  205. package/src/ruvocal/src/lib/components/players/AudioPlayer.svelte +82 -82
  206. package/src/ruvocal/src/lib/components/voice/AudioWaveform.svelte +96 -96
  207. package/src/ruvocal/src/lib/components/wasm/GalleryPanel.svelte +357 -357
  208. package/src/ruvocal/src/lib/constants/mcpExamples.ts +114 -114
  209. package/src/ruvocal/src/lib/constants/mime.ts +11 -11
  210. package/src/ruvocal/src/lib/constants/pagination.ts +1 -1
  211. package/src/ruvocal/src/lib/constants/publicSepToken.ts +1 -1
  212. package/src/ruvocal/src/lib/constants/routerExamples.ts +133 -133
  213. package/src/ruvocal/src/lib/constants/rvagentPresets.ts +206 -206
  214. package/src/ruvocal/src/lib/createShareLink.ts +27 -27
  215. package/src/ruvocal/src/lib/jobs/refresh-conversation-stats.ts +297 -297
  216. package/src/ruvocal/src/lib/migrations/lock.ts +56 -56
  217. package/src/ruvocal/src/lib/migrations/migrations.spec.ts +74 -74
  218. package/src/ruvocal/src/lib/migrations/migrations.ts +109 -109
  219. package/src/ruvocal/src/lib/migrations/routines/01-update-search-assistants.ts +50 -50
  220. package/src/ruvocal/src/lib/migrations/routines/02-update-assistants-models.ts +48 -48
  221. package/src/ruvocal/src/lib/migrations/routines/04-update-message-updates.ts +151 -151
  222. package/src/ruvocal/src/lib/migrations/routines/05-update-message-files.ts +56 -56
  223. package/src/ruvocal/src/lib/migrations/routines/06-trim-message-updates.ts +56 -56
  224. package/src/ruvocal/src/lib/migrations/routines/08-update-featured-to-review.ts +32 -32
  225. package/src/ruvocal/src/lib/migrations/routines/09-delete-empty-conversations.spec.ts +214 -214
  226. package/src/ruvocal/src/lib/migrations/routines/09-delete-empty-conversations.ts +88 -88
  227. package/src/ruvocal/src/lib/migrations/routines/10-update-reports-assistantid.ts +29 -29
  228. package/src/ruvocal/src/lib/migrations/routines/index.ts +15 -15
  229. package/src/ruvocal/src/lib/server/__tests__/conversation-stop-generating.spec.ts +103 -103
  230. package/src/ruvocal/src/lib/server/abortRegistry.ts +57 -57
  231. package/src/ruvocal/src/lib/server/abortedGenerations.ts +43 -43
  232. package/src/ruvocal/src/lib/server/adminToken.ts +62 -62
  233. package/src/ruvocal/src/lib/server/api/__tests__/conversations-id.spec.ts +296 -296
  234. package/src/ruvocal/src/lib/server/api/__tests__/conversations-message.spec.ts +216 -216
  235. package/src/ruvocal/src/lib/server/api/__tests__/conversations.spec.ts +235 -235
  236. package/src/ruvocal/src/lib/server/api/__tests__/misc.spec.ts +72 -72
  237. package/src/ruvocal/src/lib/server/api/__tests__/testHelpers.ts +86 -86
  238. package/src/ruvocal/src/lib/server/api/__tests__/user-reports.spec.ts +78 -78
  239. package/src/ruvocal/src/lib/server/api/__tests__/user.spec.ts +239 -239
  240. package/src/ruvocal/src/lib/server/api/types.ts +37 -37
  241. package/src/ruvocal/src/lib/server/api/utils/requireAuth.ts +22 -22
  242. package/src/ruvocal/src/lib/server/api/utils/resolveConversation.ts +69 -69
  243. package/src/ruvocal/src/lib/server/api/utils/resolveModel.ts +27 -27
  244. package/src/ruvocal/src/lib/server/api/utils/superjsonResponse.ts +15 -15
  245. package/src/ruvocal/src/lib/server/apiToken.ts +11 -11
  246. package/src/ruvocal/src/lib/server/auth.ts +554 -554
  247. package/src/ruvocal/src/lib/server/config.ts +187 -187
  248. package/src/ruvocal/src/lib/server/conversation.ts +83 -83
  249. package/src/ruvocal/src/lib/server/database/__tests__/rvf.spec.ts +709 -709
  250. package/src/ruvocal/src/lib/server/database/postgres.ts +700 -700
  251. package/src/ruvocal/src/lib/server/database/rvf.ts +1078 -1078
  252. package/src/ruvocal/src/lib/server/database.ts +145 -145
  253. package/src/ruvocal/src/lib/server/endpoints/document.ts +68 -68
  254. package/src/ruvocal/src/lib/server/endpoints/endpoints.ts +43 -43
  255. package/src/ruvocal/src/lib/server/endpoints/images.ts +211 -211
  256. package/src/ruvocal/src/lib/server/endpoints/openai/endpointOai.ts +266 -266
  257. package/src/ruvocal/src/lib/server/endpoints/openai/openAIChatToTextGenerationStream.ts +212 -212
  258. package/src/ruvocal/src/lib/server/endpoints/openai/openAICompletionToTextGenerationStream.ts +32 -32
  259. package/src/ruvocal/src/lib/server/endpoints/preprocessMessages.ts +61 -61
  260. package/src/ruvocal/src/lib/server/exitHandler.ts +59 -59
  261. package/src/ruvocal/src/lib/server/files/downloadFile.ts +34 -34
  262. package/src/ruvocal/src/lib/server/files/uploadFile.ts +29 -29
  263. package/src/ruvocal/src/lib/server/findRepoRoot.ts +13 -13
  264. package/src/ruvocal/src/lib/server/generateFromDefaultEndpoint.ts +46 -46
  265. package/src/ruvocal/src/lib/server/hooks/error.ts +37 -37
  266. package/src/ruvocal/src/lib/server/hooks/fetch.ts +22 -22
  267. package/src/ruvocal/src/lib/server/hooks/handle.ts +250 -250
  268. package/src/ruvocal/src/lib/server/hooks/init.ts +51 -51
  269. package/src/ruvocal/src/lib/server/isURLLocal.spec.ts +31 -31
  270. package/src/ruvocal/src/lib/server/isURLLocal.ts +74 -74
  271. package/src/ruvocal/src/lib/server/logger.ts +42 -42
  272. package/src/ruvocal/src/lib/server/mcp/clientPool.spec.ts +175 -175
  273. package/src/ruvocal/src/lib/server/mcp/hf.ts +32 -32
  274. package/src/ruvocal/src/lib/server/mcp/httpClient.ts +122 -122
  275. package/src/ruvocal/src/lib/server/mcp/registry.ts +76 -76
  276. package/src/ruvocal/src/lib/server/mcp/tools.ts +196 -196
  277. package/src/ruvocal/src/lib/server/metrics.ts +255 -255
  278. package/src/ruvocal/src/lib/server/models.ts +518 -518
  279. package/src/ruvocal/src/lib/server/requestContext.ts +55 -55
  280. package/src/ruvocal/src/lib/server/router/arch.ts +230 -230
  281. package/src/ruvocal/src/lib/server/router/endpoint.ts +316 -316
  282. package/src/ruvocal/src/lib/server/router/multimodal.ts +28 -28
  283. package/src/ruvocal/src/lib/server/router/policy.ts +49 -49
  284. package/src/ruvocal/src/lib/server/router/toolsRoute.ts +51 -51
  285. package/src/ruvocal/src/lib/server/router/types.ts +21 -21
  286. package/src/ruvocal/src/lib/server/sendSlack.ts +23 -23
  287. package/src/ruvocal/src/lib/server/textGeneration/generate.ts +258 -258
  288. package/src/ruvocal/src/lib/server/textGeneration/index.ts +96 -96
  289. package/src/ruvocal/src/lib/server/textGeneration/mcp/fileRefs.ts +155 -155
  290. package/src/ruvocal/src/lib/server/textGeneration/mcp/routerResolution.ts +108 -108
  291. package/src/ruvocal/src/lib/server/textGeneration/mcp/runMcpFlow.ts +831 -831
  292. package/src/ruvocal/src/lib/server/textGeneration/mcp/toolInvocation.ts +349 -349
  293. package/src/ruvocal/src/lib/server/textGeneration/mcp/wasmTools.test.ts +633 -633
  294. package/src/ruvocal/src/lib/server/textGeneration/reasoning.ts +23 -23
  295. package/src/ruvocal/src/lib/server/textGeneration/title.ts +83 -83
  296. package/src/ruvocal/src/lib/server/textGeneration/types.ts +28 -28
  297. package/src/ruvocal/src/lib/server/textGeneration/utils/prepareFiles.ts +88 -88
  298. package/src/ruvocal/src/lib/server/textGeneration/utils/routing.ts +21 -21
  299. package/src/ruvocal/src/lib/server/textGeneration/utils/toolPrompt.ts +49 -49
  300. package/src/ruvocal/src/lib/server/urlSafety.ts +77 -77
  301. package/src/ruvocal/src/lib/server/usageLimits.ts +30 -30
  302. package/src/ruvocal/src/lib/stores/autopilotStore.svelte.ts +175 -175
  303. package/src/ruvocal/src/lib/stores/backgroundGenerations.svelte.ts +32 -32
  304. package/src/ruvocal/src/lib/stores/backgroundGenerations.ts +1 -1
  305. package/src/ruvocal/src/lib/stores/errors.ts +9 -9
  306. package/src/ruvocal/src/lib/stores/isAborted.ts +3 -3
  307. package/src/ruvocal/src/lib/stores/isPro.ts +4 -4
  308. package/src/ruvocal/src/lib/stores/loading.ts +3 -3
  309. package/src/ruvocal/src/lib/stores/mcpServers.ts +534 -534
  310. package/src/ruvocal/src/lib/stores/pendingChatInput.ts +3 -3
  311. package/src/ruvocal/src/lib/stores/pendingMessage.ts +9 -9
  312. package/src/ruvocal/src/lib/stores/settings.ts +182 -182
  313. package/src/ruvocal/src/lib/stores/shareModal.ts +13 -13
  314. package/src/ruvocal/src/lib/stores/titleUpdate.ts +8 -8
  315. package/src/ruvocal/src/lib/stores/wasmMcp.ts +472 -472
  316. package/src/ruvocal/src/lib/switchTheme.ts +124 -124
  317. package/src/ruvocal/src/lib/types/AbortedGeneration.ts +8 -8
  318. package/src/ruvocal/src/lib/types/Assistant.ts +31 -31
  319. package/src/ruvocal/src/lib/types/AssistantStats.ts +11 -11
  320. package/src/ruvocal/src/lib/types/ConfigKey.ts +4 -4
  321. package/src/ruvocal/src/lib/types/ConvSidebar.ts +9 -9
  322. package/src/ruvocal/src/lib/types/Conversation.ts +27 -27
  323. package/src/ruvocal/src/lib/types/ConversationStats.ts +13 -13
  324. package/src/ruvocal/src/lib/types/Message.ts +41 -41
  325. package/src/ruvocal/src/lib/types/MessageEvent.ts +10 -10
  326. package/src/ruvocal/src/lib/types/MessageUpdate.ts +139 -139
  327. package/src/ruvocal/src/lib/types/MigrationResult.ts +7 -7
  328. package/src/ruvocal/src/lib/types/Model.ts +23 -23
  329. package/src/ruvocal/src/lib/types/Report.ts +12 -12
  330. package/src/ruvocal/src/lib/types/Review.ts +6 -6
  331. package/src/ruvocal/src/lib/types/Semaphore.ts +19 -19
  332. package/src/ruvocal/src/lib/types/Session.ts +22 -22
  333. package/src/ruvocal/src/lib/types/Settings.ts +93 -93
  334. package/src/ruvocal/src/lib/types/SharedConversation.ts +9 -9
  335. package/src/ruvocal/src/lib/types/Template.ts +6 -6
  336. package/src/ruvocal/src/lib/types/Timestamps.ts +4 -4
  337. package/src/ruvocal/src/lib/types/TokenCache.ts +6 -6
  338. package/src/ruvocal/src/lib/types/Tool.ts +77 -77
  339. package/src/ruvocal/src/lib/types/UrlDependency.ts +5 -5
  340. package/src/ruvocal/src/lib/types/User.ts +14 -14
  341. package/src/ruvocal/src/lib/utils/PublicConfig.svelte.ts +75 -75
  342. package/src/ruvocal/src/lib/utils/auth.ts +17 -17
  343. package/src/ruvocal/src/lib/utils/chunk.ts +33 -33
  344. package/src/ruvocal/src/lib/utils/cookiesAreEnabled.ts +13 -13
  345. package/src/ruvocal/src/lib/utils/debounce.ts +17 -17
  346. package/src/ruvocal/src/lib/utils/deepestChild.ts +6 -6
  347. package/src/ruvocal/src/lib/utils/favicon.ts +21 -21
  348. package/src/ruvocal/src/lib/utils/fetchJSON.ts +23 -23
  349. package/src/ruvocal/src/lib/utils/file2base64.ts +14 -14
  350. package/src/ruvocal/src/lib/utils/formatUserCount.ts +37 -37
  351. package/src/ruvocal/src/lib/utils/generationState.spec.ts +75 -75
  352. package/src/ruvocal/src/lib/utils/generationState.ts +26 -26
  353. package/src/ruvocal/src/lib/utils/getHref.ts +41 -41
  354. package/src/ruvocal/src/lib/utils/getReturnFromGenerator.ts +7 -7
  355. package/src/ruvocal/src/lib/utils/haptics.ts +64 -64
  356. package/src/ruvocal/src/lib/utils/hashConv.ts +12 -12
  357. package/src/ruvocal/src/lib/utils/hf.ts +17 -17
  358. package/src/ruvocal/src/lib/utils/isDesktop.ts +7 -7
  359. package/src/ruvocal/src/lib/utils/isUrl.ts +8 -8
  360. package/src/ruvocal/src/lib/utils/isVirtualKeyboard.ts +16 -16
  361. package/src/ruvocal/src/lib/utils/loadAttachmentsFromUrls.ts +115 -115
  362. package/src/ruvocal/src/lib/utils/marked.spec.ts +96 -96
  363. package/src/ruvocal/src/lib/utils/marked.ts +531 -531
  364. package/src/ruvocal/src/lib/utils/mcpValidation.ts +147 -147
  365. package/src/ruvocal/src/lib/utils/mergeAsyncGenerators.ts +38 -38
  366. package/src/ruvocal/src/lib/utils/messageUpdates.spec.ts +262 -262
  367. package/src/ruvocal/src/lib/utils/messageUpdates.ts +324 -324
  368. package/src/ruvocal/src/lib/utils/mime.ts +56 -56
  369. package/src/ruvocal/src/lib/utils/models.ts +14 -14
  370. package/src/ruvocal/src/lib/utils/parseBlocks.ts +120 -120
  371. package/src/ruvocal/src/lib/utils/parseIncompleteMarkdown.ts +644 -644
  372. package/src/ruvocal/src/lib/utils/parseStringToList.ts +10 -10
  373. package/src/ruvocal/src/lib/utils/randomUuid.ts +14 -14
  374. package/src/ruvocal/src/lib/utils/searchTokens.ts +33 -33
  375. package/src/ruvocal/src/lib/utils/sha256.ts +7 -7
  376. package/src/ruvocal/src/lib/utils/stringifyError.ts +12 -12
  377. package/src/ruvocal/src/lib/utils/sum.ts +3 -3
  378. package/src/ruvocal/src/lib/utils/template.spec.ts +59 -59
  379. package/src/ruvocal/src/lib/utils/template.ts +53 -53
  380. package/src/ruvocal/src/lib/utils/timeout.ts +9 -9
  381. package/src/ruvocal/src/lib/utils/toolProgress.spec.ts +46 -46
  382. package/src/ruvocal/src/lib/utils/toolProgress.ts +11 -11
  383. package/src/ruvocal/src/lib/utils/tree/addChildren.spec.ts +102 -102
  384. package/src/ruvocal/src/lib/utils/tree/addChildren.ts +48 -48
  385. package/src/ruvocal/src/lib/utils/tree/addSibling.spec.ts +81 -81
  386. package/src/ruvocal/src/lib/utils/tree/addSibling.ts +41 -41
  387. package/src/ruvocal/src/lib/utils/tree/buildSubtree.spec.ts +110 -110
  388. package/src/ruvocal/src/lib/utils/tree/buildSubtree.ts +24 -24
  389. package/src/ruvocal/src/lib/utils/tree/convertLegacyConversation.spec.ts +31 -31
  390. package/src/ruvocal/src/lib/utils/tree/convertLegacyConversation.ts +36 -36
  391. package/src/ruvocal/src/lib/utils/tree/isMessageId.spec.ts +15 -15
  392. package/src/ruvocal/src/lib/utils/tree/isMessageId.ts +5 -5
  393. package/src/ruvocal/src/lib/utils/tree/tree.d.ts +14 -14
  394. package/src/ruvocal/src/lib/utils/tree/treeHelpers.spec.ts +167 -167
  395. package/src/ruvocal/src/lib/utils/updates.ts +39 -39
  396. package/src/ruvocal/src/lib/utils/urlParams.ts +13 -13
  397. package/src/ruvocal/src/lib/wasm/idb.ts +438 -438
  398. package/src/ruvocal/src/lib/wasm/index.ts +1213 -1213
  399. package/src/ruvocal/src/lib/wasm/tests/wasm-capabilities.test.ts +565 -565
  400. package/src/ruvocal/src/lib/wasm/wasm.worker.ts +332 -332
  401. package/src/ruvocal/src/lib/wasm/workerClient.ts +166 -166
  402. package/src/ruvocal/src/lib/workers/autopilotWorker.ts +221 -221
  403. package/src/ruvocal/src/lib/workers/detailFetchWorker.ts +100 -100
  404. package/src/ruvocal/src/lib/workers/markdownWorker.ts +61 -61
  405. package/src/ruvocal/src/routes/+error.svelte +20 -20
  406. package/src/ruvocal/src/routes/+layout.svelte +324 -324
  407. package/src/ruvocal/src/routes/+layout.ts +91 -91
  408. package/src/ruvocal/src/routes/+page.svelte +168 -168
  409. package/src/ruvocal/src/routes/.well-known/oauth-cimd/+server.ts +37 -37
  410. package/src/ruvocal/src/routes/__debug/openai/+server.ts +21 -21
  411. package/src/ruvocal/src/routes/admin/export/+server.ts +159 -159
  412. package/src/ruvocal/src/routes/admin/stats/compute/+server.ts +16 -16
  413. package/src/ruvocal/src/routes/api/conversation/[id]/+server.ts +40 -40
  414. package/src/ruvocal/src/routes/api/conversation/[id]/message/[messageId]/+server.ts +42 -42
  415. package/src/ruvocal/src/routes/api/conversations/+server.ts +48 -48
  416. package/src/ruvocal/src/routes/api/fetch-url/+server.ts +147 -147
  417. package/src/ruvocal/src/routes/api/mcp/health/+server.ts +292 -292
  418. package/src/ruvocal/src/routes/api/mcp/servers/+server.ts +32 -32
  419. package/src/ruvocal/src/routes/api/models/+server.ts +25 -25
  420. package/src/ruvocal/src/routes/api/transcribe/+server.ts +104 -104
  421. package/src/ruvocal/src/routes/api/user/+server.ts +15 -15
  422. package/src/ruvocal/src/routes/api/user/validate-token/+server.ts +20 -20
  423. package/src/ruvocal/src/routes/api/v2/conversations/+server.ts +48 -48
  424. package/src/ruvocal/src/routes/api/v2/conversations/[id]/+server.ts +94 -94
  425. package/src/ruvocal/src/routes/api/v2/conversations/[id]/message/[messageId]/+server.ts +43 -43
  426. package/src/ruvocal/src/routes/api/v2/conversations/import-share/+server.ts +23 -23
  427. package/src/ruvocal/src/routes/api/v2/debug/config/+server.ts +16 -16
  428. package/src/ruvocal/src/routes/api/v2/debug/refresh/+server.ts +30 -30
  429. package/src/ruvocal/src/routes/api/v2/export/+server.ts +196 -196
  430. package/src/ruvocal/src/routes/api/v2/feature-flags/+server.ts +14 -14
  431. package/src/ruvocal/src/routes/api/v2/models/+server.ts +38 -38
  432. package/src/ruvocal/src/routes/api/v2/models/[namespace]/+server.ts +8 -8
  433. package/src/ruvocal/src/routes/api/v2/models/[namespace]/[model]/+server.ts +8 -8
  434. package/src/ruvocal/src/routes/api/v2/models/[namespace]/[model]/subscribe/+server.ts +28 -28
  435. package/src/ruvocal/src/routes/api/v2/models/[namespace]/subscribe/+server.ts +28 -28
  436. package/src/ruvocal/src/routes/api/v2/models/old/+server.ts +7 -7
  437. package/src/ruvocal/src/routes/api/v2/models/refresh/+server.ts +33 -33
  438. package/src/ruvocal/src/routes/api/v2/public-config/+server.ts +7 -7
  439. package/src/ruvocal/src/routes/api/v2/user/+server.ts +17 -17
  440. package/src/ruvocal/src/routes/api/v2/user/billing-orgs/+server.ts +73 -73
  441. package/src/ruvocal/src/routes/api/v2/user/reports/+server.ts +17 -17
  442. package/src/ruvocal/src/routes/api/v2/user/settings/+server.ts +110 -110
  443. package/src/ruvocal/src/routes/conversation/+server.ts +115 -115
  444. package/src/ruvocal/src/routes/conversation/[id]/+page.svelte +586 -586
  445. package/src/ruvocal/src/routes/conversation/[id]/+page.ts +60 -60
  446. package/src/ruvocal/src/routes/conversation/[id]/+server.ts +740 -740
  447. package/src/ruvocal/src/routes/conversation/[id]/message/[messageId]/prompt/+server.ts +66 -66
  448. package/src/ruvocal/src/routes/conversation/[id]/share/+server.ts +69 -69
  449. package/src/ruvocal/src/routes/conversation/[id]/stop-generating/+server.ts +35 -35
  450. package/src/ruvocal/src/routes/healthcheck/+server.ts +3 -3
  451. package/src/ruvocal/src/routes/login/+server.ts +5 -5
  452. package/src/ruvocal/src/routes/login/callback/+server.ts +103 -103
  453. package/src/ruvocal/src/routes/login/callback/updateUser.spec.ts +157 -157
  454. package/src/ruvocal/src/routes/login/callback/updateUser.ts +215 -215
  455. package/src/ruvocal/src/routes/logout/+server.ts +18 -18
  456. package/src/ruvocal/src/routes/metrics/+server.ts +18 -18
  457. package/src/ruvocal/src/routes/models/+page.svelte +233 -233
  458. package/src/ruvocal/src/routes/models/[...model]/+page.svelte +161 -161
  459. package/src/ruvocal/src/routes/models/[...model]/+page.ts +14 -14
  460. package/src/ruvocal/src/routes/models/[...model]/thumbnail.png/+server.ts +64 -64
  461. package/src/ruvocal/src/routes/models/[...model]/thumbnail.png/ModelThumbnail.svelte +28 -28
  462. package/src/ruvocal/src/routes/privacy/+page.svelte +11 -11
  463. package/src/ruvocal/src/routes/r/[id]/+page.ts +34 -34
  464. package/src/ruvocal/src/routes/settings/(nav)/+layout.svelte +282 -282
  465. package/src/ruvocal/src/routes/settings/(nav)/+layout.ts +1 -1
  466. package/src/ruvocal/src/routes/settings/(nav)/+server.ts +59 -59
  467. package/src/ruvocal/src/routes/settings/(nav)/[...model]/+page.svelte +464 -464
  468. package/src/ruvocal/src/routes/settings/(nav)/[...model]/+page.ts +14 -14
  469. package/src/ruvocal/src/routes/settings/(nav)/application/+page.svelte +362 -362
  470. package/src/ruvocal/src/routes/settings/+layout.svelte +40 -40
  471. package/src/ruvocal/src/styles/highlight-js.css +195 -195
  472. package/src/ruvocal/src/styles/main.css +144 -144
  473. package/src/ruvocal/static/chatui/favicon-dark.svg +3 -3
  474. package/src/ruvocal/static/chatui/favicon-dev.svg +3 -3
  475. package/src/ruvocal/static/chatui/favicon.svg +3 -3
  476. package/src/ruvocal/static/chatui/icon.svg +3 -3
  477. package/src/ruvocal/static/chatui/logo.svg +7 -7
  478. package/src/ruvocal/static/chatui/manifest.json +54 -54
  479. package/src/ruvocal/static/chatui/welcome.js +184 -184
  480. package/src/ruvocal/static/huggingchat/favicon-dark.svg +4 -4
  481. package/src/ruvocal/static/huggingchat/favicon-dev.svg +4 -4
  482. package/src/ruvocal/static/huggingchat/favicon.svg +4 -4
  483. package/src/ruvocal/static/huggingchat/fulltext-logo.svg +1 -1
  484. package/src/ruvocal/static/huggingchat/icon.svg +4 -4
  485. package/src/ruvocal/static/huggingchat/logo.svg +4 -4
  486. package/src/ruvocal/static/huggingchat/manifest.json +54 -54
  487. package/src/ruvocal/static/huggingchat/routes.chat.json +226 -226
  488. package/src/ruvocal/static/robots.txt +10 -10
  489. package/src/ruvocal/static/wasm/rvagent_wasm.js +1539 -1539
  490. package/src/ruvocal/stub/@reflink/reflink/package.json +5 -5
  491. package/src/ruvocal/svelte.config.js +53 -53
  492. package/src/ruvocal/tailwind.config.cjs +30 -30
  493. package/src/ruvocal/tsconfig.json +19 -19
  494. package/src/ruvocal/vite.config.ts +87 -87
  495. package/src/scripts/deploy.sh +116 -116
  496. package/src/scripts/generate-config.js +245 -245
  497. package/src/scripts/generate-welcome.js +187 -187
  498. package/src/scripts/package-rvf.sh +116 -116
@@ -0,0 +1,100 @@
1
+ [2026-05-05T02:58:33.227Z] RESULT
2
+ ============================================================
3
+ {
4
+ "success": true,
5
+ "output": "Here is the full coverage gap analysis with test skeletons for each area.\n\n---\n\n## Test Coverage Gap Analysis\n\n### 1. `src/lib/APIClient.ts` — Zero test coverage\n\n**Untested:** `apiCall` (all branches), `handleResponse`, query param serialization, error body parsing fallback chain, ObjectId superjson registration.\n\n```typescript\n// src/lib/APIClient.spec.ts\nimport { describe, it, expect, vi, afterEach } from \"vitest\";\nimport { handleResponse, useAPIClient } from \"./APIClient\";\n\nconst mockFetch = (status: number, body: string | null, ok = status < 400) =>\n vi.fn().mockResolvedValue({\n ok,\n status,\n text: () => Promise.resolve(body ?? \"\"),\n json: () => (body ? Promise.resolve(JSON.parse(body)) : Promise.reject(new Error(\"no body\"))),\n statusText: \"Error\",\n });\n\ndescribe(\"apiCall — success paths\", () => {\n it(\"sends GET with no body\", async () => {\n const fetch = mockFetch(200, '{\"json\":\"body\"}');\n const client = useAPIClient({ fetch, origin: \"http://localhost\" });\n await client[\"public-config\"].get();\n expect(fetch).toHaveBeenCalledWith(\n expect.stringContaining(\"/api/v2/public-config\"),\n expect.objectContaining({ method: \"GET\" })\n );\n });\n\n it(\"attaches query params to GET URL\", async () => {\n const fetch = mockFetch(200, '{\"x\":1}');\n const client = useAPIClient({ fetch, origin: \"http://localhost\" });\n await client[\"public-config\"].get({ query: { page: 2, skip: undefined } });\n const url = fetch.mock.calls[0][0] as string;\n expect(url).toContain(\"page=2\");\n expect(url).not.toContain(\"skip\"); // undefined values are omitted\n });\n\n it(\"returns null data on empty 200 body\", async () => {\n const fetch = mockFetch(200, \"\");\n const client = useAPIClient({ fetch, origin: \"http://localhost\" });\n const res = await client.user.settings.get();\n expect(res.data).toBeNull();\n expect(res.error).toBeNull();\n });\n\n it(\"serialises body as JSON for POST\", async () => {\n const fetch = mockFetch(200, \"null\");\n const client = useAPIClient({ fetch, origin: \"http://localhost\" });\n await client.user.settings.post({ theme: \"dark\" });\n const init = fetch.mock.calls[0][1] as RequestInit;\n expect(init.headers).toMatchObject({ \"Content-Type\": \"application/json\" });\n expect(JSON.parse(init.body as string)).toEqual({ theme: \"dark\" });\n });\n});\n\ndescribe(\"apiCall — error paths\", () => {\n it(\"returns error with JSON body on 4xx\", async () => {\n const fetch = mockFetch(400, '{\"message\":\"bad request\"}');\n const client = useAPIClient({ fetch, origin: \"http://localhost\" });\n const res = await client.user.get();\n expect(res.error).toEqual({ message: \"bad request\" });\n expect(res.data).toBeNull();\n expect(res.status).toBe(400);\n });\n\n it(\"falls back to text when error body is not JSON\", async () => {\n const fetch = vi.fn().mockResolvedValue({\n ok: false,\n status: 500,\n json: () => Promise.reject(new Error(\"not json\")),\n text: () => Promise.resolve(\"Internal Server Error\"),\n statusText: \"ISE\",\n });\n const client = useAPIClient({ fetch, origin: \"http://localhost\" });\n const res = await client.user.get();\n expect(res.error).toBe(\"Internal Server Error\");\n });\n\n it(\"falls back to statusText when text() also fails\", async () => {\n const fetch = vi.fn().mockResolvedValue({\n ok: false,\n status: 503,\n json: () => Promise.reject(new Error()),\n text: () => Promise.reject(new Error()),\n statusText: \"Service Unavailable\",\n });\n const client = useAPIClient({ fetch, origin: \"http://localhost\" });\n const res = await client.user.get();\n expect(res.error).toBe(\"Service Unavailable\");\n });\n});\n\ndescribe(\"handleResponse\", () => {\n it(\"returns null when data is null and no error\", () => {\n expect(handleResponse({ data: null, error: null, status: 204 })).toBeNull();\n });\n\n it(\"throws when response contains an error\", () => {\n expect(() =>\n handleResponse({ data: null, error: { message: \"forbidden\" }, status: 403 })\n ).toThrow();\n });\n\n it(\"parses superjson string data\", () => {\n const payload = JSON.stringify({ json: { value: 42 }, meta: undefined });\n expect(handleResponse({ data: payload, error: null, status: 200 })).toEqual({ value: 42 });\n });\n});\n\ndescribe(\"useAPIClient — URL construction\", () => {\n it(\"uses provided origin for SSR base URL\", async () => {\n const fetch = mockFetch(200, \"null\");\n const client = useAPIClient({ fetch, origin: \"http://api.internal:4000\" });\n await client.export.get();\n expect(fetch.mock.calls[0][0]).toContain(\"http://api.internal:4000\");\n });\n\n it(\"builds nested conversation message endpoint\", async () => {\n const fetch = mockFetch(200, \"null\");\n const client = useAPIClient({ fetch, origin: \"http://localhost\" });\n await client.conversations({ id: \"abc123\" }).message({ messageId: \"msg456\" }).get();\n expect(fetch.mock.calls[0][0]).toContain(\"/conversations/abc123/message/msg456\");\n });\n});\n```\n\n---\n\n### 2. `src/lib/actions/clickOutside.ts` — Zero test coverage\n\n**Untested:** click outside triggers callback, click inside does not, `update` replaces callback, `destroy` removes listener.\n\n```typescript\n// src/lib/actions/clickOutside.spec.ts\nimport { describe, it, expect, vi, beforeEach } from \"vitest\";\nimport { clickOutside } from \"./clickOutside\";\n\ndescribe(\"clickOutside action\", () => {\n let element: HTMLDivElement;\n let outsideElement: HTMLDivElement;\n let callback: ReturnType<typeof vi.fn>;\n\n beforeEach(() => {\n document.body.innerHTML = \"\";\n element = document.createElement(\"div\");\n outsideElement = document.createElement(\"div\");\n document.body.appendChild(element);\n document.body.appendChild(outsideElement);\n callback = vi.fn();\n });\n\n it(\"fires callback when clicking outside the element\", () => {\n clickOutside(element, callback);\n outsideElement.dispatchEvent(new MouseEvent(\"click\", { bubbles: true }));\n expect(callback).toHaveBeenCalledOnce();\n });\n\n it(\"does not fire callback when clicking inside the element\", () => {\n clickOutside(element, callback);\n element.dispatchEvent(new MouseEvent(\"click\", { bubbles: true }));\n expect(callback).not.toHaveBeenCalled();\n });\n\n it(\"does not fire when clicking the element itself\", () => {\n clickOutside(element, callback);\n const inner = document.createElement(\"span\");\n element.appendChild(inner);\n inner.dispatchEvent(new MouseEvent(\"click\", { bubbles: true }));\n expect(callback).not.toHaveBeenCalled();\n });\n\n it(\"update() replaces the callback\", () => {\n const action = clickOutside(element, callback);\n const newCallback = vi.fn();\n action.update(newCallback);\n outsideElement.dispatchEvent(new MouseEvent(\"click\", { bubbles: true }));\n expect(callback).not.toHaveBeenCalled();\n expect(newCallback).toHaveBeenCalledOnce();\n });\n\n it(\"destroy() stops all future callbacks\", () => {\n const action = clickOutside(element, callback);\n action.destroy();\n outsideElement.dispatchEvent(new MouseEvent(\"click\", { bubbles: true }));\n expect(callback).not.toHaveBeenCalled();\n });\n\n it(\"handles multiple destroy() calls without error\", () => {\n const action = clickOutside(element, callback);\n expect(() => { action.destroy(); action.destroy(); }).not.toThrow();\n });\n});\n```\n\n---\n\n### 3. `src/lib/buildPrompt.ts` — Zero test coverage\n\n**Untested:** system message replacement with preprompt, no system message scenario, truncation slicing, zero truncate parameter (no slicing).\n\n```typescript\n// src/lib/buildPrompt.spec.ts\nimport { describe, it, expect } from \"vitest\";\nimport { buildPrompt } from \"./buildPrompt\";\nimport type { BackendModel } from \"./server/models\";\n\nconst makeModel = (truncate?: number): BackendModel =>\n ({\n chatPromptRender: ({ messages, preprompt }: { messages: unknown[]; preprompt?: string }) =>\n messages.map((m: any) => `${m.role}:${m.content}`).join(\" \"),\n parameters: truncate !== undefined ? { truncate } : undefined,\n } as unknown as BackendModel);\n\ndescribe(\"buildPrompt\", () => {\n it(\"replaces system message content with preprompt when first message is system\", async () => {\n const model = makeModel(1000);\n const messages = [\n { from: \"system\" as const, content: \"old system prompt\" },\n { from: \"user\" as const, content: \"hello\" },\n ];\n const result = await buildPrompt({ messages, model, preprompt: \"new system prompt\" });\n expect(result).toContain(\"new system prompt\");\n expect(result).not.toContain(\"old system prompt\");\n });\n\n it(\"does not replace content when first message is not system\", async () => {\n const model = makeModel(1000);\n const messages = [{ from: \"user\" as const, content: \"hello\" }];\n await buildPrompt({ messages, model, preprompt: \"some preprompt\" });\n // original content unchanged\n expect(messages[0].content).toBe(\"hello\");\n });\n\n it(\"does not replace content when preprompt is undefined\", async () => {\n const model = makeModel(1000);\n const messages = [{ from: \"system\" as const, content: \"original\" }];\n await buildPrompt({ messages, model, preprompt: undefined });\n expect(messages[0].content).toBe(\"original\");\n });\n\n it(\"truncates to the last N words when truncate is set\", async () => {\n const model = makeModel(3);\n const messages = [{ from: \"user\" as const, content: \"one two three four five\" }];\n const result = await buildPrompt({ messages, model });\n const wordCount = result.trim().split(/\\s+/).length;\n expect(wordCount).toBeLessThanOrEqual(3);\n });\n\n it(\"returns full prompt when truncate is 0 (no slicing)\", async () => {\n const model = makeModel(0);\n const messages = [{ from: \"user\" as const, content: \"word1 word2 word3\" }];\n const result = await buildPrompt({ messages, model });\n // slice(0) returns empty array → join → empty string\n expect(result).toBe(\"\");\n });\n\n it(\"returns full prompt when model has no truncate parameter\", async () => {\n const model = makeModel(undefined);\n const messages = [{ from: \"user\" as const, content: \"word1 word2 word3\" }];\n const result = await buildPrompt({ messages, model });\n expect(result).toContain(\"word1\");\n });\n\n it(\"maps message from to role field\", async () => {\n const receivedMessages: unknown[] = [];\n const model = {\n chatPromptRender: ({ messages }: { messages: unknown[] }) => {\n receivedMessages.push(...messages);\n return \"\";\n },\n parameters: { truncate: 1000 },\n } as unknown as BackendModel;\n await buildPrompt({\n messages: [{ from: \"assistant\" as const, content: \"hi\" }],\n model,\n });\n expect((receivedMessages[0] as any).role).toBe(\"assistant\");\n });\n});\n```\n\n---\n\n### 4. `src/lib/createShareLink.ts` — Zero test coverage\n\n**Untested:** 7-char ID shortcut path, successful API call, API failure with text body, API failure with empty body, URL prefix priority logic.\n\n```typescript\n// src/lib/createShareLink.spec.ts\nimport { describe, it, expect, vi, beforeEach } from \"vitest\";\n\n// Mock SvelteKit modules before importing\nvi.mock(\"$app/paths\", () => ({ base: \"\" }));\nvi.mock(\"$app/state\", () => ({\n page: {\n data: { publicConfig: { PUBLIC_SHARE_PREFIX: \"\", PUBLIC_ORIGIN: \"https://app.example.com\" } },\n url: { origin: \"https://fallback.example.com\" },\n },\n}));\n\nimport { createShareLink } from \"./createShareLink\";\n\ndescribe(\"createShareLink\", () => {\n beforeEach(() => {\n vi.resetAllMocks();\n });\n\n it(\"returns share URL directly for 7-char id without fetching\", async () => {\n const fetchSpy = vi.spyOn(globalThis, \"fetch\");\n const url = await createShareLink(\"abc1234\");\n expect(fetchSpy).not.toHaveBeenCalled();\n expect(url).toBe(\"https://app.example.com/r/abc1234\");\n });\n\n it(\"calls the share endpoint for non-7-char ids\", async () => {\n vi.spyOn(globalThis, \"fetch\").mockResolvedValueOnce({\n ok: true,\n json: () => Promise.resolve({ shareId: \"xyz9876\" }),\n } as Response);\n const url = await createShareLink(\"longconversationid\");\n expect(url).toBe(\"https://app.example.com/r/xyz9876\");\n });\n\n it(\"throws when the share endpoint returns non-ok with a text body\", async () => {\n vi.spyOn(globalThis, \"fetch\").mockResolvedValueOnce({\n ok: false,\n text: () => Promise.resolve(\"Conversation not found\"),\n } as unknown as Response);\n await expect(createShareLink(\"longconversationid\")).rejects.toThrow(\"Conversation not found\");\n });\n\n it(\"throws generic message when error body is empty\", async () => {\n vi.spyOn(globalThis, \"fetch\").mockResolvedValueOnce({\n ok: false,\n text: () => Promise.resolve(\"\"),\n } as unknown as Response);\n await expect(createShareLink(\"longconversationid\")).rejects.toThrow(\n \"Failed to create share link\"\n );\n });\n\n it(\"uses PUBLIC_SHARE_PREFIX when set\", async () => {\n const { page } = await import(\"$app/state\");\n (page.data.publicConfig as any).PUBLIC_SHARE_PREFIX = \"https://cdn.share.example.com\";\n const url = await createShareLink(\"abc1234\");\n expect(url).toStartWith(\"https://cdn.share.example.com\");\n (page.data.publicConfig as any).PUBLIC_SHARE_PREFIX = \"\";\n });\n});\n```\n\n---\n\n### 5. `src/lib/migrations/lock.ts` — Gaps in existing tests\n\nThe existing suite covers the happy path and concurrent acquisition. Missing:\n\n| Gap | Why it matters |\n|---|---|\n| `releaseLock` with mismatched `lockId` | Should not delete the wrong lock |\n| `refreshLock` on a non-existent lock | Returns `false` — not asserted anywhere |\n| `isDBLocked` with a different key | Isolation between semaphore keys |\n| `releaseLock` when lock already released | Should be idempotent |\n\n```typescript\n// Add to src/lib/migrations/migrations.spec.ts (inside the describe block)\n\nit(\"releaseLock does not delete lock held by different id\", async () => {\n const lockId = await acquireLock(Semaphores.TEST_MIGRATION);\n assert(lockId);\n const fakeId = new ObjectId();\n await releaseLock(Semaphores.TEST_MIGRATION, fakeId); // wrong id\n expect(await isDBLocked(Semaphores.TEST_MIGRATION)).toBe(true);\n await releaseLock(Semaphores.TEST_MIGRATION, lockId); // cleanup\n});\n\nit(\"refreshLock returns false when lock does not exist\", async () => {\n const fakeId = new ObjectId();\n const result = await refreshLock(Semaphores.TEST_MIGRATION, fakeId);\n expect(result).toBe(false);\n});\n\nit(\"isDBLocked is key-scoped — one key locked does not affect another\", async () => {\n const lockId = await acquireLock(Semaphores.TEST_MIGRATION);\n assert(lockId);\n expect(await isDBLocked(\"other-key\")).toBe(false);\n await releaseLock(Semaphores.TEST_MIGRATION, lockId);\n});\n\nit(\"releaseLock is idempotent — releasing twice does not throw\", async () => {\n const lockId = await acquireLock(Semaphores.TEST_MIGRATION);\n assert(lockId);\n await releaseLock(Semaphores.TEST_MIGRATION, lockId);\n await expect(releaseLock(Semaphores.TEST_MIGRATION, lockId)).resolves.not.toThrow();\n});\n```\n\n---\n\n### 6. `src/lib/migrations/migrations.ts` — Zero test coverage for `checkAndRunMigrations`\n\n**Untested:** duplicate GUID detection, lock contention wait-loop, `runEveryTime`, `runForHuggingChat` guards, migration failure recording, lock refresh interval, lock release on completion.\n\n```typescript\n// src/lib/migrations/checkAndRunMigrations.spec.ts\nimport { beforeEach, describe, expect, it, vi } from \"vitest\";\nimport type { Database } from \"$lib/server/database\";\nimport { Semaphores } from \"$lib/types/Semaphore\";\nimport ObjectId from \"bson-objectid\";\n\n// Stubs for all side-effecting modules\nvi.mock(\"./lock\");\nvi.mock(\"$lib/server/logger\", () => ({ logger: { debug: vi.fn(), error: vi.fn() } }));\nvi.mock(\"$lib/server/config\", () => ({ config: { isHuggingChat: false } }));\n\nimport * as lockModule from \"./lock\";\nimport { checkAndRunMigrations } from \"./migrations\";\n\nconst makeDb = (results: unknown[] = []) => ({\n getCollections: () => ({\n migrationResults: {\n find: () => ({ toArray: () => Promise.resolve(results) }),\n updateOne: vi.fn().mockResolvedValue({}),\n },\n }),\n});\n\nvi.mock(\"$lib/server/database\", () => ({\n Database: { getInstance: vi.fn() },\n}));\n\nimport { Database } from \"$lib/server/database\";\n\ndescribe(\"checkAndRunMigrations\", () => {\n beforeEach(() => {\n vi.clearAllMocks();\n vi.mocked(Database.getInstance).mockResolvedValue(makeDb() as unknown as Database);\n vi.mocked(lockModule.acquireLock).mockResolvedValue(new ObjectId());\n vi.mocked(lockModule.releaseLock).mockResolvedValue(undefined);\n vi.mocked(lockModule.isDBLocked).mockResolvedValue(false);\n vi.mocked(lockModule.refreshLock).mockResolvedValue(true);\n });\n\n it(\"throws when duplicate migration GUIDs exist\", async () => {\n // Override routines to inject duplicates — easiest with vi.doMock inside test\n await expect(async () => {\n vi.doMock(\"./routines\", () => {\n const id = new ObjectId();\n return { migrations: [{ _id: id, name: \"dup\", up: vi.fn() }, { _id: id, name: \"dup2\", up: vi.fn() }] };\n });\n const { checkAndRunMigrations: fn } = await import(\"./migrations?dup=1\");\n await fn();\n }).rejects.toThrow(\"Duplicate migration GUIDs found\");\n });\n\n it(\"waits for DB unlock when lock is already held\", async () => {\n vi.mocked(lockModule.acquireLock).mockResolvedValueOnce(false);\n // isDBLocked returns true once, then false to exit the while loop\n vi.mocked(lockModule.isDBLocked)\n .mockResolvedValueOnce(true)\n .mockResolvedValueOnce(false);\n await checkAndRunMigrations();\n expect(lockModule.isDBLocked).toHaveBeenCalledWith(Semaphores.MIGRATION);\n });\n\n it(\"skips migration already in results\", async () => {\n const id = new ObjectId();\n vi.mocked(Database.getInstance).mockResolvedValue(\n makeDb([{ _id: id, name: \"done\", status: \"success\" }]) as unknown as Database\n );\n const upFn = vi.fn().mockResolvedValue(true);\n vi.doMock(\"./routines\", () => ({\n migrations: [{ _id: id, name: \"done\", up: upFn, runEveryTime: false }],\n }));\n const { checkAndRunMigrations: fn } = await import(\"./migrations?skip=1\");\n await fn();\n expect(upFn).not.toHaveBeenCalled();\n });\n\n it(\"records failure status when migration throws\", async () => {\n const id = new ObjectId();\n const updateOne = vi.fn().mockResolvedValue({});\n vi.mocked(Database.getInstance).mockResolvedValue({\n getCollections: () => ({\n migrationResults: { find: () => ({ toArray: () => Promise.resolve([]) }), updateOne },\n }),\n } as unknown as Database);\n vi.doMock(\"./routines\", () => ({\n migrations: [{ _id: id, name: \"failing\", up: vi.fn().mockRejectedValue(new Error(\"boom\")) }],\n }));\n const { checkAndRunMigrations: fn } = await import(\"./migrations?fail=1\");\n await fn();\n const lastCall = updateOne.mock.calls.at(-1)?.[1].$set;\n expect(lastCall?.status).toBe(\"failure\");\n });\n\n it(\"skips runForHuggingChat=only migration when not HuggingChat\", async () => {\n const { config } = await import(\"$lib/server/config\");\n (config as any).isHuggingChat = false;\n const upFn = vi.fn().mockResolvedValue(true);\n const id = new ObjectId();\n vi.doMock(\"./routines\", () => ({\n migrations: [{ _id: id, name: \"hc-only\", up: upFn, runForHuggingChat: \"only\" }],\n }));\n const { checkAndRunMigrations: fn } = await import(\"./migrations?hconly=1\");\n await fn();\n expect(upFn).not.toHaveBeenCalled();\n });\n\n it(\"releases lock and clears interval after all migrations run\", async () => {\n vi.useFakeTimers();\n await checkAndRunMigrations();\n vi.useRealTimers();\n expect(lockModule.releaseLock).toHaveBeenCalledWith(\n Semaphores.MIGRATION,\n expect.any(ObjectId)\n );\n });\n});\n```\n\n---\n\n### 7. `src/lib/constants/rvagentPresets.ts` — Utility functions untested\n\n**Untested:** `getPresetById`, `buildPresetUrl`, `buildPresetCliCommand`.\n\n```typescript\n// src/lib/constants/rvagentPresets.spec.ts\nimport { describe, it, expect } from \"vitest\";\nimport {\n getPresetById,\n buildPresetUrl,\n buildPresetCliCommand,\n RVAGENT_PRESETS,\n} from \"./rvagentPresets\";\n\ndescribe(\"getPresetById\", () => {\n it(\"returns the matching preset\", () => {\n const preset = getPresetById(\"all-tools\");\n expect(preset).toBeDefined();\n expect(preset?.id).toBe(\"all-tools\");\n });\n\n it(\"returns undefined for an unknown id\", () => {\n expect(getPresetById(\"nonexistent\")).toBeUndefined();\n });\n\n it(\"all preset IDs in RVAGENT_PRESETS are findable\", () => {\n for (const p of RVAGENT_PRESETS) {\n expect(getPresetById(p.id)).toBe(p);\n }\n });\n});\n\ndescribe(\"buildPresetUrl\", () => {\n it(\"uses preset defaultPort when no port override is given\", () => {\n const preset = getPresetById(\"all-tools\")!;\n expect(buildPresetUrl(preset)).toBe(`http://localhost:${preset.defaultPort}/sse`);\n });\n\n it(\"uses the overridden port\", () => {\n const preset = getPresetById(\"all-tools\")!;\n expect(buildPresetUrl(preset, \"localhost\", 9999)).toBe(\"http://localhost:9999/sse\");\n });\n\n it(\"uses custom host\", () => {\n const preset = getPresetById(\"all-tools\")!;\n expect(buildPresetUrl(preset, \"192.168.1.10\")).toContain(\"192.168.1.10\");\n });\n});\n\ndescribe(\"buildPresetCliCommand\", () => {\n it(\"uses --all flag for the all-tools preset\", () => {\n const preset = getPresetById(\"all-tools\")!;\n const cmd = buildPresetCliCommand(preset);\n expect(cmd).toContain(\"--all\");\n expect(cmd).not.toContain(\"--groups\");\n });\n\n it(\"uses --groups for presets with specific tool groups\", () => {\n const preset = getPresetById(\"file-shell\")!;\n const cmd = buildPresetCliCommand(preset);\n expect(cmd).toContain(\"--groups file,shell\");\n expect(cmd).not.toContain(\"--all\");\n });\n\n it(\"uses custom port in the command\", () => {\n const preset = getPresetById(\"memory-agent\")!;\n const cmd = buildPresetCliCommand(preset, 8888);\n expect(cmd).toContain(\"--port 8888\");\n });\n\n it(\"falls back to defaultPort in the command when no port given\", () => {\n const preset = getPresetById(\"memory-agent\")!;\n const cmd = buildPresetCliCommand(preset);\n expect(cmd).toContain(`--port ${preset.defaultPort}`);\n });\n});\n```\n\n---\n\n### 8. `src/lib/components/chat/MarkdownRenderer.svelte.test.ts` — Edge case gaps\n\n**Missing:** XSS via `javascript:` links, empty content, bold/italic inside lists, tables, image rendering, multiple code blocks with language labels.\n\n```typescript\n// Add to existing MarkdownRenderer.svelte.test.ts\n\ndescribe(\"MarkdownRenderer — security\", () => {\n it(\"strips javascript: href from links\", () => {\n const { baseElement } = render(MarkdownRenderer, {\n content: \"[click me](javascript:alert(1))\",\n });\n const anchor = baseElement.querySelector(\"a\");\n expect(anchor?.getAttribute(\"href\")).not.toContain(\"javascript:\");\n });\n\n it(\"renders empty content without crashing\", () => {\n expect(() => render(MarkdownRenderer, { content: \"\" })).not.toThrow();\n });\n});\n\ndescribe(\"MarkdownRenderer — tables\", () => {\n it(\"renders a markdown table\", () => {\n const md = \"| A | B |\\n|---|---|\\n| 1 | 2 |\";\n const { baseElement } = render(MarkdownRenderer, { content: md });\n expect(baseElement.querySelector(\"table\")).toBeTruthy();\n expect(baseElement.querySelector(\"thead\")).toBeTruthy();\n expect(baseElement.querySelector(\"tbody\")).toBeTruthy();\n });\n});\n\ndescribe(\"MarkdownRenderer — nested formatting\", () => {\n it(\"renders bold inside a list item\", () => {\n const { baseElement } = render(MarkdownRenderer, { content: \"- **bold item**\" });\n const strong = baseElement.querySelector(\"strong\");\n expect(strong).toBeTruthy();\n expect(strong?.textContent).toBe(\"bold item\");\n });\n\n it(\"renders multiple fenced code blocks with language labels\", () => {\n const md = \"```ts\\nconst x = 1;\\n```\\n\\n```py\\nprint('hi')\\n```\";\n const { baseElement } = render(MarkdownRenderer, { content: md });\n const codeBlocks = baseElement.querySelectorAll(\"code\");\n expect(codeBlocks.length).toBeGreaterThanOrEqual(2);\n });\n});\n\ndescribe(\"MarkdownRenderer — images\", () => {\n it(\"renders an image with alt text\", () => {\n const { baseElement } = render(MarkdownRenderer, {\n content: \"![Alt text](https://example.com/img.png)\",\n });\n const img = baseElement.querySelector(\"img\");\n expect(img).toBeTruthy();\n expect(img?.getAttribute(\"alt\")).toBe(\"Alt text\");\n });\n});\n```\n\n---\n\n### Summary of Gaps\n\n| File | Coverage | Priority gaps |\n|---|---|---|\n| `APIClient.ts` | None | Error fallback chain, empty body, query filtering, `handleResponse` |\n| `clickOutside.ts` | None | All branches: outside, inside, nested, update, destroy |\n| `buildPrompt.ts` | None | Preprompt replacement, truncation edge cases (0, undefined) |\n| `createShareLink.ts` | None | 7-char shortcut, API failure, prefix priority |\n| `lock.ts` | Partial | `refreshLock(false)`, mismatched lockId, key isolation |\n| `migrations.ts` | None | Duplicate GUID, lock contention, `runEveryTime`, HuggingChat guard, failure recording |\n| `rvagentPresets.ts` | None | All three exported functions |\n| `MarkdownRenderer` | Partial | XSS links, empty content, tables, images, nested formatting |\n",
6
+ "parsedOutput": {
7
+ "sections": [
8
+ {
9
+ "title": "Test Coverage Gap Analysis",
10
+ "content": "\n",
11
+ "level": 2
12
+ },
13
+ {
14
+ "title": "1. `src/lib/APIClient.ts` — Zero test coverage",
15
+ "content": "\n**Untested:** `apiCall` (all branches), `handleResponse`, query param serialization, error body parsing fallback chain, ObjectId superjson registration.\n\n```typescript\n// src/lib/APIClient.spec.ts\nimport { describe, it, expect, vi, afterEach } from \"vitest\";\nimport { handleResponse, useAPIClient } from \"./APIClient\";\n\nconst mockFetch = (status: number, body: string | null, ok = status < 400) =>\n vi.fn().mockResolvedValue({\n ok,\n status,\n text: () => Promise.resolve(body ?? \"\"),\n json: () => (body ? Promise.resolve(JSON.parse(body)) : Promise.reject(new Error(\"no body\"))),\n statusText: \"Error\",\n });\n\ndescribe(\"apiCall — success paths\", () => {\n it(\"sends GET with no body\", async () => {\n const fetch = mockFetch(200, '{\"json\":\"body\"}');\n const client = useAPIClient({ fetch, origin: \"http://localhost\" });\n await client[\"public-config\"].get();\n expect(fetch).toHaveBeenCalledWith(\n expect.stringContaining(\"/api/v2/public-config\"),\n expect.objectContaining({ method: \"GET\" })\n );\n });\n\n it(\"attaches query params to GET URL\", async () => {\n const fetch = mockFetch(200, '{\"x\":1}');\n const client = useAPIClient({ fetch, origin: \"http://localhost\" });\n await client[\"public-config\"].get({ query: { page: 2, skip: undefined } });\n const url = fetch.mock.calls[0][0] as string;\n expect(url).toContain(\"page=2\");\n expect(url).not.toContain(\"skip\"); // undefined values are omitted\n });\n\n it(\"returns null data on empty 200 body\", async () => {\n const fetch = mockFetch(200, \"\");\n const client = useAPIClient({ fetch, origin: \"http://localhost\" });\n const res = await client.user.settings.get();\n expect(res.data).toBeNull();\n expect(res.error).toBeNull();\n });\n\n it(\"serialises body as JSON for POST\", async () => {\n const fetch = mockFetch(200, \"null\");\n const client = useAPIClient({ fetch, origin: \"http://localhost\" });\n await client.user.settings.post({ theme: \"dark\" });\n const init = fetch.mock.calls[0][1] as RequestInit;\n expect(init.headers).toMatchObject({ \"Content-Type\": \"application/json\" });\n expect(JSON.parse(init.body as string)).toEqual({ theme: \"dark\" });\n });\n});\n\ndescribe(\"apiCall — error paths\", () => {\n it(\"returns error with JSON body on 4xx\", async () => {\n const fetch = mockFetch(400, '{\"message\":\"bad request\"}');\n const client = useAPIClient({ fetch, origin: \"http://localhost\" });\n const res = await client.user.get();\n expect(res.error).toEqual({ message: \"bad request\" });\n expect(res.data).toBeNull();\n expect(res.status).toBe(400);\n });\n\n it(\"falls back to text when error body is not JSON\", async () => {\n const fetch = vi.fn().mockResolvedValue({\n ok: false,\n status: 500,\n json: () => Promise.reject(new Error(\"not json\")),\n text: () => Promise.resolve(\"Internal Server Error\"),\n statusText: \"ISE\",\n });\n const client = useAPIClient({ fetch, origin: \"http://localhost\" });\n const res = await client.user.get();\n expect(res.error).toBe(\"Internal Server Error\");\n });\n\n it(\"falls back to statusText when text() also fails\", async () => {\n const fetch = vi.fn().mockResolvedValue({\n ok: false,\n status: 503,\n json: () => Promise.reject(new Error()),\n text: () => Promise.reject(new Error()),\n statusText: \"Service Unavailable\",\n });\n const client = useAPIClient({ fetch, origin: \"http://localhost\" });\n const res = await client.user.get();\n expect(res.error).toBe(\"Service Unavailable\");\n });\n});\n\ndescribe(\"handleResponse\", () => {\n it(\"returns null when data is null and no error\", () => {\n expect(handleResponse({ data: null, error: null, status: 204 })).toBeNull();\n });\n\n it(\"throws when response contains an error\", () => {\n expect(() =>\n handleResponse({ data: null, error: { message: \"forbidden\" }, status: 403 })\n ).toThrow();\n });\n\n it(\"parses superjson string data\", () => {\n const payload = JSON.stringify({ json: { value: 42 }, meta: undefined });\n expect(handleResponse({ data: payload, error: null, status: 200 })).toEqual({ value: 42 });\n });\n});\n\ndescribe(\"useAPIClient — URL construction\", () => {\n it(\"uses provided origin for SSR base URL\", async () => {\n const fetch = mockFetch(200, \"null\");\n const client = useAPIClient({ fetch, origin: \"http://api.internal:4000\" });\n await client.export.get();\n expect(fetch.mock.calls[0][0]).toContain(\"http://api.internal:4000\");\n });\n\n it(\"builds nested conversation message endpoint\", async () => {\n const fetch = mockFetch(200, \"null\");\n const client = useAPIClient({ fetch, origin: \"http://localhost\" });\n await client.conversations({ id: \"abc123\" }).message({ messageId: \"msg456\" }).get();\n expect(fetch.mock.calls[0][0]).toContain(\"/conversations/abc123/message/msg456\");\n });\n});\n```\n\n---\n\n",
16
+ "level": 3
17
+ },
18
+ {
19
+ "title": "2. `src/lib/actions/clickOutside.ts` — Zero test coverage",
20
+ "content": "\n**Untested:** click outside triggers callback, click inside does not, `update` replaces callback, `destroy` removes listener.\n\n```typescript\n// src/lib/actions/clickOutside.spec.ts\nimport { describe, it, expect, vi, beforeEach } from \"vitest\";\nimport { clickOutside } from \"./clickOutside\";\n\ndescribe(\"clickOutside action\", () => {\n let element: HTMLDivElement;\n let outsideElement: HTMLDivElement;\n let callback: ReturnType<typeof vi.fn>;\n\n beforeEach(() => {\n document.body.innerHTML = \"\";\n element = document.createElement(\"div\");\n outsideElement = document.createElement(\"div\");\n document.body.appendChild(element);\n document.body.appendChild(outsideElement);\n callback = vi.fn();\n });\n\n it(\"fires callback when clicking outside the element\", () => {\n clickOutside(element, callback);\n outsideElement.dispatchEvent(new MouseEvent(\"click\", { bubbles: true }));\n expect(callback).toHaveBeenCalledOnce();\n });\n\n it(\"does not fire callback when clicking inside the element\", () => {\n clickOutside(element, callback);\n element.dispatchEvent(new MouseEvent(\"click\", { bubbles: true }));\n expect(callback).not.toHaveBeenCalled();\n });\n\n it(\"does not fire when clicking the element itself\", () => {\n clickOutside(element, callback);\n const inner = document.createElement(\"span\");\n element.appendChild(inner);\n inner.dispatchEvent(new MouseEvent(\"click\", { bubbles: true }));\n expect(callback).not.toHaveBeenCalled();\n });\n\n it(\"update() replaces the callback\", () => {\n const action = clickOutside(element, callback);\n const newCallback = vi.fn();\n action.update(newCallback);\n outsideElement.dispatchEvent(new MouseEvent(\"click\", { bubbles: true }));\n expect(callback).not.toHaveBeenCalled();\n expect(newCallback).toHaveBeenCalledOnce();\n });\n\n it(\"destroy() stops all future callbacks\", () => {\n const action = clickOutside(element, callback);\n action.destroy();\n outsideElement.dispatchEvent(new MouseEvent(\"click\", { bubbles: true }));\n expect(callback).not.toHaveBeenCalled();\n });\n\n it(\"handles multiple destroy() calls without error\", () => {\n const action = clickOutside(element, callback);\n expect(() => { action.destroy(); action.destroy(); }).not.toThrow();\n });\n});\n```\n\n---\n\n",
21
+ "level": 3
22
+ },
23
+ {
24
+ "title": "3. `src/lib/buildPrompt.ts` — Zero test coverage",
25
+ "content": "\n**Untested:** system message replacement with preprompt, no system message scenario, truncation slicing, zero truncate parameter (no slicing).\n\n```typescript\n// src/lib/buildPrompt.spec.ts\nimport { describe, it, expect } from \"vitest\";\nimport { buildPrompt } from \"./buildPrompt\";\nimport type { BackendModel } from \"./server/models\";\n\nconst makeModel = (truncate?: number): BackendModel =>\n ({\n chatPromptRender: ({ messages, preprompt }: { messages: unknown[]; preprompt?: string }) =>\n messages.map((m: any) => `${m.role}:${m.content}`).join(\" \"),\n parameters: truncate !== undefined ? { truncate } : undefined,\n } as unknown as BackendModel);\n\ndescribe(\"buildPrompt\", () => {\n it(\"replaces system message content with preprompt when first message is system\", async () => {\n const model = makeModel(1000);\n const messages = [\n { from: \"system\" as const, content: \"old system prompt\" },\n { from: \"user\" as const, content: \"hello\" },\n ];\n const result = await buildPrompt({ messages, model, preprompt: \"new system prompt\" });\n expect(result).toContain(\"new system prompt\");\n expect(result).not.toContain(\"old system prompt\");\n });\n\n it(\"does not replace content when first message is not system\", async () => {\n const model = makeModel(1000);\n const messages = [{ from: \"user\" as const, content: \"hello\" }];\n await buildPrompt({ messages, model, preprompt: \"some preprompt\" });\n // original content unchanged\n expect(messages[0].content).toBe(\"hello\");\n });\n\n it(\"does not replace content when preprompt is undefined\", async () => {\n const model = makeModel(1000);\n const messages = [{ from: \"system\" as const, content: \"original\" }];\n await buildPrompt({ messages, model, preprompt: undefined });\n expect(messages[0].content).toBe(\"original\");\n });\n\n it(\"truncates to the last N words when truncate is set\", async () => {\n const model = makeModel(3);\n const messages = [{ from: \"user\" as const, content: \"one two three four five\" }];\n const result = await buildPrompt({ messages, model });\n const wordCount = result.trim().split(/\\s+/).length;\n expect(wordCount).toBeLessThanOrEqual(3);\n });\n\n it(\"returns full prompt when truncate is 0 (no slicing)\", async () => {\n const model = makeModel(0);\n const messages = [{ from: \"user\" as const, content: \"word1 word2 word3\" }];\n const result = await buildPrompt({ messages, model });\n // slice(0) returns empty array → join → empty string\n expect(result).toBe(\"\");\n });\n\n it(\"returns full prompt when model has no truncate parameter\", async () => {\n const model = makeModel(undefined);\n const messages = [{ from: \"user\" as const, content: \"word1 word2 word3\" }];\n const result = await buildPrompt({ messages, model });\n expect(result).toContain(\"word1\");\n });\n\n it(\"maps message from to role field\", async () => {\n const receivedMessages: unknown[] = [];\n const model = {\n chatPromptRender: ({ messages }: { messages: unknown[] }) => {\n receivedMessages.push(...messages);\n return \"\";\n },\n parameters: { truncate: 1000 },\n } as unknown as BackendModel;\n await buildPrompt({\n messages: [{ from: \"assistant\" as const, content: \"hi\" }],\n model,\n });\n expect((receivedMessages[0] as any).role).toBe(\"assistant\");\n });\n});\n```\n\n---\n\n",
26
+ "level": 3
27
+ },
28
+ {
29
+ "title": "4. `src/lib/createShareLink.ts` — Zero test coverage",
30
+ "content": "\n**Untested:** 7-char ID shortcut path, successful API call, API failure with text body, API failure with empty body, URL prefix priority logic.\n\n```typescript\n// src/lib/createShareLink.spec.ts\nimport { describe, it, expect, vi, beforeEach } from \"vitest\";\n\n// Mock SvelteKit modules before importing\nvi.mock(\"$app/paths\", () => ({ base: \"\" }));\nvi.mock(\"$app/state\", () => ({\n page: {\n data: { publicConfig: { PUBLIC_SHARE_PREFIX: \"\", PUBLIC_ORIGIN: \"https://app.example.com\" } },\n url: { origin: \"https://fallback.example.com\" },\n },\n}));\n\nimport { createShareLink } from \"./createShareLink\";\n\ndescribe(\"createShareLink\", () => {\n beforeEach(() => {\n vi.resetAllMocks();\n });\n\n it(\"returns share URL directly for 7-char id without fetching\", async () => {\n const fetchSpy = vi.spyOn(globalThis, \"fetch\");\n const url = await createShareLink(\"abc1234\");\n expect(fetchSpy).not.toHaveBeenCalled();\n expect(url).toBe(\"https://app.example.com/r/abc1234\");\n });\n\n it(\"calls the share endpoint for non-7-char ids\", async () => {\n vi.spyOn(globalThis, \"fetch\").mockResolvedValueOnce({\n ok: true,\n json: () => Promise.resolve({ shareId: \"xyz9876\" }),\n } as Response);\n const url = await createShareLink(\"longconversationid\");\n expect(url).toBe(\"https://app.example.com/r/xyz9876\");\n });\n\n it(\"throws when the share endpoint returns non-ok with a text body\", async () => {\n vi.spyOn(globalThis, \"fetch\").mockResolvedValueOnce({\n ok: false,\n text: () => Promise.resolve(\"Conversation not found\"),\n } as unknown as Response);\n await expect(createShareLink(\"longconversationid\")).rejects.toThrow(\"Conversation not found\");\n });\n\n it(\"throws generic message when error body is empty\", async () => {\n vi.spyOn(globalThis, \"fetch\").mockResolvedValueOnce({\n ok: false,\n text: () => Promise.resolve(\"\"),\n } as unknown as Response);\n await expect(createShareLink(\"longconversationid\")).rejects.toThrow(\n \"Failed to create share link\"\n );\n });\n\n it(\"uses PUBLIC_SHARE_PREFIX when set\", async () => {\n const { page } = await import(\"$app/state\");\n (page.data.publicConfig as any).PUBLIC_SHARE_PREFIX = \"https://cdn.share.example.com\";\n const url = await createShareLink(\"abc1234\");\n expect(url).toStartWith(\"https://cdn.share.example.com\");\n (page.data.publicConfig as any).PUBLIC_SHARE_PREFIX = \"\";\n });\n});\n```\n\n---\n\n",
31
+ "level": 3
32
+ },
33
+ {
34
+ "title": "5. `src/lib/migrations/lock.ts` — Gaps in existing tests",
35
+ "content": "\nThe existing suite covers the happy path and concurrent acquisition. Missing:\n\n| Gap | Why it matters |\n|---|---|\n| `releaseLock` with mismatched `lockId` | Should not delete the wrong lock |\n| `refreshLock` on a non-existent lock | Returns `false` — not asserted anywhere |\n| `isDBLocked` with a different key | Isolation between semaphore keys |\n| `releaseLock` when lock already released | Should be idempotent |\n\n```typescript\n// Add to src/lib/migrations/migrations.spec.ts (inside the describe block)\n\nit(\"releaseLock does not delete lock held by different id\", async () => {\n const lockId = await acquireLock(Semaphores.TEST_MIGRATION);\n assert(lockId);\n const fakeId = new ObjectId();\n await releaseLock(Semaphores.TEST_MIGRATION, fakeId); // wrong id\n expect(await isDBLocked(Semaphores.TEST_MIGRATION)).toBe(true);\n await releaseLock(Semaphores.TEST_MIGRATION, lockId); // cleanup\n});\n\nit(\"refreshLock returns false when lock does not exist\", async () => {\n const fakeId = new ObjectId();\n const result = await refreshLock(Semaphores.TEST_MIGRATION, fakeId);\n expect(result).toBe(false);\n});\n\nit(\"isDBLocked is key-scoped — one key locked does not affect another\", async () => {\n const lockId = await acquireLock(Semaphores.TEST_MIGRATION);\n assert(lockId);\n expect(await isDBLocked(\"other-key\")).toBe(false);\n await releaseLock(Semaphores.TEST_MIGRATION, lockId);\n});\n\nit(\"releaseLock is idempotent — releasing twice does not throw\", async () => {\n const lockId = await acquireLock(Semaphores.TEST_MIGRATION);\n assert(lockId);\n await releaseLock(Semaphores.TEST_MIGRATION, lockId);\n await expect(releaseLock(Semaphores.TEST_MIGRATION, lockId)).resolves.not.toThrow();\n});\n```\n\n---\n\n",
36
+ "level": 3
37
+ },
38
+ {
39
+ "title": "6. `src/lib/migrations/migrations.ts` — Zero test coverage for `checkAndRunMigrations`",
40
+ "content": "\n**Untested:** duplicate GUID detection, lock contention wait-loop, `runEveryTime`, `runForHuggingChat` guards, migration failure recording, lock refresh interval, lock release on completion.\n\n```typescript\n// src/lib/migrations/checkAndRunMigrations.spec.ts\nimport { beforeEach, describe, expect, it, vi } from \"vitest\";\nimport type { Database } from \"$lib/server/database\";\nimport { Semaphores } from \"$lib/types/Semaphore\";\nimport ObjectId from \"bson-objectid\";\n\n// Stubs for all side-effecting modules\nvi.mock(\"./lock\");\nvi.mock(\"$lib/server/logger\", () => ({ logger: { debug: vi.fn(), error: vi.fn() } }));\nvi.mock(\"$lib/server/config\", () => ({ config: { isHuggingChat: false } }));\n\nimport * as lockModule from \"./lock\";\nimport { checkAndRunMigrations } from \"./migrations\";\n\nconst makeDb = (results: unknown[] = []) => ({\n getCollections: () => ({\n migrationResults: {\n find: () => ({ toArray: () => Promise.resolve(results) }),\n updateOne: vi.fn().mockResolvedValue({}),\n },\n }),\n});\n\nvi.mock(\"$lib/server/database\", () => ({\n Database: { getInstance: vi.fn() },\n}));\n\nimport { Database } from \"$lib/server/database\";\n\ndescribe(\"checkAndRunMigrations\", () => {\n beforeEach(() => {\n vi.clearAllMocks();\n vi.mocked(Database.getInstance).mockResolvedValue(makeDb() as unknown as Database);\n vi.mocked(lockModule.acquireLock).mockResolvedValue(new ObjectId());\n vi.mocked(lockModule.releaseLock).mockResolvedValue(undefined);\n vi.mocked(lockModule.isDBLocked).mockResolvedValue(false);\n vi.mocked(lockModule.refreshLock).mockResolvedValue(true);\n });\n\n it(\"throws when duplicate migration GUIDs exist\", async () => {\n // Override routines to inject duplicates — easiest with vi.doMock inside test\n await expect(async () => {\n vi.doMock(\"./routines\", () => {\n const id = new ObjectId();\n return { migrations: [{ _id: id, name: \"dup\", up: vi.fn() }, { _id: id, name: \"dup2\", up: vi.fn() }] };\n });\n const { checkAndRunMigrations: fn } = await import(\"./migrations?dup=1\");\n await fn();\n }).rejects.toThrow(\"Duplicate migration GUIDs found\");\n });\n\n it(\"waits for DB unlock when lock is already held\", async () => {\n vi.mocked(lockModule.acquireLock).mockResolvedValueOnce(false);\n // isDBLocked returns true once, then false to exit the while loop\n vi.mocked(lockModule.isDBLocked)\n .mockResolvedValueOnce(true)\n .mockResolvedValueOnce(false);\n await checkAndRunMigrations();\n expect(lockModule.isDBLocked).toHaveBeenCalledWith(Semaphores.MIGRATION);\n });\n\n it(\"skips migration already in results\", async () => {\n const id = new ObjectId();\n vi.mocked(Database.getInstance).mockResolvedValue(\n makeDb([{ _id: id, name: \"done\", status: \"success\" }]) as unknown as Database\n );\n const upFn = vi.fn().mockResolvedValue(true);\n vi.doMock(\"./routines\", () => ({\n migrations: [{ _id: id, name: \"done\", up: upFn, runEveryTime: false }],\n }));\n const { checkAndRunMigrations: fn } = await import(\"./migrations?skip=1\");\n await fn();\n expect(upFn).not.toHaveBeenCalled();\n });\n\n it(\"records failure status when migration throws\", async () => {\n const id = new ObjectId();\n const updateOne = vi.fn().mockResolvedValue({});\n vi.mocked(Database.getInstance).mockResolvedValue({\n getCollections: () => ({\n migrationResults: { find: () => ({ toArray: () => Promise.resolve([]) }), updateOne },\n }),\n } as unknown as Database);\n vi.doMock(\"./routines\", () => ({\n migrations: [{ _id: id, name: \"failing\", up: vi.fn().mockRejectedValue(new Error(\"boom\")) }],\n }));\n const { checkAndRunMigrations: fn } = await import(\"./migrations?fail=1\");\n await fn();\n const lastCall = updateOne.mock.calls.at(-1)?.[1].$set;\n expect(lastCall?.status).toBe(\"failure\");\n });\n\n it(\"skips runForHuggingChat=only migration when not HuggingChat\", async () => {\n const { config } = await import(\"$lib/server/config\");\n (config as any).isHuggingChat = false;\n const upFn = vi.fn().mockResolvedValue(true);\n const id = new ObjectId();\n vi.doMock(\"./routines\", () => ({\n migrations: [{ _id: id, name: \"hc-only\", up: upFn, runForHuggingChat: \"only\" }],\n }));\n const { checkAndRunMigrations: fn } = await import(\"./migrations?hconly=1\");\n await fn();\n expect(upFn).not.toHaveBeenCalled();\n });\n\n it(\"releases lock and clears interval after all migrations run\", async () => {\n vi.useFakeTimers();\n await checkAndRunMigrations();\n vi.useRealTimers();\n expect(lockModule.releaseLock).toHaveBeenCalledWith(\n Semaphores.MIGRATION,\n expect.any(ObjectId)\n );\n });\n});\n```\n\n---\n\n",
41
+ "level": 3
42
+ },
43
+ {
44
+ "title": "7. `src/lib/constants/rvagentPresets.ts` — Utility functions untested",
45
+ "content": "\n**Untested:** `getPresetById`, `buildPresetUrl`, `buildPresetCliCommand`.\n\n```typescript\n// src/lib/constants/rvagentPresets.spec.ts\nimport { describe, it, expect } from \"vitest\";\nimport {\n getPresetById,\n buildPresetUrl,\n buildPresetCliCommand,\n RVAGENT_PRESETS,\n} from \"./rvagentPresets\";\n\ndescribe(\"getPresetById\", () => {\n it(\"returns the matching preset\", () => {\n const preset = getPresetById(\"all-tools\");\n expect(preset).toBeDefined();\n expect(preset?.id).toBe(\"all-tools\");\n });\n\n it(\"returns undefined for an unknown id\", () => {\n expect(getPresetById(\"nonexistent\")).toBeUndefined();\n });\n\n it(\"all preset IDs in RVAGENT_PRESETS are findable\", () => {\n for (const p of RVAGENT_PRESETS) {\n expect(getPresetById(p.id)).toBe(p);\n }\n });\n});\n\ndescribe(\"buildPresetUrl\", () => {\n it(\"uses preset defaultPort when no port override is given\", () => {\n const preset = getPresetById(\"all-tools\")!;\n expect(buildPresetUrl(preset)).toBe(`http://localhost:${preset.defaultPort}/sse`);\n });\n\n it(\"uses the overridden port\", () => {\n const preset = getPresetById(\"all-tools\")!;\n expect(buildPresetUrl(preset, \"localhost\", 9999)).toBe(\"http://localhost:9999/sse\");\n });\n\n it(\"uses custom host\", () => {\n const preset = getPresetById(\"all-tools\")!;\n expect(buildPresetUrl(preset, \"192.168.1.10\")).toContain(\"192.168.1.10\");\n });\n});\n\ndescribe(\"buildPresetCliCommand\", () => {\n it(\"uses --all flag for the all-tools preset\", () => {\n const preset = getPresetById(\"all-tools\")!;\n const cmd = buildPresetCliCommand(preset);\n expect(cmd).toContain(\"--all\");\n expect(cmd).not.toContain(\"--groups\");\n });\n\n it(\"uses --groups for presets with specific tool groups\", () => {\n const preset = getPresetById(\"file-shell\")!;\n const cmd = buildPresetCliCommand(preset);\n expect(cmd).toContain(\"--groups file,shell\");\n expect(cmd).not.toContain(\"--all\");\n });\n\n it(\"uses custom port in the command\", () => {\n const preset = getPresetById(\"memory-agent\")!;\n const cmd = buildPresetCliCommand(preset, 8888);\n expect(cmd).toContain(\"--port 8888\");\n });\n\n it(\"falls back to defaultPort in the command when no port given\", () => {\n const preset = getPresetById(\"memory-agent\")!;\n const cmd = buildPresetCliCommand(preset);\n expect(cmd).toContain(`--port ${preset.defaultPort}`);\n });\n});\n```\n\n---\n\n",
46
+ "level": 3
47
+ },
48
+ {
49
+ "title": "8. `src/lib/components/chat/MarkdownRenderer.svelte.test.ts` — Edge case gaps",
50
+ "content": "\n**Missing:** XSS via `javascript:` links, empty content, bold/italic inside lists, tables, image rendering, multiple code blocks with language labels.\n\n```typescript\n// Add to existing MarkdownRenderer.svelte.test.ts\n\ndescribe(\"MarkdownRenderer — security\", () => {\n it(\"strips javascript: href from links\", () => {\n const { baseElement } = render(MarkdownRenderer, {\n content: \"[click me](javascript:alert(1))\",\n });\n const anchor = baseElement.querySelector(\"a\");\n expect(anchor?.getAttribute(\"href\")).not.toContain(\"javascript:\");\n });\n\n it(\"renders empty content without crashing\", () => {\n expect(() => render(MarkdownRenderer, { content: \"\" })).not.toThrow();\n });\n});\n\ndescribe(\"MarkdownRenderer — tables\", () => {\n it(\"renders a markdown table\", () => {\n const md = \"| A | B |\\n|---|---|\\n| 1 | 2 |\";\n const { baseElement } = render(MarkdownRenderer, { content: md });\n expect(baseElement.querySelector(\"table\")).toBeTruthy();\n expect(baseElement.querySelector(\"thead\")).toBeTruthy();\n expect(baseElement.querySelector(\"tbody\")).toBeTruthy();\n });\n});\n\ndescribe(\"MarkdownRenderer — nested formatting\", () => {\n it(\"renders bold inside a list item\", () => {\n const { baseElement } = render(MarkdownRenderer, { content: \"- **bold item**\" });\n const strong = baseElement.querySelector(\"strong\");\n expect(strong).toBeTruthy();\n expect(strong?.textContent).toBe(\"bold item\");\n });\n\n it(\"renders multiple fenced code blocks with language labels\", () => {\n const md = \"```ts\\nconst x = 1;\\n```\\n\\n```py\\nprint('hi')\\n```\";\n const { baseElement } = render(MarkdownRenderer, { content: md });\n const codeBlocks = baseElement.querySelectorAll(\"code\");\n expect(codeBlocks.length).toBeGreaterThanOrEqual(2);\n });\n});\n\ndescribe(\"MarkdownRenderer — images\", () => {\n it(\"renders an image with alt text\", () => {\n const { baseElement } = render(MarkdownRenderer, {\n content: \"![Alt text](https://example.com/img.png)\",\n });\n const img = baseElement.querySelector(\"img\");\n expect(img).toBeTruthy();\n expect(img?.getAttribute(\"alt\")).toBe(\"Alt text\");\n });\n});\n```\n\n---\n\n",
51
+ "level": 3
52
+ },
53
+ {
54
+ "title": "Summary of Gaps",
55
+ "content": "| File | Coverage | Priority gaps |\n|---|---|---|\n| `APIClient.ts` | None | Error fallback chain, empty body, query filtering, `handleResponse` |\n| `clickOutside.ts` | None | All branches: outside, inside, nested, update, destroy |\n| `buildPrompt.ts` | None | Preprompt replacement, truncation edge cases (0, undefined) |\n| `createShareLink.ts` | None | 7-char shortcut, API failure, prefix priority |\n| `lock.ts` | Partial | `refreshLock(false)`, mismatched lockId, key isolation |\n| `migrations.ts` | None | Duplicate GUID, lock contention, `runEveryTime`, HuggingChat guard, failure recording |\n| `rvagentPresets.ts` | None | All three exported functions |\n| `MarkdownRenderer` | Partial | XSS links, empty content, tables, images, nested formatting |",
56
+ "level": 3
57
+ }
58
+ ],
59
+ "codeBlocks": [
60
+ {
61
+ "language": "typescript",
62
+ "code": "// src/lib/APIClient.spec.ts\nimport { describe, it, expect, vi, afterEach } from \"vitest\";\nimport { handleResponse, useAPIClient } from \"./APIClient\";\n\nconst mockFetch = (status: number, body: string | null, ok = status < 400) =>\n vi.fn().mockResolvedValue({\n ok,\n status,\n text: () => Promise.resolve(body ?? \"\"),\n json: () => (body ? Promise.resolve(JSON.parse(body)) : Promise.reject(new Error(\"no body\"))),\n statusText: \"Error\",\n });\n\ndescribe(\"apiCall — success paths\", () => {\n it(\"sends GET with no body\", async () => {\n const fetch = mockFetch(200, '{\"json\":\"body\"}');\n const client = useAPIClient({ fetch, origin: \"http://localhost\" });\n await client[\"public-config\"].get();\n expect(fetch).toHaveBeenCalledWith(\n expect.stringContaining(\"/api/v2/public-config\"),\n expect.objectContaining({ method: \"GET\" })\n );\n });\n\n it(\"attaches query params to GET URL\", async () => {\n const fetch = mockFetch(200, '{\"x\":1}');\n const client = useAPIClient({ fetch, origin: \"http://localhost\" });\n await client[\"public-config\"].get({ query: { page: 2, skip: undefined } });\n const url = fetch.mock.calls[0][0] as string;\n expect(url).toContain(\"page=2\");\n expect(url).not.toContain(\"skip\"); // undefined values are omitted\n });\n\n it(\"returns null data on empty 200 body\", async () => {\n const fetch = mockFetch(200, \"\");\n const client = useAPIClient({ fetch, origin: \"http://localhost\" });\n const res = await client.user.settings.get();\n expect(res.data).toBeNull();\n expect(res.error).toBeNull();\n });\n\n it(\"serialises body as JSON for POST\", async () => {\n const fetch = mockFetch(200, \"null\");\n const client = useAPIClient({ fetch, origin: \"http://localhost\" });\n await client.user.settings.post({ theme: \"dark\" });\n const init = fetch.mock.calls[0][1] as RequestInit;\n expect(init.headers).toMatchObject({ \"Content-Type\": \"application/json\" });\n expect(JSON.parse(init.body as string)).toEqual({ theme: \"dark\" });\n });\n});\n\ndescribe(\"apiCall — error paths\", () => {\n it(\"returns error with JSON body on 4xx\", async () => {\n const fetch = mockFetch(400, '{\"message\":\"bad request\"}');\n const client = useAPIClient({ fetch, origin: \"http://localhost\" });\n const res = await client.user.get();\n expect(res.error).toEqual({ message: \"bad request\" });\n expect(res.data).toBeNull();\n expect(res.status).toBe(400);\n });\n\n it(\"falls back to text when error body is not JSON\", async () => {\n const fetch = vi.fn().mockResolvedValue({\n ok: false,\n status: 500,\n json: () => Promise.reject(new Error(\"not json\")),\n text: () => Promise.resolve(\"Internal Server Error\"),\n statusText: \"ISE\",\n });\n const client = useAPIClient({ fetch, origin: \"http://localhost\" });\n const res = await client.user.get();\n expect(res.error).toBe(\"Internal Server Error\");\n });\n\n it(\"falls back to statusText when text() also fails\", async () => {\n const fetch = vi.fn().mockResolvedValue({\n ok: false,\n status: 503,\n json: () => Promise.reject(new Error()),\n text: () => Promise.reject(new Error()),\n statusText: \"Service Unavailable\",\n });\n const client = useAPIClient({ fetch, origin: \"http://localhost\" });\n const res = await client.user.get();\n expect(res.error).toBe(\"Service Unavailable\");\n });\n});\n\ndescribe(\"handleResponse\", () => {\n it(\"returns null when data is null and no error\", () => {\n expect(handleResponse({ data: null, error: null, status: 204 })).toBeNull();\n });\n\n it(\"throws when response contains an error\", () => {\n expect(() =>\n handleResponse({ data: null, error: { message: \"forbidden\" }, status: 403 })\n ).toThrow();\n });\n\n it(\"parses superjson string data\", () => {\n const payload = JSON.stringify({ json: { value: 42 }, meta: undefined });\n expect(handleResponse({ data: payload, error: null, status: 200 })).toEqual({ value: 42 });\n });\n});\n\ndescribe(\"useAPIClient — URL construction\", () => {\n it(\"uses provided origin for SSR base URL\", async () => {\n const fetch = mockFetch(200, \"null\");\n const client = useAPIClient({ fetch, origin: \"http://api.internal:4000\" });\n await client.export.get();\n expect(fetch.mock.calls[0][0]).toContain(\"http://api.internal:4000\");\n });\n\n it(\"builds nested conversation message endpoint\", async () => {\n const fetch = mockFetch(200, \"null\");\n const client = useAPIClient({ fetch, origin: \"http://localhost\" });\n await client.conversations({ id: \"abc123\" }).message({ messageId: \"msg456\" }).get();\n expect(fetch.mock.calls[0][0]).toContain(\"/conversations/abc123/message/msg456\");\n });\n});"
63
+ },
64
+ {
65
+ "language": "typescript",
66
+ "code": "// src/lib/actions/clickOutside.spec.ts\nimport { describe, it, expect, vi, beforeEach } from \"vitest\";\nimport { clickOutside } from \"./clickOutside\";\n\ndescribe(\"clickOutside action\", () => {\n let element: HTMLDivElement;\n let outsideElement: HTMLDivElement;\n let callback: ReturnType<typeof vi.fn>;\n\n beforeEach(() => {\n document.body.innerHTML = \"\";\n element = document.createElement(\"div\");\n outsideElement = document.createElement(\"div\");\n document.body.appendChild(element);\n document.body.appendChild(outsideElement);\n callback = vi.fn();\n });\n\n it(\"fires callback when clicking outside the element\", () => {\n clickOutside(element, callback);\n outsideElement.dispatchEvent(new MouseEvent(\"click\", { bubbles: true }));\n expect(callback).toHaveBeenCalledOnce();\n });\n\n it(\"does not fire callback when clicking inside the element\", () => {\n clickOutside(element, callback);\n element.dispatchEvent(new MouseEvent(\"click\", { bubbles: true }));\n expect(callback).not.toHaveBeenCalled();\n });\n\n it(\"does not fire when clicking the element itself\", () => {\n clickOutside(element, callback);\n const inner = document.createElement(\"span\");\n element.appendChild(inner);\n inner.dispatchEvent(new MouseEvent(\"click\", { bubbles: true }));\n expect(callback).not.toHaveBeenCalled();\n });\n\n it(\"update() replaces the callback\", () => {\n const action = clickOutside(element, callback);\n const newCallback = vi.fn();\n action.update(newCallback);\n outsideElement.dispatchEvent(new MouseEvent(\"click\", { bubbles: true }));\n expect(callback).not.toHaveBeenCalled();\n expect(newCallback).toHaveBeenCalledOnce();\n });\n\n it(\"destroy() stops all future callbacks\", () => {\n const action = clickOutside(element, callback);\n action.destroy();\n outsideElement.dispatchEvent(new MouseEvent(\"click\", { bubbles: true }));\n expect(callback).not.toHaveBeenCalled();\n });\n\n it(\"handles multiple destroy() calls without error\", () => {\n const action = clickOutside(element, callback);\n expect(() => { action.destroy(); action.destroy(); }).not.toThrow();\n });\n});"
67
+ },
68
+ {
69
+ "language": "typescript",
70
+ "code": "// src/lib/buildPrompt.spec.ts\nimport { describe, it, expect } from \"vitest\";\nimport { buildPrompt } from \"./buildPrompt\";\nimport type { BackendModel } from \"./server/models\";\n\nconst makeModel = (truncate?: number): BackendModel =>\n ({\n chatPromptRender: ({ messages, preprompt }: { messages: unknown[]; preprompt?: string }) =>\n messages.map((m: any) => `${m.role}:${m.content}`).join(\" \"),\n parameters: truncate !== undefined ? { truncate } : undefined,\n } as unknown as BackendModel);\n\ndescribe(\"buildPrompt\", () => {\n it(\"replaces system message content with preprompt when first message is system\", async () => {\n const model = makeModel(1000);\n const messages = [\n { from: \"system\" as const, content: \"old system prompt\" },\n { from: \"user\" as const, content: \"hello\" },\n ];\n const result = await buildPrompt({ messages, model, preprompt: \"new system prompt\" });\n expect(result).toContain(\"new system prompt\");\n expect(result).not.toContain(\"old system prompt\");\n });\n\n it(\"does not replace content when first message is not system\", async () => {\n const model = makeModel(1000);\n const messages = [{ from: \"user\" as const, content: \"hello\" }];\n await buildPrompt({ messages, model, preprompt: \"some preprompt\" });\n // original content unchanged\n expect(messages[0].content).toBe(\"hello\");\n });\n\n it(\"does not replace content when preprompt is undefined\", async () => {\n const model = makeModel(1000);\n const messages = [{ from: \"system\" as const, content: \"original\" }];\n await buildPrompt({ messages, model, preprompt: undefined });\n expect(messages[0].content).toBe(\"original\");\n });\n\n it(\"truncates to the last N words when truncate is set\", async () => {\n const model = makeModel(3);\n const messages = [{ from: \"user\" as const, content: \"one two three four five\" }];\n const result = await buildPrompt({ messages, model });\n const wordCount = result.trim().split(/\\s+/).length;\n expect(wordCount).toBeLessThanOrEqual(3);\n });\n\n it(\"returns full prompt when truncate is 0 (no slicing)\", async () => {\n const model = makeModel(0);\n const messages = [{ from: \"user\" as const, content: \"word1 word2 word3\" }];\n const result = await buildPrompt({ messages, model });\n // slice(0) returns empty array → join → empty string\n expect(result).toBe(\"\");\n });\n\n it(\"returns full prompt when model has no truncate parameter\", async () => {\n const model = makeModel(undefined);\n const messages = [{ from: \"user\" as const, content: \"word1 word2 word3\" }];\n const result = await buildPrompt({ messages, model });\n expect(result).toContain(\"word1\");\n });\n\n it(\"maps message from to role field\", async () => {\n const receivedMessages: unknown[] = [];\n const model = {\n chatPromptRender: ({ messages }: { messages: unknown[] }) => {\n receivedMessages.push(...messages);\n return \"\";\n },\n parameters: { truncate: 1000 },\n } as unknown as BackendModel;\n await buildPrompt({\n messages: [{ from: \"assistant\" as const, content: \"hi\" }],\n model,\n });\n expect((receivedMessages[0] as any).role).toBe(\"assistant\");\n });\n});"
71
+ },
72
+ {
73
+ "language": "typescript",
74
+ "code": "// src/lib/createShareLink.spec.ts\nimport { describe, it, expect, vi, beforeEach } from \"vitest\";\n\n// Mock SvelteKit modules before importing\nvi.mock(\"$app/paths\", () => ({ base: \"\" }));\nvi.mock(\"$app/state\", () => ({\n page: {\n data: { publicConfig: { PUBLIC_SHARE_PREFIX: \"\", PUBLIC_ORIGIN: \"https://app.example.com\" } },\n url: { origin: \"https://fallback.example.com\" },\n },\n}));\n\nimport { createShareLink } from \"./createShareLink\";\n\ndescribe(\"createShareLink\", () => {\n beforeEach(() => {\n vi.resetAllMocks();\n });\n\n it(\"returns share URL directly for 7-char id without fetching\", async () => {\n const fetchSpy = vi.spyOn(globalThis, \"fetch\");\n const url = await createShareLink(\"abc1234\");\n expect(fetchSpy).not.toHaveBeenCalled();\n expect(url).toBe(\"https://app.example.com/r/abc1234\");\n });\n\n it(\"calls the share endpoint for non-7-char ids\", async () => {\n vi.spyOn(globalThis, \"fetch\").mockResolvedValueOnce({\n ok: true,\n json: () => Promise.resolve({ shareId: \"xyz9876\" }),\n } as Response);\n const url = await createShareLink(\"longconversationid\");\n expect(url).toBe(\"https://app.example.com/r/xyz9876\");\n });\n\n it(\"throws when the share endpoint returns non-ok with a text body\", async () => {\n vi.spyOn(globalThis, \"fetch\").mockResolvedValueOnce({\n ok: false,\n text: () => Promise.resolve(\"Conversation not found\"),\n } as unknown as Response);\n await expect(createShareLink(\"longconversationid\")).rejects.toThrow(\"Conversation not found\");\n });\n\n it(\"throws generic message when error body is empty\", async () => {\n vi.spyOn(globalThis, \"fetch\").mockResolvedValueOnce({\n ok: false,\n text: () => Promise.resolve(\"\"),\n } as unknown as Response);\n await expect(createShareLink(\"longconversationid\")).rejects.toThrow(\n \"Failed to create share link\"\n );\n });\n\n it(\"uses PUBLIC_SHARE_PREFIX when set\", async () => {\n const { page } = await import(\"$app/state\");\n (page.data.publicConfig as any).PUBLIC_SHARE_PREFIX = \"https://cdn.share.example.com\";\n const url = await createShareLink(\"abc1234\");\n expect(url).toStartWith(\"https://cdn.share.example.com\");\n (page.data.publicConfig as any).PUBLIC_SHARE_PREFIX = \"\";\n });\n});"
75
+ },
76
+ {
77
+ "language": "typescript",
78
+ "code": "// Add to src/lib/migrations/migrations.spec.ts (inside the describe block)\n\nit(\"releaseLock does not delete lock held by different id\", async () => {\n const lockId = await acquireLock(Semaphores.TEST_MIGRATION);\n assert(lockId);\n const fakeId = new ObjectId();\n await releaseLock(Semaphores.TEST_MIGRATION, fakeId); // wrong id\n expect(await isDBLocked(Semaphores.TEST_MIGRATION)).toBe(true);\n await releaseLock(Semaphores.TEST_MIGRATION, lockId); // cleanup\n});\n\nit(\"refreshLock returns false when lock does not exist\", async () => {\n const fakeId = new ObjectId();\n const result = await refreshLock(Semaphores.TEST_MIGRATION, fakeId);\n expect(result).toBe(false);\n});\n\nit(\"isDBLocked is key-scoped — one key locked does not affect another\", async () => {\n const lockId = await acquireLock(Semaphores.TEST_MIGRATION);\n assert(lockId);\n expect(await isDBLocked(\"other-key\")).toBe(false);\n await releaseLock(Semaphores.TEST_MIGRATION, lockId);\n});\n\nit(\"releaseLock is idempotent — releasing twice does not throw\", async () => {\n const lockId = await acquireLock(Semaphores.TEST_MIGRATION);\n assert(lockId);\n await releaseLock(Semaphores.TEST_MIGRATION, lockId);\n await expect(releaseLock(Semaphores.TEST_MIGRATION, lockId)).resolves.not.toThrow();\n});"
79
+ },
80
+ {
81
+ "language": "typescript",
82
+ "code": "// src/lib/migrations/checkAndRunMigrations.spec.ts\nimport { beforeEach, describe, expect, it, vi } from \"vitest\";\nimport type { Database } from \"$lib/server/database\";\nimport { Semaphores } from \"$lib/types/Semaphore\";\nimport ObjectId from \"bson-objectid\";\n\n// Stubs for all side-effecting modules\nvi.mock(\"./lock\");\nvi.mock(\"$lib/server/logger\", () => ({ logger: { debug: vi.fn(), error: vi.fn() } }));\nvi.mock(\"$lib/server/config\", () => ({ config: { isHuggingChat: false } }));\n\nimport * as lockModule from \"./lock\";\nimport { checkAndRunMigrations } from \"./migrations\";\n\nconst makeDb = (results: unknown[] = []) => ({\n getCollections: () => ({\n migrationResults: {\n find: () => ({ toArray: () => Promise.resolve(results) }),\n updateOne: vi.fn().mockResolvedValue({}),\n },\n }),\n});\n\nvi.mock(\"$lib/server/database\", () => ({\n Database: { getInstance: vi.fn() },\n}));\n\nimport { Database } from \"$lib/server/database\";\n\ndescribe(\"checkAndRunMigrations\", () => {\n beforeEach(() => {\n vi.clearAllMocks();\n vi.mocked(Database.getInstance).mockResolvedValue(makeDb() as unknown as Database);\n vi.mocked(lockModule.acquireLock).mockResolvedValue(new ObjectId());\n vi.mocked(lockModule.releaseLock).mockResolvedValue(undefined);\n vi.mocked(lockModule.isDBLocked).mockResolvedValue(false);\n vi.mocked(lockModule.refreshLock).mockResolvedValue(true);\n });\n\n it(\"throws when duplicate migration GUIDs exist\", async () => {\n // Override routines to inject duplicates — easiest with vi.doMock inside test\n await expect(async () => {\n vi.doMock(\"./routines\", () => {\n const id = new ObjectId();\n return { migrations: [{ _id: id, name: \"dup\", up: vi.fn() }, { _id: id, name: \"dup2\", up: vi.fn() }] };\n });\n const { checkAndRunMigrations: fn } = await import(\"./migrations?dup=1\");\n await fn();\n }).rejects.toThrow(\"Duplicate migration GUIDs found\");\n });\n\n it(\"waits for DB unlock when lock is already held\", async () => {\n vi.mocked(lockModule.acquireLock).mockResolvedValueOnce(false);\n // isDBLocked returns true once, then false to exit the while loop\n vi.mocked(lockModule.isDBLocked)\n .mockResolvedValueOnce(true)\n .mockResolvedValueOnce(false);\n await checkAndRunMigrations();\n expect(lockModule.isDBLocked).toHaveBeenCalledWith(Semaphores.MIGRATION);\n });\n\n it(\"skips migration already in results\", async () => {\n const id = new ObjectId();\n vi.mocked(Database.getInstance).mockResolvedValue(\n makeDb([{ _id: id, name: \"done\", status: \"success\" }]) as unknown as Database\n );\n const upFn = vi.fn().mockResolvedValue(true);\n vi.doMock(\"./routines\", () => ({\n migrations: [{ _id: id, name: \"done\", up: upFn, runEveryTime: false }],\n }));\n const { checkAndRunMigrations: fn } = await import(\"./migrations?skip=1\");\n await fn();\n expect(upFn).not.toHaveBeenCalled();\n });\n\n it(\"records failure status when migration throws\", async () => {\n const id = new ObjectId();\n const updateOne = vi.fn().mockResolvedValue({});\n vi.mocked(Database.getInstance).mockResolvedValue({\n getCollections: () => ({\n migrationResults: { find: () => ({ toArray: () => Promise.resolve([]) }), updateOne },\n }),\n } as unknown as Database);\n vi.doMock(\"./routines\", () => ({\n migrations: [{ _id: id, name: \"failing\", up: vi.fn().mockRejectedValue(new Error(\"boom\")) }],\n }));\n const { checkAndRunMigrations: fn } = await import(\"./migrations?fail=1\");\n await fn();\n const lastCall = updateOne.mock.calls.at(-1)?.[1].$set;\n expect(lastCall?.status).toBe(\"failure\");\n });\n\n it(\"skips runForHuggingChat=only migration when not HuggingChat\", async () => {\n const { config } = await import(\"$lib/server/config\");\n (config as any).isHuggingChat = false;\n const upFn = vi.fn().mockResolvedValue(true);\n const id = new ObjectId();\n vi.doMock(\"./routines\", () => ({\n migrations: [{ _id: id, name: \"hc-only\", up: upFn, runForHuggingChat: \"only\" }],\n }));\n const { checkAndRunMigrations: fn } = await import(\"./migrations?hconly=1\");\n await fn();\n expect(upFn).not.toHaveBeenCalled();\n });\n\n it(\"releases lock and clears interval after all migrations run\", async () => {\n vi.useFakeTimers();\n await checkAndRunMigrations();\n vi.useRealTimers();\n expect(lockModule.releaseLock).toHaveBeenCalledWith(\n Semaphores.MIGRATION,\n expect.any(ObjectId)\n );\n });\n});"
83
+ },
84
+ {
85
+ "language": "typescript",
86
+ "code": "// src/lib/constants/rvagentPresets.spec.ts\nimport { describe, it, expect } from \"vitest\";\nimport {\n getPresetById,\n buildPresetUrl,\n buildPresetCliCommand,\n RVAGENT_PRESETS,\n} from \"./rvagentPresets\";\n\ndescribe(\"getPresetById\", () => {\n it(\"returns the matching preset\", () => {\n const preset = getPresetById(\"all-tools\");\n expect(preset).toBeDefined();\n expect(preset?.id).toBe(\"all-tools\");\n });\n\n it(\"returns undefined for an unknown id\", () => {\n expect(getPresetById(\"nonexistent\")).toBeUndefined();\n });\n\n it(\"all preset IDs in RVAGENT_PRESETS are findable\", () => {\n for (const p of RVAGENT_PRESETS) {\n expect(getPresetById(p.id)).toBe(p);\n }\n });\n});\n\ndescribe(\"buildPresetUrl\", () => {\n it(\"uses preset defaultPort when no port override is given\", () => {\n const preset = getPresetById(\"all-tools\")!;\n expect(buildPresetUrl(preset)).toBe(`http://localhost:${preset.defaultPort}/sse`);\n });\n\n it(\"uses the overridden port\", () => {\n const preset = getPresetById(\"all-tools\")!;\n expect(buildPresetUrl(preset, \"localhost\", 9999)).toBe(\"http://localhost:9999/sse\");\n });\n\n it(\"uses custom host\", () => {\n const preset = getPresetById(\"all-tools\")!;\n expect(buildPresetUrl(preset, \"192.168.1.10\")).toContain(\"192.168.1.10\");\n });\n});\n\ndescribe(\"buildPresetCliCommand\", () => {\n it(\"uses --all flag for the all-tools preset\", () => {\n const preset = getPresetById(\"all-tools\")!;\n const cmd = buildPresetCliCommand(preset);\n expect(cmd).toContain(\"--all\");\n expect(cmd).not.toContain(\"--groups\");\n });\n\n it(\"uses --groups for presets with specific tool groups\", () => {\n const preset = getPresetById(\"file-shell\")!;\n const cmd = buildPresetCliCommand(preset);\n expect(cmd).toContain(\"--groups file,shell\");\n expect(cmd).not.toContain(\"--all\");\n });\n\n it(\"uses custom port in the command\", () => {\n const preset = getPresetById(\"memory-agent\")!;\n const cmd = buildPresetCliCommand(preset, 8888);\n expect(cmd).toContain(\"--port 8888\");\n });\n\n it(\"falls back to defaultPort in the command when no port given\", () => {\n const preset = getPresetById(\"memory-agent\")!;\n const cmd = buildPresetCliCommand(preset);\n expect(cmd).toContain(`--port ${preset.defaultPort}`);\n });\n});"
87
+ },
88
+ {
89
+ "language": "typescript",
90
+ "code": "// Add to existing MarkdownRenderer.svelte.test.ts\n\ndescribe(\"MarkdownRenderer — security\", () => {\n it(\"strips javascript: href from links\", () => {\n const { baseElement } = render(MarkdownRenderer, {\n content: \"[click me](javascript:alert(1))\",\n });\n const anchor = baseElement.querySelector(\"a\");\n expect(anchor?.getAttribute(\"href\")).not.toContain(\"javascript:\");\n });\n\n it(\"renders empty content without crashing\", () => {\n expect(() => render(MarkdownRenderer, { content: \"\" })).not.toThrow();\n });\n});\n\ndescribe(\"MarkdownRenderer — tables\", () => {\n it(\"renders a markdown table\", () => {\n const md = \"| A | B |\\n|---|---|\\n| 1 | 2 |\";\n const { baseElement } = render(MarkdownRenderer, { content: md });\n expect(baseElement.querySelector(\"table\")).toBeTruthy();\n expect(baseElement.querySelector(\"thead\")).toBeTruthy();\n expect(baseElement.querySelector(\"tbody\")).toBeTruthy();\n });\n});\n\ndescribe(\"MarkdownRenderer — nested formatting\", () => {\n it(\"renders bold inside a list item\", () => {\n const { baseElement } = render(MarkdownRenderer, { content: \"- **bold item**\" });\n const strong = baseElement.querySelector(\"strong\");\n expect(strong).toBeTruthy();\n expect(strong?.textContent).toBe(\"bold item\");\n });\n\n it(\"renders multiple fenced code blocks with language labels\", () => {\n const md = \""
91
+ }
92
+ ]
93
+ },
94
+ "durationMs": 141406,
95
+ "model": "sonnet",
96
+ "sandboxMode": "permissive",
97
+ "workerType": "testgaps",
98
+ "timestamp": "2026-05-05T02:58:33.227Z",
99
+ "executionId": "testgaps_1777949771821_elw1j4"
100
+ }
@@ -0,0 +1,11 @@
1
+ {
2
+ "timestamp": "2026-05-05T02:48:11.838Z",
3
+ "projectRoot": "/Users/cohen/Projects/ruflo/ruflo/src/ruvocal",
4
+ "structure": {
5
+ "hasPackageJson": true,
6
+ "hasTsConfig": true,
7
+ "hasClaudeConfig": true,
8
+ "hasClaudeFlow": true
9
+ },
10
+ "scannedAt": 1777949291838
11
+ }
@@ -0,0 +1,6 @@
1
+ {
2
+ "timestamp": "2026-05-05T02:54:11.820Z",
3
+ "patternsConsolidated": 0,
4
+ "memoryCleaned": 0,
5
+ "duplicatesRemoved": 0
6
+ }
@@ -0,0 +1,6 @@
1
+ {
2
+ "trajectoriesRecorded": 0,
3
+ "patternsLearned": 0,
4
+ "signalsProcessed": 0,
5
+ "lastAdaptation": null
6
+ }
@@ -0,0 +1,13 @@
1
+ {
2
+ "id": "session-1777949290618",
3
+ "startedAt": "2026-05-05T02:48:10.618Z",
4
+ "platform": "darwin",
5
+ "cwd": "/Users/cohen/Projects/ruflo/ruflo/src/ruvocal",
6
+ "context": {},
7
+ "metrics": {
8
+ "edits": 0,
9
+ "commands": 0,
10
+ "tasks": 0,
11
+ "errors": 0
12
+ }
13
+ }
@@ -1,9 +1,9 @@
1
- FROM mcr.microsoft.com/devcontainers/typescript-node:1-22-bookworm
2
-
3
- # Install MongoDB tools (mongosh, mongorestore, mongodump) directly from MongoDB repository
4
- RUN curl -fsSL https://www.mongodb.org/static/pgp/server-8.0.asc | gpg --dearmor -o /usr/share/keyrings/mongodb-server-8.0.gpg && \
5
- echo "deb [ signed-by=/usr/share/keyrings/mongodb-server-8.0.gpg ] http://repo.mongodb.org/apt/debian bookworm/mongodb-org/8.0 main" | tee /etc/apt/sources.list.d/mongodb-org-8.0.list && \
6
- apt-get update && \
7
- apt-get install -y mongodb-mongosh mongodb-database-tools vim && \
8
- apt-get autoremove -y && \
9
- rm -rf /var/lib/apt/lists/*
1
+ FROM mcr.microsoft.com/devcontainers/typescript-node:1-22-bookworm
2
+
3
+ # Install MongoDB tools (mongosh, mongorestore, mongodump) directly from MongoDB repository
4
+ RUN curl -fsSL https://www.mongodb.org/static/pgp/server-8.0.asc | gpg --dearmor -o /usr/share/keyrings/mongodb-server-8.0.gpg && \
5
+ echo "deb [ signed-by=/usr/share/keyrings/mongodb-server-8.0.gpg ] http://repo.mongodb.org/apt/debian bookworm/mongodb-org/8.0 main" | tee /etc/apt/sources.list.d/mongodb-org-8.0.list && \
6
+ apt-get update && \
7
+ apt-get install -y mongodb-mongosh mongodb-database-tools vim && \
8
+ apt-get autoremove -y && \
9
+ rm -rf /var/lib/apt/lists/*
@@ -1,36 +1,36 @@
1
- // For format details, see https://aka.ms/devcontainer.json. For config options, see the
2
- // README at: https://github.com/devcontainers/templates/tree/main/src/typescript-node
3
- {
4
- "name": "Node.js & TypeScript",
5
- // Or use a Dockerfile or Docker Compose file. More info: https://containers.dev/guide/dockerfile
6
- "build": {
7
- "dockerfile": "Dockerfile"
8
- },
9
-
10
- "customizations": {
11
- "vscode": {
12
- "extensions": ["esbenp.prettier-vscode", "dbaeumer.vscode-eslint", "svelte.svelte-vscode"]
13
- }
14
- },
15
-
16
- "features": {
17
- // Install docker in container
18
- "ghcr.io/devcontainers/features/docker-in-docker:2": {
19
- // Use proprietary docker engine. I get a timeout error when using the default moby engine and loading
20
- // microsoft's PGP keys
21
- "moby": false
22
- }
23
- }
24
-
25
- // Use 'forwardPorts' to make a list of ports inside the container available locally.
26
- // "forwardPorts": [],
27
-
28
- // Use 'postCreateCommand' to run commands after the container is created.
29
- // "postCreateCommand": "yarn install",
30
-
31
- // Configure tool-specific properties.
32
- // "customizations": {},
33
-
34
- // Uncomment to connect as root instead. More info: https://aka.ms/dev-containers-non-root.
35
- // "remoteUser": "root"
36
- }
1
+ // For format details, see https://aka.ms/devcontainer.json. For config options, see the
2
+ // README at: https://github.com/devcontainers/templates/tree/main/src/typescript-node
3
+ {
4
+ "name": "Node.js & TypeScript",
5
+ // Or use a Dockerfile or Docker Compose file. More info: https://containers.dev/guide/dockerfile
6
+ "build": {
7
+ "dockerfile": "Dockerfile"
8
+ },
9
+
10
+ "customizations": {
11
+ "vscode": {
12
+ "extensions": ["esbenp.prettier-vscode", "dbaeumer.vscode-eslint", "svelte.svelte-vscode"]
13
+ }
14
+ },
15
+
16
+ "features": {
17
+ // Install docker in container
18
+ "ghcr.io/devcontainers/features/docker-in-docker:2": {
19
+ // Use proprietary docker engine. I get a timeout error when using the default moby engine and loading
20
+ // microsoft's PGP keys
21
+ "moby": false
22
+ }
23
+ }
24
+
25
+ // Use 'forwardPorts' to make a list of ports inside the container available locally.
26
+ // "forwardPorts": [],
27
+
28
+ // Use 'postCreateCommand' to run commands after the container is created.
29
+ // "postCreateCommand": "yarn install",
30
+
31
+ // Configure tool-specific properties.
32
+ // "customizations": {},
33
+
34
+ // Uncomment to connect as root instead. More info: https://aka.ms/dev-containers-non-root.
35
+ // "remoteUser": "root"
36
+ }
@@ -1,17 +1,17 @@
1
- Dockerfile
2
- .vscode/
3
- .idea
4
- .gitignore
5
- LICENSE
6
- README.md
7
- node_modules/
8
- .svelte-kit/
9
- .env*
10
- !.env
11
- .env.local
12
- db
13
- # NOTE: previously had `models/**` here to ignore a top-level placeholder
14
- # directory. Even when anchored (`/models/**`), Docker's pattern matcher
15
- # was eliding nested src/routes/api/v2/models/ and src/routes/models/
16
- # from the build context, producing prod images missing those routes.
1
+ Dockerfile
2
+ .vscode/
3
+ .idea
4
+ .gitignore
5
+ LICENSE
6
+ README.md
7
+ node_modules/
8
+ .svelte-kit/
9
+ .env*
10
+ !.env
11
+ .env.local
12
+ db
13
+ # NOTE: previously had `models/**` here to ignore a top-level placeholder
14
+ # directory. Even when anchored (`/models/**`), Docker's pattern matcher
15
+ # was eliding nested src/routes/api/v2/models/ and src/routes/models/
16
+ # from the build context, producing prod images missing those routes.
17
17
  # The placeholder dir is small (single .txt) — letting it through is fine.
@@ -1,13 +1,13 @@
1
- .DS_Store
2
- node_modules
3
- /build
4
- /.svelte-kit
5
- /package
6
- .env
7
- .env.*
8
- !.env.example
9
-
10
- # Ignore files for PNPM, NPM and YARN
11
- pnpm-lock.yaml
12
- package-lock.json
13
- yarn.lock
1
+ .DS_Store
2
+ node_modules
3
+ /build
4
+ /.svelte-kit
5
+ /package
6
+ .env
7
+ .env.*
8
+ !.env.example
9
+
10
+ # Ignore files for PNPM, NPM and YARN
11
+ pnpm-lock.yaml
12
+ package-lock.json
13
+ yarn.lock
@@ -1,45 +1,45 @@
1
- module.exports = {
2
- root: true,
3
- parser: "@typescript-eslint/parser",
4
- extends: [
5
- "eslint:recommended",
6
- "plugin:@typescript-eslint/recommended",
7
- "plugin:svelte/recommended",
8
- "prettier",
9
- ],
10
- plugins: ["@typescript-eslint"],
11
- ignorePatterns: ["*.cjs"],
12
- overrides: [
13
- {
14
- files: ["*.svelte"],
15
- parser: "svelte-eslint-parser",
16
- parserOptions: {
17
- parser: "@typescript-eslint/parser",
18
- },
19
- },
20
- ],
21
- parserOptions: {
22
- sourceType: "module",
23
- ecmaVersion: 2020,
24
- extraFileExtensions: [".svelte"],
25
- },
26
- rules: {
27
- "no-empty": "off",
28
- "require-yield": "off",
29
- "@typescript-eslint/no-explicit-any": "error",
30
- "@typescript-eslint/no-non-null-assertion": "error",
31
- "@typescript-eslint/no-unused-vars": [
32
- // prevent variables with a _ prefix from being marked as unused
33
- "error",
34
- {
35
- argsIgnorePattern: "^_",
36
- },
37
- ],
38
- "object-shorthand": ["error", "always"],
39
- },
40
- env: {
41
- browser: true,
42
- es2017: true,
43
- node: true,
44
- },
45
- };
1
+ module.exports = {
2
+ root: true,
3
+ parser: "@typescript-eslint/parser",
4
+ extends: [
5
+ "eslint:recommended",
6
+ "plugin:@typescript-eslint/recommended",
7
+ "plugin:svelte/recommended",
8
+ "prettier",
9
+ ],
10
+ plugins: ["@typescript-eslint"],
11
+ ignorePatterns: ["*.cjs"],
12
+ overrides: [
13
+ {
14
+ files: ["*.svelte"],
15
+ parser: "svelte-eslint-parser",
16
+ parserOptions: {
17
+ parser: "@typescript-eslint/parser",
18
+ },
19
+ },
20
+ ],
21
+ parserOptions: {
22
+ sourceType: "module",
23
+ ecmaVersion: 2020,
24
+ extraFileExtensions: [".svelte"],
25
+ },
26
+ rules: {
27
+ "no-empty": "off",
28
+ "require-yield": "off",
29
+ "@typescript-eslint/no-explicit-any": "error",
30
+ "@typescript-eslint/no-non-null-assertion": "error",
31
+ "@typescript-eslint/no-unused-vars": [
32
+ // prevent variables with a _ prefix from being marked as unused
33
+ "error",
34
+ {
35
+ argsIgnorePattern: "^_",
36
+ },
37
+ ],
38
+ "object-shorthand": ["error", "always"],
39
+ },
40
+ env: {
41
+ browser: true,
42
+ es2017: true,
43
+ node: true,
44
+ },
45
+ };
@@ -1,18 +1,18 @@
1
- # gcloud builds submit upload filter.
2
- # Without this file, gcloud falls back to .gitignore, which has
3
- # `models/*` (line 16) — that excludes src/routes/api/v2/models/*
4
- # and src/routes/models/* from the build context, producing
5
- # production images missing those SvelteKit routes (root cause
6
- # of /api/v2/models 404 → /favicon 500 → / 500 in Cloud Run).
7
-
8
- .git/
9
- .gitignore
10
- .dockerignore
11
- .gcloudignore
12
- node_modules/
13
- .svelte-kit/
14
- build/
15
- .env.local
16
- .vscode/
17
- .idea/
18
- *.log
1
+ # gcloud builds submit upload filter.
2
+ # Without this file, gcloud falls back to .gitignore, which has
3
+ # `models/*` (line 16) — that excludes src/routes/api/v2/models/*
4
+ # and src/routes/models/* from the build context, producing
5
+ # production images missing those SvelteKit routes (root cause
6
+ # of /api/v2/models 404 → /favicon 500 → / 500 in Cloud Run).
7
+
8
+ .git/
9
+ .gitignore
10
+ .dockerignore
11
+ .gcloudignore
12
+ node_modules/
13
+ .svelte-kit/
14
+ build/
15
+ .env.local
16
+ .vscode/
17
+ .idea/
18
+ *.log