ruflo 3.5.1 → 3.5.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (521) hide show
  1. package/dist/rvf.manifest.json +295 -0
  2. package/package.json +16 -2
  3. package/src/chat-ui/Dockerfile +25 -0
  4. package/src/chat-ui/patch-mcp-url-safety.sh +28 -0
  5. package/src/chat-ui/static/chatui/icon-144x144.png +0 -0
  6. package/src/chat-ui/static/chatui/omni-welcome.gif +0 -0
  7. package/src/config/config.example.json +76 -0
  8. package/src/mcp-bridge/Dockerfile +45 -0
  9. package/src/mcp-bridge/index.js +1668 -0
  10. package/src/mcp-bridge/mcp-stdio-kernel.js +159 -0
  11. package/src/mcp-bridge/package.json +17 -0
  12. package/src/mcp-bridge/test-harness.js +470 -0
  13. package/src/nginx/Dockerfile +10 -0
  14. package/src/nginx/nginx.conf +67 -0
  15. package/src/nginx/static/favicon-dark.svg +4 -0
  16. package/src/nginx/static/favicon.svg +4 -0
  17. package/src/nginx/static/icon.svg +5 -0
  18. package/src/nginx/static/logo.svg +9 -0
  19. package/src/nginx/static/manifest.json +22 -0
  20. package/src/nginx/static/welcome.js +184 -0
  21. package/src/ruvocal/.claude/skills/add-model-descriptions/SKILL.md +73 -0
  22. package/src/ruvocal/.devcontainer/Dockerfile +9 -0
  23. package/src/ruvocal/.devcontainer/devcontainer.json +36 -0
  24. package/src/ruvocal/.dockerignore +13 -0
  25. package/src/ruvocal/.env +194 -0
  26. package/src/ruvocal/.env.ci +1 -0
  27. package/src/ruvocal/.eslintignore +13 -0
  28. package/src/ruvocal/.eslintrc.cjs +45 -0
  29. package/src/ruvocal/.github/ISSUE_TEMPLATE/bug-report--chat-ui-.md +43 -0
  30. package/src/ruvocal/.github/ISSUE_TEMPLATE/config-support.md +9 -0
  31. package/src/ruvocal/.github/ISSUE_TEMPLATE/feature-request--chat-ui-.md +17 -0
  32. package/src/ruvocal/.github/ISSUE_TEMPLATE/huggingchat.md +11 -0
  33. package/src/ruvocal/.github/release.yml +16 -0
  34. package/src/ruvocal/.github/workflows/build-docs.yml +18 -0
  35. package/src/ruvocal/.github/workflows/build-image.yml +142 -0
  36. package/src/ruvocal/.github/workflows/build-pr-docs.yml +20 -0
  37. package/src/ruvocal/.github/workflows/deploy-dev.yml +63 -0
  38. package/src/ruvocal/.github/workflows/deploy-prod.yml +78 -0
  39. package/src/ruvocal/.github/workflows/lint-and-test.yml +84 -0
  40. package/src/ruvocal/.github/workflows/slugify.yaml +72 -0
  41. package/src/ruvocal/.github/workflows/trufflehog.yml +17 -0
  42. package/src/ruvocal/.github/workflows/upload-pr-documentation.yml +16 -0
  43. package/src/ruvocal/.husky/lint-stage-config.js +4 -0
  44. package/src/ruvocal/.husky/pre-commit +2 -0
  45. package/src/ruvocal/.prettierignore +14 -0
  46. package/src/ruvocal/.prettierrc +7 -0
  47. package/src/ruvocal/.vscode/launch.json +11 -0
  48. package/src/ruvocal/.vscode/settings.json +14 -0
  49. package/src/ruvocal/CLAUDE.md +126 -0
  50. package/src/ruvocal/Dockerfile +93 -0
  51. package/src/ruvocal/LICENSE +203 -0
  52. package/src/ruvocal/PRIVACY.md +41 -0
  53. package/src/ruvocal/README.md +190 -0
  54. package/src/ruvocal/chart/Chart.yaml +5 -0
  55. package/src/ruvocal/chart/env/dev.yaml +260 -0
  56. package/src/ruvocal/chart/env/prod.yaml +273 -0
  57. package/src/ruvocal/chart/templates/_helpers.tpl +22 -0
  58. package/src/ruvocal/chart/templates/config.yaml +10 -0
  59. package/src/ruvocal/chart/templates/deployment.yaml +81 -0
  60. package/src/ruvocal/chart/templates/hpa.yaml +45 -0
  61. package/src/ruvocal/chart/templates/infisical.yaml +24 -0
  62. package/src/ruvocal/chart/templates/ingress-internal.yaml +32 -0
  63. package/src/ruvocal/chart/templates/ingress.yaml +32 -0
  64. package/src/ruvocal/chart/templates/network-policy.yaml +36 -0
  65. package/src/ruvocal/chart/templates/service-account.yaml +13 -0
  66. package/src/ruvocal/chart/templates/service-monitor.yaml +17 -0
  67. package/src/ruvocal/chart/templates/service.yaml +21 -0
  68. package/src/ruvocal/chart/values.yaml +73 -0
  69. package/src/ruvocal/docker-compose.yml +21 -0
  70. package/src/ruvocal/docs/adr/ADR-029-HUGGINGFACE-CHAT-UI-CLOUD-RUN.md +1236 -0
  71. package/src/ruvocal/docs/adr/ADR-033-RUVECTOR-RUFLO-MCP-INTEGRATION.md +111 -0
  72. package/src/ruvocal/docs/adr/ADR-034-OPTIONAL-MCP-BACKENDS.md +117 -0
  73. package/src/ruvocal/docs/adr/ADR-035-MCP-TOOL-GROUPS.md +186 -0
  74. package/src/ruvocal/docs/adr/ADR-037-AUTOPILOT-CHAT-MODE.md +1500 -0
  75. package/src/ruvocal/docs/adr/ADR-038-RUVOCAL-FORK.md +286 -0
  76. package/src/ruvocal/docs/source/_toctree.yml +30 -0
  77. package/src/ruvocal/docs/source/configuration/common-issues.md +38 -0
  78. package/src/ruvocal/docs/source/configuration/llm-router.md +105 -0
  79. package/src/ruvocal/docs/source/configuration/mcp-tools.md +84 -0
  80. package/src/ruvocal/docs/source/configuration/metrics.md +9 -0
  81. package/src/ruvocal/docs/source/configuration/open-id.md +57 -0
  82. package/src/ruvocal/docs/source/configuration/overview.md +89 -0
  83. package/src/ruvocal/docs/source/configuration/theming.md +20 -0
  84. package/src/ruvocal/docs/source/developing/architecture.md +48 -0
  85. package/src/ruvocal/docs/source/index.md +53 -0
  86. package/src/ruvocal/docs/source/installation/docker.md +43 -0
  87. package/src/ruvocal/docs/source/installation/helm.md +43 -0
  88. package/src/ruvocal/docs/source/installation/local.md +62 -0
  89. package/src/ruvocal/entrypoint.sh +19 -0
  90. package/src/ruvocal/mcp-bridge/.claude-flow/agents/store.json +27 -0
  91. package/src/ruvocal/mcp-bridge/.claude-flow/daemon-state.json +130 -0
  92. package/src/ruvocal/mcp-bridge/.claude-flow/daemon.log +0 -0
  93. package/src/ruvocal/mcp-bridge/.claude-flow/daemon.pid +1 -0
  94. package/src/ruvocal/mcp-bridge/.claude-flow/tasks/store.json +21 -0
  95. package/src/ruvocal/mcp-bridge/.swarm/hnsw.index +0 -0
  96. package/src/ruvocal/mcp-bridge/.swarm/hnsw.metadata.json +1 -0
  97. package/src/ruvocal/mcp-bridge/.swarm/memory.db +0 -0
  98. package/src/ruvocal/mcp-bridge/.swarm/model-router-state.json +14 -0
  99. package/src/ruvocal/mcp-bridge/.swarm/schema.sql +305 -0
  100. package/src/ruvocal/mcp-bridge/Dockerfile +45 -0
  101. package/src/ruvocal/mcp-bridge/cloudbuild.yaml +49 -0
  102. package/src/ruvocal/mcp-bridge/index.js +1864 -0
  103. package/src/ruvocal/mcp-bridge/mcp-stdio-kernel.js +159 -0
  104. package/src/ruvocal/mcp-bridge/package-lock.json +762 -0
  105. package/src/ruvocal/mcp-bridge/package.json +17 -0
  106. package/src/ruvocal/mcp-bridge/test-harness.js +470 -0
  107. package/src/ruvocal/models/add-your-models-here.txt +1 -0
  108. package/src/ruvocal/package-lock.json +11741 -0
  109. package/src/ruvocal/package.json +121 -0
  110. package/src/ruvocal/postcss.config.js +6 -0
  111. package/src/ruvocal/rvf.manifest.json +204 -0
  112. package/src/ruvocal/scripts/config.ts +64 -0
  113. package/src/ruvocal/scripts/generate-welcome.mjs +181 -0
  114. package/src/ruvocal/scripts/populate.ts +288 -0
  115. package/src/ruvocal/scripts/samples.txt +194 -0
  116. package/src/ruvocal/scripts/setups/vitest-setup-client.ts +0 -0
  117. package/src/ruvocal/scripts/setups/vitest-setup-server.ts +44 -0
  118. package/src/ruvocal/scripts/updateLocalEnv.ts +48 -0
  119. package/src/ruvocal/src/ambient.d.ts +7 -0
  120. package/src/ruvocal/src/app.d.ts +29 -0
  121. package/src/ruvocal/src/app.html +53 -0
  122. package/src/ruvocal/src/hooks.server.ts +32 -0
  123. package/src/ruvocal/src/hooks.ts +6 -0
  124. package/src/ruvocal/src/lib/APIClient.ts +148 -0
  125. package/src/ruvocal/src/lib/actions/clickOutside.ts +18 -0
  126. package/src/ruvocal/src/lib/actions/snapScrollToBottom.ts +346 -0
  127. package/src/ruvocal/src/lib/buildPrompt.ts +33 -0
  128. package/src/ruvocal/src/lib/components/AnnouncementBanner.svelte +20 -0
  129. package/src/ruvocal/src/lib/components/BackgroundGenerationPoller.svelte +168 -0
  130. package/src/ruvocal/src/lib/components/CodeBlock.svelte +73 -0
  131. package/src/ruvocal/src/lib/components/CopyToClipBoardBtn.svelte +92 -0
  132. package/src/ruvocal/src/lib/components/DeleteConversationModal.svelte +75 -0
  133. package/src/ruvocal/src/lib/components/EditConversationModal.svelte +100 -0
  134. package/src/ruvocal/src/lib/components/ExpandNavigation.svelte +22 -0
  135. package/src/ruvocal/src/lib/components/HoverTooltip.svelte +44 -0
  136. package/src/ruvocal/src/lib/components/HtmlPreviewModal.svelte +143 -0
  137. package/src/ruvocal/src/lib/components/InfiniteScroll.svelte +50 -0
  138. package/src/ruvocal/src/lib/components/MobileNav.svelte +300 -0
  139. package/src/ruvocal/src/lib/components/Modal.svelte +115 -0
  140. package/src/ruvocal/src/lib/components/ModelCardMetadata.svelte +71 -0
  141. package/src/ruvocal/src/lib/components/NavConversationItem.svelte +151 -0
  142. package/src/ruvocal/src/lib/components/NavMenu.svelte +295 -0
  143. package/src/ruvocal/src/lib/components/Pagination.svelte +97 -0
  144. package/src/ruvocal/src/lib/components/PaginationArrow.svelte +27 -0
  145. package/src/ruvocal/src/lib/components/Portal.svelte +24 -0
  146. package/src/ruvocal/src/lib/components/RetryBtn.svelte +18 -0
  147. package/src/ruvocal/src/lib/components/RuFloUniverse.svelte +185 -0
  148. package/src/ruvocal/src/lib/components/ScrollToBottomBtn.svelte +47 -0
  149. package/src/ruvocal/src/lib/components/ScrollToPreviousBtn.svelte +77 -0
  150. package/src/ruvocal/src/lib/components/ShareConversationModal.svelte +182 -0
  151. package/src/ruvocal/src/lib/components/StopGeneratingBtn.svelte +69 -0
  152. package/src/ruvocal/src/lib/components/SubscribeModal.svelte +87 -0
  153. package/src/ruvocal/src/lib/components/Switch.svelte +36 -0
  154. package/src/ruvocal/src/lib/components/SystemPromptModal.svelte +44 -0
  155. package/src/ruvocal/src/lib/components/Toast.svelte +27 -0
  156. package/src/ruvocal/src/lib/components/Tooltip.svelte +30 -0
  157. package/src/ruvocal/src/lib/components/WelcomeModal.svelte +46 -0
  158. package/src/ruvocal/src/lib/components/chat/Alternatives.svelte +77 -0
  159. package/src/ruvocal/src/lib/components/chat/BlockWrapper.svelte +72 -0
  160. package/src/ruvocal/src/lib/components/chat/ChatInput.svelte +490 -0
  161. package/src/ruvocal/src/lib/components/chat/ChatIntroduction.svelte +123 -0
  162. package/src/ruvocal/src/lib/components/chat/ChatMessage.svelte +548 -0
  163. package/src/ruvocal/src/lib/components/chat/ChatWindow.svelte +939 -0
  164. package/src/ruvocal/src/lib/components/chat/FileDropzone.svelte +92 -0
  165. package/src/ruvocal/src/lib/components/chat/ImageLightbox.svelte +66 -0
  166. package/src/ruvocal/src/lib/components/chat/MarkdownBlock.svelte +23 -0
  167. package/src/ruvocal/src/lib/components/chat/MarkdownRenderer.svelte +69 -0
  168. package/src/ruvocal/src/lib/components/chat/MarkdownRenderer.svelte.test.ts +58 -0
  169. package/src/ruvocal/src/lib/components/chat/MessageAvatar.svelte +103 -0
  170. package/src/ruvocal/src/lib/components/chat/ModelSwitch.svelte +64 -0
  171. package/src/ruvocal/src/lib/components/chat/OpenReasoningResults.svelte +81 -0
  172. package/src/ruvocal/src/lib/components/chat/TaskGroup.svelte +88 -0
  173. package/src/ruvocal/src/lib/components/chat/ToolUpdate.svelte +273 -0
  174. package/src/ruvocal/src/lib/components/chat/UploadedFile.svelte +253 -0
  175. package/src/ruvocal/src/lib/components/chat/UrlFetchModal.svelte +203 -0
  176. package/src/ruvocal/src/lib/components/chat/VoiceRecorder.svelte +214 -0
  177. package/src/ruvocal/src/lib/components/icons/IconBurger.svelte +20 -0
  178. package/src/ruvocal/src/lib/components/icons/IconCheap.svelte +20 -0
  179. package/src/ruvocal/src/lib/components/icons/IconChevron.svelte +24 -0
  180. package/src/ruvocal/src/lib/components/icons/IconDazzled.svelte +40 -0
  181. package/src/ruvocal/src/lib/components/icons/IconFast.svelte +20 -0
  182. package/src/ruvocal/src/lib/components/icons/IconLoading.svelte +22 -0
  183. package/src/ruvocal/src/lib/components/icons/IconMCP.svelte +28 -0
  184. package/src/ruvocal/src/lib/components/icons/IconMoon.svelte +21 -0
  185. package/src/ruvocal/src/lib/components/icons/IconNew.svelte +20 -0
  186. package/src/ruvocal/src/lib/components/icons/IconOmni.svelte +90 -0
  187. package/src/ruvocal/src/lib/components/icons/IconPaperclip.svelte +24 -0
  188. package/src/ruvocal/src/lib/components/icons/IconPro.svelte +37 -0
  189. package/src/ruvocal/src/lib/components/icons/IconShare.svelte +21 -0
  190. package/src/ruvocal/src/lib/components/icons/IconSun.svelte +93 -0
  191. package/src/ruvocal/src/lib/components/icons/Logo.svelte +68 -0
  192. package/src/ruvocal/src/lib/components/icons/LogoHuggingFaceBorderless.svelte +54 -0
  193. package/src/ruvocal/src/lib/components/mcp/AddServerForm.svelte +250 -0
  194. package/src/ruvocal/src/lib/components/mcp/MCPServerManager.svelte +185 -0
  195. package/src/ruvocal/src/lib/components/mcp/ServerCard.svelte +203 -0
  196. package/src/ruvocal/src/lib/components/players/AudioPlayer.svelte +82 -0
  197. package/src/ruvocal/src/lib/components/voice/AudioWaveform.svelte +96 -0
  198. package/src/ruvocal/src/lib/constants/mcpExamples.ts +135 -0
  199. package/src/ruvocal/src/lib/constants/mime.ts +11 -0
  200. package/src/ruvocal/src/lib/constants/pagination.ts +1 -0
  201. package/src/ruvocal/src/lib/constants/publicSepToken.ts +1 -0
  202. package/src/ruvocal/src/lib/constants/routerExamples.ts +209 -0
  203. package/src/ruvocal/src/lib/createShareLink.ts +27 -0
  204. package/src/ruvocal/src/lib/jobs/refresh-conversation-stats.ts +297 -0
  205. package/src/ruvocal/src/lib/migrations/lock.ts +56 -0
  206. package/src/ruvocal/src/lib/migrations/migrations.spec.ts +74 -0
  207. package/src/ruvocal/src/lib/migrations/migrations.ts +109 -0
  208. package/src/ruvocal/src/lib/migrations/routines/01-update-search-assistants.ts +50 -0
  209. package/src/ruvocal/src/lib/migrations/routines/02-update-assistants-models.ts +48 -0
  210. package/src/ruvocal/src/lib/migrations/routines/04-update-message-updates.ts +151 -0
  211. package/src/ruvocal/src/lib/migrations/routines/05-update-message-files.ts +56 -0
  212. package/src/ruvocal/src/lib/migrations/routines/06-trim-message-updates.ts +56 -0
  213. package/src/ruvocal/src/lib/migrations/routines/08-update-featured-to-review.ts +32 -0
  214. package/src/ruvocal/src/lib/migrations/routines/09-delete-empty-conversations.spec.ts +214 -0
  215. package/src/ruvocal/src/lib/migrations/routines/09-delete-empty-conversations.ts +88 -0
  216. package/src/ruvocal/src/lib/migrations/routines/10-update-reports-assistantid.ts +29 -0
  217. package/src/ruvocal/src/lib/migrations/routines/index.ts +15 -0
  218. package/src/ruvocal/src/lib/server/__tests__/conversation-stop-generating.spec.ts +103 -0
  219. package/src/ruvocal/src/lib/server/abortRegistry.ts +57 -0
  220. package/src/ruvocal/src/lib/server/abortedGenerations.ts +43 -0
  221. package/src/ruvocal/src/lib/server/adminToken.ts +62 -0
  222. package/src/ruvocal/src/lib/server/api/__tests__/conversations-id.spec.ts +296 -0
  223. package/src/ruvocal/src/lib/server/api/__tests__/conversations-message.spec.ts +216 -0
  224. package/src/ruvocal/src/lib/server/api/__tests__/conversations.spec.ts +235 -0
  225. package/src/ruvocal/src/lib/server/api/__tests__/misc.spec.ts +72 -0
  226. package/src/ruvocal/src/lib/server/api/__tests__/testHelpers.ts +86 -0
  227. package/src/ruvocal/src/lib/server/api/__tests__/user-reports.spec.ts +78 -0
  228. package/src/ruvocal/src/lib/server/api/__tests__/user.spec.ts +239 -0
  229. package/src/ruvocal/src/lib/server/api/types.ts +37 -0
  230. package/src/ruvocal/src/lib/server/api/utils/requireAuth.ts +22 -0
  231. package/src/ruvocal/src/lib/server/api/utils/resolveConversation.ts +69 -0
  232. package/src/ruvocal/src/lib/server/api/utils/resolveModel.ts +27 -0
  233. package/src/ruvocal/src/lib/server/api/utils/superjsonResponse.ts +15 -0
  234. package/src/ruvocal/src/lib/server/apiToken.ts +11 -0
  235. package/src/ruvocal/src/lib/server/auth.ts +554 -0
  236. package/src/ruvocal/src/lib/server/config.ts +187 -0
  237. package/src/ruvocal/src/lib/server/conversation.ts +83 -0
  238. package/src/ruvocal/src/lib/server/database/__tests__/rvf.spec.ts +709 -0
  239. package/src/ruvocal/src/lib/server/database/postgres.ts +700 -0
  240. package/src/ruvocal/src/lib/server/database/rvf.ts +1078 -0
  241. package/src/ruvocal/src/lib/server/database.ts +145 -0
  242. package/src/ruvocal/src/lib/server/endpoints/document.ts +68 -0
  243. package/src/ruvocal/src/lib/server/endpoints/endpoints.ts +43 -0
  244. package/src/ruvocal/src/lib/server/endpoints/images.ts +211 -0
  245. package/src/ruvocal/src/lib/server/endpoints/openai/endpointOai.ts +266 -0
  246. package/src/ruvocal/src/lib/server/endpoints/openai/openAIChatToTextGenerationStream.ts +212 -0
  247. package/src/ruvocal/src/lib/server/endpoints/openai/openAICompletionToTextGenerationStream.ts +32 -0
  248. package/src/ruvocal/src/lib/server/endpoints/preprocessMessages.ts +61 -0
  249. package/src/ruvocal/src/lib/server/exitHandler.ts +59 -0
  250. package/src/ruvocal/src/lib/server/files/downloadFile.ts +34 -0
  251. package/src/ruvocal/src/lib/server/files/uploadFile.ts +29 -0
  252. package/src/ruvocal/src/lib/server/findRepoRoot.ts +13 -0
  253. package/src/ruvocal/src/lib/server/fonts/Inter-Black.ttf +0 -0
  254. package/src/ruvocal/src/lib/server/fonts/Inter-Bold.ttf +0 -0
  255. package/src/ruvocal/src/lib/server/fonts/Inter-ExtraBold.ttf +0 -0
  256. package/src/ruvocal/src/lib/server/fonts/Inter-ExtraLight.ttf +0 -0
  257. package/src/ruvocal/src/lib/server/fonts/Inter-Light.ttf +0 -0
  258. package/src/ruvocal/src/lib/server/fonts/Inter-Medium.ttf +0 -0
  259. package/src/ruvocal/src/lib/server/fonts/Inter-Regular.ttf +0 -0
  260. package/src/ruvocal/src/lib/server/fonts/Inter-SemiBold.ttf +0 -0
  261. package/src/ruvocal/src/lib/server/fonts/Inter-Thin.ttf +0 -0
  262. package/src/ruvocal/src/lib/server/generateFromDefaultEndpoint.ts +46 -0
  263. package/src/ruvocal/src/lib/server/hooks/error.ts +37 -0
  264. package/src/ruvocal/src/lib/server/hooks/fetch.ts +22 -0
  265. package/src/ruvocal/src/lib/server/hooks/handle.ts +250 -0
  266. package/src/ruvocal/src/lib/server/hooks/init.ts +51 -0
  267. package/src/ruvocal/src/lib/server/isURLLocal.spec.ts +31 -0
  268. package/src/ruvocal/src/lib/server/isURLLocal.ts +74 -0
  269. package/src/ruvocal/src/lib/server/logger.ts +42 -0
  270. package/src/ruvocal/src/lib/server/mcp/clientPool.ts +70 -0
  271. package/src/ruvocal/src/lib/server/mcp/hf.ts +32 -0
  272. package/src/ruvocal/src/lib/server/mcp/httpClient.ts +122 -0
  273. package/src/ruvocal/src/lib/server/mcp/registry.ts +76 -0
  274. package/src/ruvocal/src/lib/server/mcp/tools.ts +196 -0
  275. package/src/ruvocal/src/lib/server/metrics.ts +255 -0
  276. package/src/ruvocal/src/lib/server/models.ts +518 -0
  277. package/src/ruvocal/src/lib/server/requestContext.ts +55 -0
  278. package/src/ruvocal/src/lib/server/router/arch.ts +230 -0
  279. package/src/ruvocal/src/lib/server/router/endpoint.ts +316 -0
  280. package/src/ruvocal/src/lib/server/router/multimodal.ts +28 -0
  281. package/src/ruvocal/src/lib/server/router/policy.ts +49 -0
  282. package/src/ruvocal/src/lib/server/router/toolsRoute.ts +51 -0
  283. package/src/ruvocal/src/lib/server/router/types.ts +21 -0
  284. package/src/ruvocal/src/lib/server/sendSlack.ts +23 -0
  285. package/src/ruvocal/src/lib/server/textGeneration/generate.ts +258 -0
  286. package/src/ruvocal/src/lib/server/textGeneration/index.ts +95 -0
  287. package/src/ruvocal/src/lib/server/textGeneration/mcp/fileRefs.ts +155 -0
  288. package/src/ruvocal/src/lib/server/textGeneration/mcp/routerResolution.ts +108 -0
  289. package/src/ruvocal/src/lib/server/textGeneration/mcp/runMcpFlow.ts +822 -0
  290. package/src/ruvocal/src/lib/server/textGeneration/mcp/toolInvocation.ts +349 -0
  291. package/src/ruvocal/src/lib/server/textGeneration/reasoning.ts +23 -0
  292. package/src/ruvocal/src/lib/server/textGeneration/title.ts +83 -0
  293. package/src/ruvocal/src/lib/server/textGeneration/types.ts +26 -0
  294. package/src/ruvocal/src/lib/server/textGeneration/utils/prepareFiles.ts +88 -0
  295. package/src/ruvocal/src/lib/server/textGeneration/utils/routing.ts +21 -0
  296. package/src/ruvocal/src/lib/server/textGeneration/utils/toolPrompt.ts +49 -0
  297. package/src/ruvocal/src/lib/server/urlSafety.ts +72 -0
  298. package/src/ruvocal/src/lib/server/usageLimits.ts +30 -0
  299. package/src/ruvocal/src/lib/stores/autopilotStore.svelte.ts +175 -0
  300. package/src/ruvocal/src/lib/stores/backgroundGenerations.svelte.ts +32 -0
  301. package/src/ruvocal/src/lib/stores/backgroundGenerations.ts +1 -0
  302. package/src/ruvocal/src/lib/stores/errors.ts +9 -0
  303. package/src/ruvocal/src/lib/stores/isAborted.ts +3 -0
  304. package/src/ruvocal/src/lib/stores/isPro.ts +4 -0
  305. package/src/ruvocal/src/lib/stores/loading.ts +3 -0
  306. package/src/ruvocal/src/lib/stores/mcpServers.ts +345 -0
  307. package/src/ruvocal/src/lib/stores/pendingChatInput.ts +3 -0
  308. package/src/ruvocal/src/lib/stores/pendingMessage.ts +9 -0
  309. package/src/ruvocal/src/lib/stores/settings.ts +182 -0
  310. package/src/ruvocal/src/lib/stores/shareModal.ts +13 -0
  311. package/src/ruvocal/src/lib/stores/titleUpdate.ts +8 -0
  312. package/src/ruvocal/src/lib/switchTheme.ts +124 -0
  313. package/src/ruvocal/src/lib/types/AbortedGeneration.ts +8 -0
  314. package/src/ruvocal/src/lib/types/Assistant.ts +31 -0
  315. package/src/ruvocal/src/lib/types/AssistantStats.ts +11 -0
  316. package/src/ruvocal/src/lib/types/ConfigKey.ts +4 -0
  317. package/src/ruvocal/src/lib/types/ConvSidebar.ts +9 -0
  318. package/src/ruvocal/src/lib/types/Conversation.ts +27 -0
  319. package/src/ruvocal/src/lib/types/ConversationStats.ts +13 -0
  320. package/src/ruvocal/src/lib/types/Message.ts +41 -0
  321. package/src/ruvocal/src/lib/types/MessageEvent.ts +10 -0
  322. package/src/ruvocal/src/lib/types/MessageUpdate.ts +139 -0
  323. package/src/ruvocal/src/lib/types/MigrationResult.ts +7 -0
  324. package/src/ruvocal/src/lib/types/Model.ts +23 -0
  325. package/src/ruvocal/src/lib/types/Report.ts +12 -0
  326. package/src/ruvocal/src/lib/types/Review.ts +6 -0
  327. package/src/ruvocal/src/lib/types/Semaphore.ts +19 -0
  328. package/src/ruvocal/src/lib/types/Session.ts +22 -0
  329. package/src/ruvocal/src/lib/types/Settings.ts +86 -0
  330. package/src/ruvocal/src/lib/types/SharedConversation.ts +9 -0
  331. package/src/ruvocal/src/lib/types/Template.ts +6 -0
  332. package/src/ruvocal/src/lib/types/Timestamps.ts +4 -0
  333. package/src/ruvocal/src/lib/types/TokenCache.ts +6 -0
  334. package/src/ruvocal/src/lib/types/Tool.ts +74 -0
  335. package/src/ruvocal/src/lib/types/UrlDependency.ts +5 -0
  336. package/src/ruvocal/src/lib/types/User.ts +14 -0
  337. package/src/ruvocal/src/lib/utils/PublicConfig.svelte.ts +75 -0
  338. package/src/ruvocal/src/lib/utils/auth.ts +17 -0
  339. package/src/ruvocal/src/lib/utils/chunk.ts +33 -0
  340. package/src/ruvocal/src/lib/utils/cookiesAreEnabled.ts +13 -0
  341. package/src/ruvocal/src/lib/utils/debounce.ts +17 -0
  342. package/src/ruvocal/src/lib/utils/deepestChild.ts +6 -0
  343. package/src/ruvocal/src/lib/utils/favicon.ts +21 -0
  344. package/src/ruvocal/src/lib/utils/fetchJSON.ts +23 -0
  345. package/src/ruvocal/src/lib/utils/file2base64.ts +14 -0
  346. package/src/ruvocal/src/lib/utils/formatUserCount.ts +37 -0
  347. package/src/ruvocal/src/lib/utils/generationState.spec.ts +75 -0
  348. package/src/ruvocal/src/lib/utils/generationState.ts +26 -0
  349. package/src/ruvocal/src/lib/utils/getHref.ts +41 -0
  350. package/src/ruvocal/src/lib/utils/getReturnFromGenerator.ts +7 -0
  351. package/src/ruvocal/src/lib/utils/haptics.ts +64 -0
  352. package/src/ruvocal/src/lib/utils/hashConv.ts +12 -0
  353. package/src/ruvocal/src/lib/utils/hf.ts +17 -0
  354. package/src/ruvocal/src/lib/utils/isDesktop.ts +7 -0
  355. package/src/ruvocal/src/lib/utils/isUrl.ts +8 -0
  356. package/src/ruvocal/src/lib/utils/isVirtualKeyboard.ts +16 -0
  357. package/src/ruvocal/src/lib/utils/loadAttachmentsFromUrls.ts +115 -0
  358. package/src/ruvocal/src/lib/utils/marked.spec.ts +96 -0
  359. package/src/ruvocal/src/lib/utils/marked.ts +531 -0
  360. package/src/ruvocal/src/lib/utils/mcpValidation.ts +147 -0
  361. package/src/ruvocal/src/lib/utils/mergeAsyncGenerators.ts +38 -0
  362. package/src/ruvocal/src/lib/utils/messageUpdates.spec.ts +262 -0
  363. package/src/ruvocal/src/lib/utils/messageUpdates.ts +324 -0
  364. package/src/ruvocal/src/lib/utils/mime.ts +56 -0
  365. package/src/ruvocal/src/lib/utils/models.ts +14 -0
  366. package/src/ruvocal/src/lib/utils/parseBlocks.ts +120 -0
  367. package/src/ruvocal/src/lib/utils/parseIncompleteMarkdown.ts +644 -0
  368. package/src/ruvocal/src/lib/utils/parseStringToList.ts +10 -0
  369. package/src/ruvocal/src/lib/utils/randomUuid.ts +14 -0
  370. package/src/ruvocal/src/lib/utils/searchTokens.ts +33 -0
  371. package/src/ruvocal/src/lib/utils/sha256.ts +7 -0
  372. package/src/ruvocal/src/lib/utils/stringifyError.ts +12 -0
  373. package/src/ruvocal/src/lib/utils/sum.ts +3 -0
  374. package/src/ruvocal/src/lib/utils/template.spec.ts +59 -0
  375. package/src/ruvocal/src/lib/utils/template.ts +53 -0
  376. package/src/ruvocal/src/lib/utils/timeout.ts +9 -0
  377. package/src/ruvocal/src/lib/utils/toolProgress.spec.ts +46 -0
  378. package/src/ruvocal/src/lib/utils/toolProgress.ts +11 -0
  379. package/src/ruvocal/src/lib/utils/tree/addChildren.spec.ts +102 -0
  380. package/src/ruvocal/src/lib/utils/tree/addChildren.ts +48 -0
  381. package/src/ruvocal/src/lib/utils/tree/addSibling.spec.ts +81 -0
  382. package/src/ruvocal/src/lib/utils/tree/addSibling.ts +41 -0
  383. package/src/ruvocal/src/lib/utils/tree/buildSubtree.spec.ts +110 -0
  384. package/src/ruvocal/src/lib/utils/tree/buildSubtree.ts +24 -0
  385. package/src/ruvocal/src/lib/utils/tree/convertLegacyConversation.spec.ts +31 -0
  386. package/src/ruvocal/src/lib/utils/tree/convertLegacyConversation.ts +36 -0
  387. package/src/ruvocal/src/lib/utils/tree/isMessageId.spec.ts +15 -0
  388. package/src/ruvocal/src/lib/utils/tree/isMessageId.ts +5 -0
  389. package/src/ruvocal/src/lib/utils/tree/tree.d.ts +14 -0
  390. package/src/ruvocal/src/lib/utils/tree/treeHelpers.spec.ts +167 -0
  391. package/src/ruvocal/src/lib/utils/updates.ts +39 -0
  392. package/src/ruvocal/src/lib/utils/urlParams.ts +13 -0
  393. package/src/ruvocal/src/lib/workers/autopilotWorker.ts +221 -0
  394. package/src/ruvocal/src/lib/workers/detailFetchWorker.ts +100 -0
  395. package/src/ruvocal/src/lib/workers/markdownWorker.ts +61 -0
  396. package/src/ruvocal/src/routes/+error.svelte +20 -0
  397. package/src/ruvocal/src/routes/+layout.svelte +324 -0
  398. package/src/ruvocal/src/routes/+layout.ts +91 -0
  399. package/src/ruvocal/src/routes/+page.svelte +168 -0
  400. package/src/ruvocal/src/routes/.well-known/oauth-cimd/+server.ts +37 -0
  401. package/src/ruvocal/src/routes/__debug/openai/+server.ts +21 -0
  402. package/src/ruvocal/src/routes/admin/export/+server.ts +159 -0
  403. package/src/ruvocal/src/routes/admin/stats/compute/+server.ts +16 -0
  404. package/src/ruvocal/src/routes/api/conversation/[id]/+server.ts +40 -0
  405. package/src/ruvocal/src/routes/api/conversation/[id]/message/[messageId]/+server.ts +42 -0
  406. package/src/ruvocal/src/routes/api/conversations/+server.ts +48 -0
  407. package/src/ruvocal/src/routes/api/fetch-url/+server.ts +147 -0
  408. package/src/ruvocal/src/routes/api/mcp/health/+server.ts +292 -0
  409. package/src/ruvocal/src/routes/api/mcp/servers/+server.ts +32 -0
  410. package/src/ruvocal/src/routes/api/models/+server.ts +25 -0
  411. package/src/ruvocal/src/routes/api/transcribe/+server.ts +104 -0
  412. package/src/ruvocal/src/routes/api/user/+server.ts +15 -0
  413. package/src/ruvocal/src/routes/api/user/validate-token/+server.ts +20 -0
  414. package/src/ruvocal/src/routes/api/v2/conversations/+server.ts +48 -0
  415. package/src/ruvocal/src/routes/api/v2/conversations/[id]/+server.ts +94 -0
  416. package/src/ruvocal/src/routes/api/v2/conversations/[id]/message/[messageId]/+server.ts +43 -0
  417. package/src/ruvocal/src/routes/api/v2/conversations/import-share/+server.ts +23 -0
  418. package/src/ruvocal/src/routes/api/v2/debug/config/+server.ts +16 -0
  419. package/src/ruvocal/src/routes/api/v2/debug/refresh/+server.ts +30 -0
  420. package/src/ruvocal/src/routes/api/v2/export/+server.ts +196 -0
  421. package/src/ruvocal/src/routes/api/v2/feature-flags/+server.ts +14 -0
  422. package/src/ruvocal/src/routes/api/v2/models/+server.ts +38 -0
  423. package/src/ruvocal/src/routes/api/v2/models/[namespace]/+server.ts +8 -0
  424. package/src/ruvocal/src/routes/api/v2/models/[namespace]/[model]/+server.ts +8 -0
  425. package/src/ruvocal/src/routes/api/v2/models/[namespace]/[model]/subscribe/+server.ts +28 -0
  426. package/src/ruvocal/src/routes/api/v2/models/[namespace]/subscribe/+server.ts +28 -0
  427. package/src/ruvocal/src/routes/api/v2/models/old/+server.ts +7 -0
  428. package/src/ruvocal/src/routes/api/v2/models/refresh/+server.ts +33 -0
  429. package/src/ruvocal/src/routes/api/v2/public-config/+server.ts +7 -0
  430. package/src/ruvocal/src/routes/api/v2/user/+server.ts +17 -0
  431. package/src/ruvocal/src/routes/api/v2/user/billing-orgs/+server.ts +73 -0
  432. package/src/ruvocal/src/routes/api/v2/user/reports/+server.ts +17 -0
  433. package/src/ruvocal/src/routes/api/v2/user/settings/+server.ts +103 -0
  434. package/src/ruvocal/src/routes/conversation/+server.ts +115 -0
  435. package/src/ruvocal/src/routes/conversation/[id]/+page.svelte +582 -0
  436. package/src/ruvocal/src/routes/conversation/[id]/+page.ts +60 -0
  437. package/src/ruvocal/src/routes/conversation/[id]/+server.ts +736 -0
  438. package/src/ruvocal/src/routes/conversation/[id]/message/[messageId]/prompt/+server.ts +66 -0
  439. package/src/ruvocal/src/routes/conversation/[id]/output/[sha256]/+server.ts +58 -0
  440. package/src/ruvocal/src/routes/conversation/[id]/share/+server.ts +69 -0
  441. package/src/ruvocal/src/routes/conversation/[id]/stop-generating/+server.ts +35 -0
  442. package/src/ruvocal/src/routes/healthcheck/+server.ts +3 -0
  443. package/src/ruvocal/src/routes/login/+server.ts +5 -0
  444. package/src/ruvocal/src/routes/login/callback/+server.ts +103 -0
  445. package/src/ruvocal/src/routes/login/callback/updateUser.spec.ts +157 -0
  446. package/src/ruvocal/src/routes/login/callback/updateUser.ts +215 -0
  447. package/src/ruvocal/src/routes/logout/+server.ts +18 -0
  448. package/src/ruvocal/src/routes/metrics/+server.ts +18 -0
  449. package/src/ruvocal/src/routes/models/+page.svelte +233 -0
  450. package/src/ruvocal/src/routes/models/[...model]/+page.svelte +161 -0
  451. package/src/ruvocal/src/routes/models/[...model]/+page.ts +14 -0
  452. package/src/ruvocal/src/routes/models/[...model]/thumbnail.png/+server.ts +64 -0
  453. package/src/ruvocal/src/routes/models/[...model]/thumbnail.png/ModelThumbnail.svelte +28 -0
  454. package/src/ruvocal/src/routes/privacy/+page.svelte +11 -0
  455. package/src/ruvocal/src/routes/r/[id]/+page.ts +34 -0
  456. package/src/ruvocal/src/routes/settings/(nav)/+layout.svelte +282 -0
  457. package/src/ruvocal/src/routes/settings/(nav)/+layout.ts +1 -0
  458. package/src/ruvocal/src/routes/settings/(nav)/+page.svelte +0 -0
  459. package/src/ruvocal/src/routes/settings/(nav)/+server.ts +53 -0
  460. package/src/ruvocal/src/routes/settings/(nav)/[...model]/+page.svelte +464 -0
  461. package/src/ruvocal/src/routes/settings/(nav)/[...model]/+page.ts +14 -0
  462. package/src/ruvocal/src/routes/settings/(nav)/application/+page.svelte +362 -0
  463. package/src/ruvocal/src/routes/settings/+layout.svelte +40 -0
  464. package/src/ruvocal/src/styles/highlight-js.css +195 -0
  465. package/src/ruvocal/src/styles/main.css +144 -0
  466. package/src/ruvocal/static/chatui/apple-touch-icon.png +0 -0
  467. package/src/ruvocal/static/chatui/favicon-dark.svg +3 -0
  468. package/src/ruvocal/static/chatui/favicon-dev.svg +3 -0
  469. package/src/ruvocal/static/chatui/favicon.ico +0 -0
  470. package/src/ruvocal/static/chatui/favicon.svg +3 -0
  471. package/src/ruvocal/static/chatui/icon-128x128.png +0 -0
  472. package/src/ruvocal/static/chatui/icon-144x144.png +0 -0
  473. package/src/ruvocal/static/chatui/icon-192x192.png +0 -0
  474. package/src/ruvocal/static/chatui/icon-256x256.png +0 -0
  475. package/src/ruvocal/static/chatui/icon-36x36.png +0 -0
  476. package/src/ruvocal/static/chatui/icon-48x48.png +0 -0
  477. package/src/ruvocal/static/chatui/icon-512x512.png +0 -0
  478. package/src/ruvocal/static/chatui/icon-72x72.png +0 -0
  479. package/src/ruvocal/static/chatui/icon-96x96.png +0 -0
  480. package/src/ruvocal/static/chatui/icon.svg +3 -0
  481. package/src/ruvocal/static/chatui/logo.svg +7 -0
  482. package/src/ruvocal/static/chatui/manifest.json +54 -0
  483. package/src/ruvocal/static/chatui/omni-welcome.gif +0 -0
  484. package/src/ruvocal/static/chatui/omni-welcome.png +0 -0
  485. package/src/ruvocal/static/chatui/welcome.js +184 -0
  486. package/src/ruvocal/static/chatui/welcome.svg +1 -0
  487. package/src/ruvocal/static/huggingchat/apple-touch-icon.png +0 -0
  488. package/src/ruvocal/static/huggingchat/assistants-thumbnail.png +0 -0
  489. package/src/ruvocal/static/huggingchat/castle-example.jpg +0 -0
  490. package/src/ruvocal/static/huggingchat/favicon-dark.svg +4 -0
  491. package/src/ruvocal/static/huggingchat/favicon-dev.svg +4 -0
  492. package/src/ruvocal/static/huggingchat/favicon.ico +0 -0
  493. package/src/ruvocal/static/huggingchat/favicon.svg +4 -0
  494. package/src/ruvocal/static/huggingchat/fulltext-logo.svg +2 -0
  495. package/src/ruvocal/static/huggingchat/icon-128x128.png +0 -0
  496. package/src/ruvocal/static/huggingchat/icon-144x144.png +0 -0
  497. package/src/ruvocal/static/huggingchat/icon-192x192.png +0 -0
  498. package/src/ruvocal/static/huggingchat/icon-256x256.png +0 -0
  499. package/src/ruvocal/static/huggingchat/icon-36x36.png +0 -0
  500. package/src/ruvocal/static/huggingchat/icon-48x48.png +0 -0
  501. package/src/ruvocal/static/huggingchat/icon-512x512.png +0 -0
  502. package/src/ruvocal/static/huggingchat/icon-72x72.png +0 -0
  503. package/src/ruvocal/static/huggingchat/icon-96x96.png +0 -0
  504. package/src/ruvocal/static/huggingchat/icon.svg +4 -0
  505. package/src/ruvocal/static/huggingchat/logo.svg +4 -0
  506. package/src/ruvocal/static/huggingchat/manifest.json +54 -0
  507. package/src/ruvocal/static/huggingchat/omni-welcome.gif +0 -0
  508. package/src/ruvocal/static/huggingchat/routes.chat.json +226 -0
  509. package/src/ruvocal/static/huggingchat/thumbnail.png +0 -0
  510. package/src/ruvocal/static/huggingchat/tools-thumbnail.png +0 -0
  511. package/src/ruvocal/static/robots.txt +10 -0
  512. package/src/ruvocal/stub/@reflink/reflink/index.js +0 -0
  513. package/src/ruvocal/stub/@reflink/reflink/package.json +5 -0
  514. package/src/ruvocal/svelte.config.js +53 -0
  515. package/src/ruvocal/tailwind.config.cjs +30 -0
  516. package/src/ruvocal/tsconfig.json +19 -0
  517. package/src/ruvocal/vite.config.ts +87 -0
  518. package/src/scripts/deploy.sh +116 -0
  519. package/src/scripts/generate-config.js +245 -0
  520. package/src/scripts/generate-welcome.js +187 -0
  521. package/src/scripts/package-rvf.sh +116 -0
@@ -0,0 +1,72 @@
1
+ import { Address4, Address6 } from "ip-address";
2
+ import { isIP } from "node:net";
3
+
4
+ const UNSAFE_IPV4_SUBNETS = [
5
+ "0.0.0.0/8",
6
+ "100.64.0.0/10",
7
+ "127.0.0.0/8",
8
+ "169.254.0.0/16",
9
+ "172.16.0.0/12",
10
+ "192.168.0.0/16",
11
+ ].map((s) => new Address4(s));
12
+
13
+ function isUnsafeIp(address: string): boolean {
14
+ const family = isIP(address);
15
+
16
+ if (family === 4) {
17
+ const addr = new Address4(address);
18
+ return UNSAFE_IPV4_SUBNETS.some((subnet) => addr.isInSubnet(subnet));
19
+ }
20
+
21
+ if (family === 6) {
22
+ const addr = new Address6(address);
23
+ // Check IPv4-mapped IPv6 addresses (e.g. ::ffff:127.0.0.1)
24
+ if (addr.is4()) {
25
+ const v4 = addr.to4();
26
+ return UNSAFE_IPV4_SUBNETS.some((subnet) => v4.isInSubnet(subnet));
27
+ }
28
+ return addr.isLoopback() || addr.isLinkLocal();
29
+ }
30
+
31
+ return true; // Unknown format → block
32
+ }
33
+
34
+ /**
35
+ * Synchronous URL validation: checks protocol and hostname string.
36
+ */
37
+ export function isValidUrl(urlString: string): boolean {
38
+ try {
39
+ const url = new URL(urlString.trim());
40
+ const hostname = url.hostname.toLowerCase();
41
+ // Allow HTTP for localhost/loopback (dev & local MCP bridge)
42
+ if (hostname === "localhost" || hostname === "127.0.0.1" || hostname === "::1") {
43
+ return url.protocol === "http:" || url.protocol === "https:";
44
+ }
45
+ // Allow HTTP for Docker-internal service names (no dots = private network)
46
+ if (!hostname.includes(".") && url.protocol === "http:") {
47
+ return true;
48
+ }
49
+ if (url.protocol !== "https:") {
50
+ return false;
51
+ }
52
+ // If the hostname is a raw IP literal, validate it
53
+ const cleanHostname = hostname.replace(/^\[|]$/g, "");
54
+ if (isIP(cleanHostname)) {
55
+ return !isUnsafeIp(cleanHostname);
56
+ }
57
+ return true;
58
+ } catch {
59
+ return false;
60
+ }
61
+ }
62
+
63
+ /**
64
+ * Assert that a resolved IP address is safe (not internal/private).
65
+ * Throws if the IP is internal. Used in undici's custom DNS lookup
66
+ * to validate IPs at connection time (prevents TOCTOU DNS rebinding).
67
+ */
68
+ export function assertSafeIp(address: string, hostname: string): void {
69
+ if (isUnsafeIp(address)) {
70
+ throw new Error(`Resolved IP for ${hostname} is internal (${address})`);
71
+ }
72
+ }
@@ -0,0 +1,30 @@
1
+ import { z } from "zod";
2
+ import { config } from "$lib/server/config";
3
+ import JSON5 from "json5";
4
+
5
+ const sanitizeJSONEnv = (val: string, fallback: string) => {
6
+ const raw = (val ?? "").trim();
7
+ const unquoted = raw.startsWith("`") && raw.endsWith("`") ? raw.slice(1, -1) : raw;
8
+ return unquoted || fallback;
9
+ };
10
+
11
+ // RATE_LIMIT is the legacy way to define messages per minute limit
12
+ export const usageLimitsSchema = z
13
+ .object({
14
+ conversations: z.coerce.number().optional(), // how many conversations
15
+ messages: z.coerce.number().optional(), // how many messages in a conversation
16
+ messageLength: z.coerce.number().optional(), // how long can a message be before we cut it off
17
+ messagesPerMinute: z
18
+ .preprocess((val) => {
19
+ if (val === undefined) {
20
+ return config.RATE_LIMIT;
21
+ }
22
+ return val;
23
+ }, z.coerce.number().optional())
24
+ .optional(), // how many messages per minute
25
+ })
26
+ .optional();
27
+
28
+ export const usageLimits = usageLimitsSchema.parse(
29
+ JSON5.parse(sanitizeJSONEnv(config.USAGE_LIMITS, "{}"))
30
+ );
@@ -0,0 +1,175 @@
1
+ /**
2
+ * Autopilot Store — Svelte 5 runes-based store for managing autopilot Web Worker state.
3
+ *
4
+ * Provides reactive state for autopilot groups, tasks, and text content.
5
+ * Communicates with AutopilotWorker and DetailFetchWorker via postMessage.
6
+ *
7
+ * ADR-037 Part 2+3: Parallel Task UI + Web Workers
8
+ */
9
+
10
+ import type { GroupState, AutopilotUIUpdate } from "$lib/workers/autopilotWorker";
11
+ import type { DetailWorkerOutgoing } from "$lib/workers/detailFetchWorker";
12
+
13
+ export interface AutopilotState {
14
+ active: boolean;
15
+ maxSteps: number;
16
+ groups: GroupState[];
17
+ textContent: string;
18
+ error: string | null;
19
+ totalSteps: number;
20
+ totalTasks: number;
21
+ duration: number;
22
+ paused: boolean;
23
+ pauseReason: string | null;
24
+ }
25
+
26
+ const defaultState: AutopilotState = {
27
+ active: false,
28
+ maxSteps: 20,
29
+ groups: [],
30
+ textContent: "",
31
+ error: null,
32
+ totalSteps: 0,
33
+ totalTasks: 0,
34
+ duration: 0,
35
+ paused: false,
36
+ pauseReason: null,
37
+ };
38
+
39
+ let state = $state<AutopilotState>({ ...defaultState });
40
+
41
+ let autopilotWorker: Worker | null = null;
42
+ let detailWorker: Worker | null = null;
43
+ const detailCallbacks = new Map<string, (content: string | null, error?: string) => void>();
44
+
45
+ async function ensureWorkers() {
46
+ if (typeof window === "undefined") return;
47
+
48
+ if (!autopilotWorker) {
49
+ const mod = await import("$lib/workers/autopilotWorker?worker");
50
+ autopilotWorker = new mod.default();
51
+ autopilotWorker.onmessage = handleWorkerMessage;
52
+ }
53
+
54
+ if (!detailWorker) {
55
+ const mod = await import("$lib/workers/detailFetchWorker?worker");
56
+ detailWorker = new mod.default();
57
+ detailWorker.onmessage = handleDetailMessage;
58
+ }
59
+ }
60
+
61
+ function handleWorkerMessage(e: MessageEvent) {
62
+ const msg = e.data;
63
+
64
+ switch (msg.type) {
65
+ case "batch_update":
66
+ state.groups = msg.groups;
67
+ for (const update of msg.updates as AutopilotUIUpdate[]) {
68
+ applyUpdate(update);
69
+ }
70
+ break;
71
+
72
+ case "text":
73
+ state.textContent += msg.content;
74
+ break;
75
+
76
+ case "done":
77
+ state.active = false;
78
+ state.groups = msg.groups;
79
+ break;
80
+
81
+ case "error":
82
+ state.active = false;
83
+ state.error = msg.error;
84
+ break;
85
+
86
+ case "stopped":
87
+ state.active = false;
88
+ state.groups = msg.groups;
89
+ break;
90
+ }
91
+ }
92
+
93
+ function applyUpdate(update: AutopilotUIUpdate) {
94
+ switch (update.type) {
95
+ case "start":
96
+ state.maxSteps = update.maxSteps;
97
+ break;
98
+ case "end":
99
+ state.totalSteps = update.totalSteps;
100
+ state.totalTasks = update.totalTasks;
101
+ state.duration = update.duration;
102
+ break;
103
+ case "text":
104
+ state.textContent += update.content;
105
+ break;
106
+ case "paused":
107
+ state.paused = true;
108
+ state.pauseReason = update.reason;
109
+ break;
110
+ case "error_event":
111
+ state.error = update.error;
112
+ break;
113
+ }
114
+ }
115
+
116
+ function handleDetailMessage(e: MessageEvent<DetailWorkerOutgoing>) {
117
+ const msg = e.data;
118
+ if (msg.type === "detail") {
119
+ const cb = detailCallbacks.get(msg.detailToken);
120
+ if (cb) {
121
+ cb(msg.content);
122
+ detailCallbacks.delete(msg.detailToken);
123
+ }
124
+ } else if (msg.type === "detail_error") {
125
+ const cb = detailCallbacks.get(msg.detailToken);
126
+ if (cb) {
127
+ cb(null, msg.error);
128
+ detailCallbacks.delete(msg.detailToken);
129
+ }
130
+ }
131
+ }
132
+
133
+ export function useAutopilot() {
134
+ return {
135
+ get state() {
136
+ return state;
137
+ },
138
+
139
+ async start(url: string, headers: Record<string, string>, body: unknown) {
140
+ await ensureWorkers();
141
+ Object.assign(state, { ...defaultState, active: true });
142
+ autopilotWorker?.postMessage({ type: "start", url, headers, body });
143
+ },
144
+
145
+ stop() {
146
+ autopilotWorker?.postMessage({ type: "stop" });
147
+ },
148
+
149
+ async fetchDetail(detailToken: string, bridgeUrl: string): Promise<string> {
150
+ await ensureWorkers();
151
+ return new Promise((resolve, reject) => {
152
+ detailCallbacks.set(detailToken, (content, error) => {
153
+ if (error) reject(new Error(error));
154
+ else resolve(content!);
155
+ });
156
+ detailWorker?.postMessage({ type: "fetch", detailToken, bridgeUrl });
157
+ });
158
+ },
159
+
160
+ prefetchDetail(detailToken: string, bridgeUrl: string) {
161
+ detailWorker?.postMessage({ type: "prefetch", detailToken, bridgeUrl });
162
+ },
163
+
164
+ evictDetail(detailToken: string) {
165
+ detailWorker?.postMessage({ type: "evict", detailToken });
166
+ },
167
+
168
+ destroy() {
169
+ autopilotWorker?.terminate();
170
+ detailWorker?.terminate();
171
+ autopilotWorker = null;
172
+ detailWorker = null;
173
+ },
174
+ };
175
+ }
@@ -0,0 +1,32 @@
1
+ export type BackgroundGeneration = {
2
+ id: string;
3
+ startedAt: number;
4
+ };
5
+
6
+ export const backgroundGenerationEntries = $state<BackgroundGeneration[]>([]);
7
+
8
+ export function addBackgroundGeneration(entry: BackgroundGeneration) {
9
+ const index = backgroundGenerationEntries.findIndex(({ id }) => id === entry.id);
10
+
11
+ if (index === -1) {
12
+ backgroundGenerationEntries.push(entry);
13
+ return;
14
+ }
15
+
16
+ backgroundGenerationEntries[index] = entry;
17
+ }
18
+
19
+ export function removeBackgroundGeneration(id: string) {
20
+ const index = backgroundGenerationEntries.findIndex((entry) => entry.id === id);
21
+ if (index === -1) return;
22
+
23
+ backgroundGenerationEntries.splice(index, 1);
24
+ }
25
+
26
+ export function clearBackgroundGenerations() {
27
+ backgroundGenerationEntries.length = 0;
28
+ }
29
+
30
+ export function hasBackgroundGeneration(id: string) {
31
+ return backgroundGenerationEntries.some((entry) => entry.id === id);
32
+ }
@@ -0,0 +1 @@
1
+ export * from "./backgroundGenerations.svelte";
@@ -0,0 +1,9 @@
1
+ import { writable } from "svelte/store";
2
+
3
+ export const ERROR_MESSAGES = {
4
+ default: "Oops, something went wrong.",
5
+ authOnly: "You have to be logged in.",
6
+ rateLimited: "You are sending too many messages. Try again later.",
7
+ };
8
+
9
+ export const error = writable<string | undefined>(undefined);
@@ -0,0 +1,3 @@
1
+ import { writable } from "svelte/store";
2
+
3
+ export const isAborted = writable<boolean>(false);
@@ -0,0 +1,4 @@
1
+ import { writable } from "svelte/store";
2
+
3
+ // null = unknown/loading, true = PRO, false = not PRO
4
+ export const isPro = writable<boolean | null>(null);
@@ -0,0 +1,3 @@
1
+ import { writable } from "svelte/store";
2
+
3
+ export const loading = writable(false);
@@ -0,0 +1,345 @@
1
+ /**
2
+ * MCP Servers Store
3
+ * Manages base (env-configured) and custom (user-added) MCP servers
4
+ * Stores custom servers and selection state in browser localStorage
5
+ */
6
+
7
+ import { writable, derived, get } from "svelte/store";
8
+ import { base } from "$app/paths";
9
+ import { env as publicEnv } from "$env/dynamic/public";
10
+ import { browser } from "$app/environment";
11
+ import type { MCPServer, ServerStatus, MCPTool } from "$lib/types/Tool";
12
+
13
+ // Namespace storage by app identity to avoid collisions across apps
14
+ function toKeyPart(s: string | undefined): string {
15
+ return (s || "").toLowerCase().replace(/[^a-z0-9_-]+/g, "-");
16
+ }
17
+
18
+ const appLabel = toKeyPart(publicEnv.PUBLIC_APP_ASSETS || publicEnv.PUBLIC_APP_NAME);
19
+ const baseLabel = toKeyPart(typeof base === "string" ? base : "");
20
+ // Final prefix format requested: "huggingchat:key" (no mcp:/chat)
21
+ const KEY_PREFIX = appLabel || baseLabel || "app";
22
+
23
+ const STORAGE_KEYS = {
24
+ CUSTOM_SERVERS: `${KEY_PREFIX}:mcp:custom-servers`,
25
+ SELECTED_IDS: `${KEY_PREFIX}:mcp:selected-ids`,
26
+ DISABLED_BASE_IDS: `${KEY_PREFIX}:mcp:disabled-base-ids`,
27
+ } as const;
28
+
29
+ // No migration needed per request — read/write only namespaced keys
30
+
31
+ // Load custom servers from localStorage
32
+ function loadCustomServers(): MCPServer[] {
33
+ if (!browser) return [];
34
+
35
+ try {
36
+ const json = localStorage.getItem(STORAGE_KEYS.CUSTOM_SERVERS);
37
+ return json ? JSON.parse(json) : [];
38
+ } catch (error) {
39
+ console.error("Failed to load custom MCP servers from localStorage:", error);
40
+ return [];
41
+ }
42
+ }
43
+
44
+ // Load selected server IDs from localStorage
45
+ function loadSelectedIds(): Set<string> {
46
+ if (!browser) return new Set();
47
+
48
+ try {
49
+ const json = localStorage.getItem(STORAGE_KEYS.SELECTED_IDS);
50
+ const ids: string[] = json ? JSON.parse(json) : [];
51
+ return new Set(ids);
52
+ } catch (error) {
53
+ console.error("Failed to load selected MCP server IDs from localStorage:", error);
54
+ return new Set();
55
+ }
56
+ }
57
+
58
+ // Save custom servers to localStorage
59
+ function saveCustomServers(servers: MCPServer[]) {
60
+ if (!browser) return;
61
+
62
+ try {
63
+ localStorage.setItem(STORAGE_KEYS.CUSTOM_SERVERS, JSON.stringify(servers));
64
+ } catch (error) {
65
+ console.error("Failed to save custom MCP servers to localStorage:", error);
66
+ }
67
+ }
68
+
69
+ // Save selected IDs to localStorage
70
+ function saveSelectedIds(ids: Set<string>) {
71
+ if (!browser) return;
72
+
73
+ try {
74
+ localStorage.setItem(STORAGE_KEYS.SELECTED_IDS, JSON.stringify([...ids]));
75
+ } catch (error) {
76
+ console.error("Failed to save selected MCP server IDs to localStorage:", error);
77
+ }
78
+ }
79
+
80
+ // Load disabled base server IDs from localStorage (empty set if missing or on error)
81
+ function loadDisabledBaseIds(): Set<string> {
82
+ if (!browser) return new Set();
83
+
84
+ try {
85
+ const json = localStorage.getItem(STORAGE_KEYS.DISABLED_BASE_IDS);
86
+ return new Set(json ? JSON.parse(json) : []);
87
+ } catch (error) {
88
+ console.error("Failed to load disabled base MCP server IDs from localStorage:", error);
89
+ return new Set();
90
+ }
91
+ }
92
+
93
+ // Save disabled base server IDs to localStorage
94
+ function saveDisabledBaseIds(ids: Set<string>) {
95
+ if (!browser) return;
96
+
97
+ try {
98
+ localStorage.setItem(STORAGE_KEYS.DISABLED_BASE_IDS, JSON.stringify([...ids]));
99
+ } catch (error) {
100
+ console.error("Failed to save disabled base MCP server IDs to localStorage:", error);
101
+ }
102
+ }
103
+
104
+ // Store for all servers (base + custom)
105
+ export const allMcpServers = writable<MCPServer[]>([]);
106
+
107
+ // Track if initial server load has completed
108
+ export const mcpServersLoaded = writable<boolean>(false);
109
+
110
+ // Store for selected server IDs
111
+ export const selectedServerIds = writable<Set<string>>(loadSelectedIds());
112
+
113
+ // Auto-persist selected IDs when they change
114
+ if (browser) {
115
+ selectedServerIds.subscribe((ids) => {
116
+ saveSelectedIds(ids);
117
+ });
118
+ }
119
+
120
+ // Derived store: only enabled servers
121
+ export const enabledServers = derived([allMcpServers, selectedServerIds], ([$all, $selected]) =>
122
+ $all.filter((s) => $selected.has(s.id))
123
+ );
124
+
125
+ // Derived store: count of enabled servers
126
+ export const enabledServersCount = derived(enabledServers, ($enabled) => $enabled.length);
127
+
128
+ // Derived store: true if all base servers are enabled
129
+ export const allBaseServersEnabled = derived(
130
+ [allMcpServers, selectedServerIds],
131
+ ([$all, $selected]) => {
132
+ const baseServers = $all.filter((s) => s.type === "base");
133
+ return baseServers.length > 0 && baseServers.every((s) => $selected.has(s.id));
134
+ }
135
+ );
136
+
137
+ // Note: Authorization overlay (with user's HF token) for the Hugging Face MCP host
138
+ // is applied server-side when enabled via MCP_FORWARD_HF_USER_TOKEN.
139
+
140
+ /**
141
+ * Refresh base servers from API and merge with custom servers
142
+ */
143
+ export async function refreshMcpServers() {
144
+ try {
145
+ const response = await fetch(`${base}/api/mcp/servers`);
146
+ if (!response.ok) {
147
+ throw new Error(`Failed to fetch base servers: ${response.statusText}`);
148
+ }
149
+
150
+ const baseServers: MCPServer[] = await response.json();
151
+ const customServers = loadCustomServers();
152
+
153
+ // Merge base and custom servers
154
+ const merged = [...baseServers, ...customServers];
155
+ allMcpServers.set(merged);
156
+
157
+ // Load disabled base servers
158
+ const disabledBaseIds = loadDisabledBaseIds();
159
+
160
+ // Auto-enable all base servers that aren't explicitly disabled
161
+ // Plus keep any custom servers that were previously selected
162
+ const validIds = new Set(merged.map((s) => s.id));
163
+ selectedServerIds.update(($currentIds) => {
164
+ const newSelection = new Set<string>();
165
+
166
+ // Add all base servers that aren't disabled
167
+ for (const server of baseServers) {
168
+ if (!disabledBaseIds.has(server.id)) {
169
+ newSelection.add(server.id);
170
+ }
171
+ }
172
+
173
+ // Keep custom servers that were selected and still exist
174
+ for (const id of $currentIds) {
175
+ if (validIds.has(id) && !id.startsWith("base-")) {
176
+ newSelection.add(id);
177
+ }
178
+ }
179
+
180
+ return newSelection;
181
+ });
182
+ mcpServersLoaded.set(true);
183
+ } catch (error) {
184
+ console.error("Failed to refresh MCP servers:", error);
185
+ // On error, just use custom servers
186
+ allMcpServers.set(loadCustomServers());
187
+ mcpServersLoaded.set(true);
188
+ }
189
+ }
190
+
191
+ /**
192
+ * Toggle a server on/off
193
+ */
194
+ export function toggleServer(id: string) {
195
+ selectedServerIds.update(($ids) => {
196
+ const newSet = new Set($ids);
197
+ if (newSet.has(id)) {
198
+ newSet.delete(id);
199
+ // Track if this is a base server being disabled
200
+ if (id.startsWith("base-")) {
201
+ const disabled = loadDisabledBaseIds();
202
+ disabled.add(id);
203
+ saveDisabledBaseIds(disabled);
204
+ }
205
+ } else {
206
+ newSet.add(id);
207
+ // Remove from disabled if re-enabling a base server
208
+ if (id.startsWith("base-")) {
209
+ const disabled = loadDisabledBaseIds();
210
+ disabled.delete(id);
211
+ saveDisabledBaseIds(disabled);
212
+ }
213
+ }
214
+ return newSet;
215
+ });
216
+ }
217
+
218
+ /**
219
+ * Disable all MCP servers (marks all base servers as disabled)
220
+ */
221
+ export function disableAllServers() {
222
+ // Get current base server IDs and mark them all as disabled
223
+ const servers = get(allMcpServers);
224
+ const baseServerIds = servers.filter((s) => s.type === "base").map((s) => s.id);
225
+
226
+ // Save all base servers as disabled
227
+ saveDisabledBaseIds(new Set(baseServerIds));
228
+
229
+ // Clear the selection
230
+ selectedServerIds.set(new Set());
231
+ }
232
+
233
+ /**
234
+ * Add a custom MCP server
235
+ */
236
+ export function addCustomServer(server: Omit<MCPServer, "id" | "type" | "status">): string {
237
+ const newServer: MCPServer = {
238
+ ...server,
239
+ id: crypto.randomUUID(),
240
+ type: "custom",
241
+ status: "disconnected",
242
+ };
243
+
244
+ const customServers = loadCustomServers();
245
+ customServers.push(newServer);
246
+ saveCustomServers(customServers);
247
+
248
+ // Refresh all servers to include the new one
249
+ refreshMcpServers();
250
+
251
+ return newServer.id;
252
+ }
253
+
254
+ /**
255
+ * Update an existing custom server
256
+ */
257
+ export function updateCustomServer(id: string, updates: Partial<MCPServer>) {
258
+ const customServers = loadCustomServers();
259
+ const index = customServers.findIndex((s) => s.id === id);
260
+
261
+ if (index !== -1) {
262
+ customServers[index] = { ...customServers[index], ...updates };
263
+ saveCustomServers(customServers);
264
+ refreshMcpServers();
265
+ }
266
+ }
267
+
268
+ /**
269
+ * Delete a custom server
270
+ */
271
+ export function deleteCustomServer(id: string) {
272
+ const customServers = loadCustomServers();
273
+ const filtered = customServers.filter((s) => s.id !== id);
274
+ saveCustomServers(filtered);
275
+
276
+ // Also remove from selected IDs
277
+ selectedServerIds.update(($ids) => {
278
+ const newSet = new Set($ids);
279
+ newSet.delete(id);
280
+ return newSet;
281
+ });
282
+
283
+ refreshMcpServers();
284
+ }
285
+
286
+ /**
287
+ * Update server status (from health check)
288
+ */
289
+ export function updateServerStatus(
290
+ id: string,
291
+ status: ServerStatus,
292
+ errorMessage?: string,
293
+ tools?: MCPTool[],
294
+ authRequired?: boolean
295
+ ) {
296
+ allMcpServers.update(($servers) =>
297
+ $servers.map((s) =>
298
+ s.id === id
299
+ ? {
300
+ ...s,
301
+ status,
302
+ errorMessage,
303
+ tools,
304
+ authRequired,
305
+ }
306
+ : s
307
+ )
308
+ );
309
+ }
310
+
311
+ /**
312
+ * Run health check on a server
313
+ */
314
+ export async function healthCheckServer(
315
+ server: MCPServer
316
+ ): Promise<{ ready: boolean; tools?: MCPTool[]; error?: string }> {
317
+ try {
318
+ updateServerStatus(server.id, "connecting");
319
+
320
+ const response = await fetch(`${base}/api/mcp/health`, {
321
+ method: "POST",
322
+ headers: { "Content-Type": "application/json" },
323
+ body: JSON.stringify({ url: server.url, headers: server.headers }),
324
+ });
325
+
326
+ const result = await response.json();
327
+
328
+ if (result.ready && result.tools) {
329
+ updateServerStatus(server.id, "connected", undefined, result.tools, false);
330
+ return { ready: true, tools: result.tools };
331
+ } else {
332
+ updateServerStatus(server.id, "error", result.error, undefined, Boolean(result.authRequired));
333
+ return { ready: false, error: result.error };
334
+ }
335
+ } catch (error) {
336
+ const errorMessage = error instanceof Error ? error.message : "Unknown error";
337
+ updateServerStatus(server.id, "error", errorMessage);
338
+ return { ready: false, error: errorMessage };
339
+ }
340
+ }
341
+
342
+ // Initialize on module load
343
+ if (browser) {
344
+ refreshMcpServers();
345
+ }
@@ -0,0 +1,3 @@
1
+ import { writable } from "svelte/store";
2
+
3
+ export const pendingChatInput = writable<string | undefined>(undefined);
@@ -0,0 +1,9 @@
1
+ import { writable } from "svelte/store";
2
+
3
+ export const pendingMessage = writable<
4
+ | {
5
+ content: string;
6
+ files: File[];
7
+ }
8
+ | undefined
9
+ >();