ruflo 3.10.35 → 3.10.37

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 +416 -416
  2. package/bin/ruflo.js +77 -77
  3. package/package.json +113 -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/.devcontainer/Dockerfile +9 -9
  22. package/src/ruvocal/.devcontainer/devcontainer.json +36 -36
  23. package/src/ruvocal/.dockerignore +16 -16
  24. package/src/ruvocal/.eslintignore +13 -13
  25. package/src/ruvocal/.eslintrc.cjs +45 -45
  26. package/src/ruvocal/.gcloudignore +18 -18
  27. package/src/ruvocal/.github/ISSUE_TEMPLATE/bug-report--chat-ui-.md +43 -43
  28. package/src/ruvocal/.github/ISSUE_TEMPLATE/config-support.md +9 -9
  29. package/src/ruvocal/.github/ISSUE_TEMPLATE/feature-request--chat-ui-.md +17 -17
  30. package/src/ruvocal/.github/ISSUE_TEMPLATE/huggingchat.md +11 -11
  31. package/src/ruvocal/.github/release.yml +16 -16
  32. package/src/ruvocal/.github/workflows/build-docs.yml +18 -18
  33. package/src/ruvocal/.github/workflows/build-image.yml +142 -142
  34. package/src/ruvocal/.github/workflows/build-pr-docs.yml +20 -20
  35. package/src/ruvocal/.github/workflows/deploy-dev.yml +63 -63
  36. package/src/ruvocal/.github/workflows/deploy-prod.yml +78 -78
  37. package/src/ruvocal/.github/workflows/lint-and-test.yml +84 -84
  38. package/src/ruvocal/.github/workflows/slugify.yaml +72 -72
  39. package/src/ruvocal/.github/workflows/trufflehog.yml +17 -17
  40. package/src/ruvocal/.github/workflows/upload-pr-documentation.yml +16 -16
  41. package/src/ruvocal/.husky/lint-stage-config.js +4 -4
  42. package/src/ruvocal/.husky/pre-commit +2 -2
  43. package/src/ruvocal/.prettierignore +14 -14
  44. package/src/ruvocal/.prettierrc +7 -7
  45. package/src/ruvocal/CLAUDE.md +126 -126
  46. package/src/ruvocal/Dockerfile +96 -96
  47. package/src/ruvocal/LICENSE +202 -202
  48. package/src/ruvocal/PRIVACY.md +41 -41
  49. package/src/ruvocal/README.md +164 -164
  50. package/src/ruvocal/chart/Chart.yaml +5 -5
  51. package/src/ruvocal/chart/env/dev.yaml +260 -260
  52. package/src/ruvocal/chart/env/prod.yaml +273 -273
  53. package/src/ruvocal/chart/templates/_helpers.tpl +22 -22
  54. package/src/ruvocal/chart/templates/config.yaml +10 -10
  55. package/src/ruvocal/chart/templates/deployment.yaml +81 -81
  56. package/src/ruvocal/chart/templates/hpa.yaml +45 -45
  57. package/src/ruvocal/chart/templates/infisical.yaml +24 -24
  58. package/src/ruvocal/chart/templates/ingress-internal.yaml +32 -32
  59. package/src/ruvocal/chart/templates/ingress.yaml +32 -32
  60. package/src/ruvocal/chart/templates/network-policy.yaml +36 -36
  61. package/src/ruvocal/chart/templates/service-account.yaml +13 -13
  62. package/src/ruvocal/chart/templates/service-monitor.yaml +17 -17
  63. package/src/ruvocal/chart/templates/service.yaml +21 -21
  64. package/src/ruvocal/chart/values.yaml +73 -73
  65. package/src/ruvocal/cloudbuild.yaml +68 -68
  66. package/src/ruvocal/config/branding.env.example +19 -19
  67. package/src/ruvocal/docker-compose.yml +21 -21
  68. package/src/ruvocal/docs/adr/ADR-029-HUGGINGFACE-CHAT-UI-CLOUD-RUN.md +1236 -1236
  69. package/src/ruvocal/docs/adr/ADR-033-RUVECTOR-RUFLO-MCP-INTEGRATION.md +111 -111
  70. package/src/ruvocal/docs/adr/ADR-034-OPTIONAL-MCP-BACKENDS.md +117 -117
  71. package/src/ruvocal/docs/adr/ADR-035-MCP-TOOL-GROUPS.md +186 -186
  72. package/src/ruvocal/docs/adr/ADR-037-AUTOPILOT-CHAT-MODE.md +1500 -1500
  73. package/src/ruvocal/docs/adr/ADR-038-RUVOCAL-FORK.md +286 -286
  74. package/src/ruvocal/docs/source/_toctree.yml +30 -30
  75. package/src/ruvocal/docs/source/configuration/common-issues.md +38 -38
  76. package/src/ruvocal/docs/source/configuration/llm-router.md +105 -105
  77. package/src/ruvocal/docs/source/configuration/mcp-tools.md +84 -84
  78. package/src/ruvocal/docs/source/configuration/metrics.md +9 -9
  79. package/src/ruvocal/docs/source/configuration/open-id.md +57 -57
  80. package/src/ruvocal/docs/source/configuration/overview.md +89 -89
  81. package/src/ruvocal/docs/source/configuration/theming.md +20 -20
  82. package/src/ruvocal/docs/source/developing/architecture.md +48 -48
  83. package/src/ruvocal/docs/source/index.md +53 -53
  84. package/src/ruvocal/docs/source/installation/docker.md +43 -43
  85. package/src/ruvocal/docs/source/installation/helm.md +43 -43
  86. package/src/ruvocal/docs/source/installation/local.md +62 -62
  87. package/src/ruvocal/entrypoint.sh +18 -18
  88. package/src/ruvocal/mcp-bridge/Dockerfile +45 -45
  89. package/src/ruvocal/mcp-bridge/cloudbuild.yaml +49 -49
  90. package/src/ruvocal/mcp-bridge/index.js +1902 -1902
  91. package/src/ruvocal/mcp-bridge/mcp-stdio-kernel.js +159 -159
  92. package/src/ruvocal/mcp-bridge/package-lock.json +762 -762
  93. package/src/ruvocal/mcp-bridge/package.json +17 -17
  94. package/src/ruvocal/mcp-bridge/test-harness.js +470 -470
  95. package/src/ruvocal/package-lock.json +11741 -11741
  96. package/src/ruvocal/package.json +121 -121
  97. package/src/ruvocal/postcss.config.js +6 -6
  98. package/src/ruvocal/rvf.manifest.json +204 -204
  99. package/src/ruvocal/scripts/config.ts +64 -64
  100. package/src/ruvocal/scripts/generate-welcome.mjs +181 -181
  101. package/src/ruvocal/scripts/populate.ts +288 -288
  102. package/src/ruvocal/scripts/samples.txt +194 -194
  103. package/src/ruvocal/scripts/setups/vitest-setup-server.ts +44 -44
  104. package/src/ruvocal/scripts/updateLocalEnv.ts +48 -48
  105. package/src/ruvocal/src/ambient.d.ts +7 -7
  106. package/src/ruvocal/src/app.d.ts +29 -29
  107. package/src/ruvocal/src/app.html +53 -53
  108. package/src/ruvocal/src/hooks.server.ts +32 -32
  109. package/src/ruvocal/src/hooks.ts +6 -6
  110. package/src/ruvocal/src/lib/APIClient.ts +148 -148
  111. package/src/ruvocal/src/lib/actions/clickOutside.ts +18 -18
  112. package/src/ruvocal/src/lib/actions/snapScrollToBottom.ts +346 -346
  113. package/src/ruvocal/src/lib/buildPrompt.ts +33 -33
  114. package/src/ruvocal/src/lib/components/AnnouncementBanner.svelte +20 -20
  115. package/src/ruvocal/src/lib/components/BackgroundGenerationPoller.svelte +168 -168
  116. package/src/ruvocal/src/lib/components/CodeBlock.svelte +73 -73
  117. package/src/ruvocal/src/lib/components/CopyToClipBoardBtn.svelte +92 -92
  118. package/src/ruvocal/src/lib/components/DeleteConversationModal.svelte +75 -75
  119. package/src/ruvocal/src/lib/components/EditConversationModal.svelte +100 -100
  120. package/src/ruvocal/src/lib/components/ExpandNavigation.svelte +22 -22
  121. package/src/ruvocal/src/lib/components/FoundationBackground.svelte +242 -242
  122. package/src/ruvocal/src/lib/components/HoverTooltip.svelte +44 -44
  123. package/src/ruvocal/src/lib/components/HtmlPreviewModal.svelte +143 -143
  124. package/src/ruvocal/src/lib/components/InfiniteScroll.svelte +50 -50
  125. package/src/ruvocal/src/lib/components/MobileNav.svelte +300 -300
  126. package/src/ruvocal/src/lib/components/Modal.svelte +115 -115
  127. package/src/ruvocal/src/lib/components/ModelCardMetadata.svelte +71 -71
  128. package/src/ruvocal/src/lib/components/NavConversationItem.svelte +151 -151
  129. package/src/ruvocal/src/lib/components/NavMenu.svelte +313 -313
  130. package/src/ruvocal/src/lib/components/Pagination.svelte +97 -97
  131. package/src/ruvocal/src/lib/components/PaginationArrow.svelte +27 -27
  132. package/src/ruvocal/src/lib/components/Portal.svelte +24 -24
  133. package/src/ruvocal/src/lib/components/RetryBtn.svelte +18 -18
  134. package/src/ruvocal/src/lib/components/RuFloUniverse.svelte +185 -185
  135. package/src/ruvocal/src/lib/components/RufloHelpModal.svelte +411 -411
  136. package/src/ruvocal/src/lib/components/ScrollToBottomBtn.svelte +47 -47
  137. package/src/ruvocal/src/lib/components/ScrollToPreviousBtn.svelte +77 -77
  138. package/src/ruvocal/src/lib/components/ShareConversationModal.svelte +182 -182
  139. package/src/ruvocal/src/lib/components/StopGeneratingBtn.svelte +69 -69
  140. package/src/ruvocal/src/lib/components/SubscribeModal.svelte +87 -87
  141. package/src/ruvocal/src/lib/components/Switch.svelte +36 -36
  142. package/src/ruvocal/src/lib/components/SystemPromptModal.svelte +44 -44
  143. package/src/ruvocal/src/lib/components/Toast.svelte +27 -27
  144. package/src/ruvocal/src/lib/components/Tooltip.svelte +30 -30
  145. package/src/ruvocal/src/lib/components/WelcomeModal.svelte +46 -46
  146. package/src/ruvocal/src/lib/components/chat/Alternatives.svelte +77 -77
  147. package/src/ruvocal/src/lib/components/chat/BlockWrapper.svelte +72 -72
  148. package/src/ruvocal/src/lib/components/chat/ChatInput.svelte +490 -490
  149. package/src/ruvocal/src/lib/components/chat/ChatIntroduction.svelte +123 -123
  150. package/src/ruvocal/src/lib/components/chat/ChatMessage.svelte +548 -548
  151. package/src/ruvocal/src/lib/components/chat/ChatWindow.svelte +1057 -1057
  152. package/src/ruvocal/src/lib/components/chat/FileDropzone.svelte +92 -92
  153. package/src/ruvocal/src/lib/components/chat/ImageLightbox.svelte +66 -66
  154. package/src/ruvocal/src/lib/components/chat/MarkdownBlock.svelte +23 -23
  155. package/src/ruvocal/src/lib/components/chat/MarkdownRenderer.svelte +69 -69
  156. package/src/ruvocal/src/lib/components/chat/MarkdownRenderer.svelte.test.ts +58 -58
  157. package/src/ruvocal/src/lib/components/chat/MessageAvatar.svelte +103 -103
  158. package/src/ruvocal/src/lib/components/chat/ModelSwitch.svelte +64 -64
  159. package/src/ruvocal/src/lib/components/chat/OpenReasoningResults.svelte +81 -81
  160. package/src/ruvocal/src/lib/components/chat/TaskGroup.svelte +88 -88
  161. package/src/ruvocal/src/lib/components/chat/ToolUpdate.svelte +273 -273
  162. package/src/ruvocal/src/lib/components/chat/UploadedFile.svelte +253 -253
  163. package/src/ruvocal/src/lib/components/chat/UrlFetchModal.svelte +203 -203
  164. package/src/ruvocal/src/lib/components/chat/VoiceRecorder.svelte +214 -214
  165. package/src/ruvocal/src/lib/components/icons/IconBurger.svelte +20 -20
  166. package/src/ruvocal/src/lib/components/icons/IconCheap.svelte +20 -20
  167. package/src/ruvocal/src/lib/components/icons/IconChevron.svelte +24 -24
  168. package/src/ruvocal/src/lib/components/icons/IconDazzled.svelte +40 -40
  169. package/src/ruvocal/src/lib/components/icons/IconFast.svelte +20 -20
  170. package/src/ruvocal/src/lib/components/icons/IconLoading.svelte +22 -22
  171. package/src/ruvocal/src/lib/components/icons/IconMCP.svelte +28 -28
  172. package/src/ruvocal/src/lib/components/icons/IconMoon.svelte +21 -21
  173. package/src/ruvocal/src/lib/components/icons/IconNew.svelte +20 -20
  174. package/src/ruvocal/src/lib/components/icons/IconOmni.svelte +90 -90
  175. package/src/ruvocal/src/lib/components/icons/IconPaperclip.svelte +24 -24
  176. package/src/ruvocal/src/lib/components/icons/IconPro.svelte +37 -37
  177. package/src/ruvocal/src/lib/components/icons/IconShare.svelte +21 -21
  178. package/src/ruvocal/src/lib/components/icons/IconSun.svelte +93 -93
  179. package/src/ruvocal/src/lib/components/icons/Logo.svelte +68 -68
  180. package/src/ruvocal/src/lib/components/icons/LogoHuggingFaceBorderless.svelte +54 -54
  181. package/src/ruvocal/src/lib/components/mcp/AddServerForm.svelte +250 -250
  182. package/src/ruvocal/src/lib/components/mcp/MCPServerManager.svelte +185 -185
  183. package/src/ruvocal/src/lib/components/mcp/ServerCard.svelte +203 -203
  184. package/src/ruvocal/src/lib/components/players/AudioPlayer.svelte +82 -82
  185. package/src/ruvocal/src/lib/components/voice/AudioWaveform.svelte +96 -96
  186. package/src/ruvocal/src/lib/components/wasm/GalleryPanel.svelte +357 -357
  187. package/src/ruvocal/src/lib/constants/mcpExamples.ts +114 -114
  188. package/src/ruvocal/src/lib/constants/mime.ts +11 -11
  189. package/src/ruvocal/src/lib/constants/pagination.ts +1 -1
  190. package/src/ruvocal/src/lib/constants/publicSepToken.ts +1 -1
  191. package/src/ruvocal/src/lib/constants/routerExamples.ts +133 -133
  192. package/src/ruvocal/src/lib/constants/rvagentPresets.ts +206 -206
  193. package/src/ruvocal/src/lib/createShareLink.ts +27 -27
  194. package/src/ruvocal/src/lib/jobs/refresh-conversation-stats.ts +297 -297
  195. package/src/ruvocal/src/lib/migrations/lock.ts +56 -56
  196. package/src/ruvocal/src/lib/migrations/migrations.spec.ts +74 -74
  197. package/src/ruvocal/src/lib/migrations/migrations.ts +109 -109
  198. package/src/ruvocal/src/lib/migrations/routines/01-update-search-assistants.ts +50 -50
  199. package/src/ruvocal/src/lib/migrations/routines/02-update-assistants-models.ts +48 -48
  200. package/src/ruvocal/src/lib/migrations/routines/04-update-message-updates.ts +151 -151
  201. package/src/ruvocal/src/lib/migrations/routines/05-update-message-files.ts +56 -56
  202. package/src/ruvocal/src/lib/migrations/routines/06-trim-message-updates.ts +56 -56
  203. package/src/ruvocal/src/lib/migrations/routines/08-update-featured-to-review.ts +32 -32
  204. package/src/ruvocal/src/lib/migrations/routines/09-delete-empty-conversations.spec.ts +214 -214
  205. package/src/ruvocal/src/lib/migrations/routines/09-delete-empty-conversations.ts +88 -88
  206. package/src/ruvocal/src/lib/migrations/routines/10-update-reports-assistantid.ts +29 -29
  207. package/src/ruvocal/src/lib/migrations/routines/index.ts +15 -15
  208. package/src/ruvocal/src/lib/server/__tests__/conversation-stop-generating.spec.ts +103 -103
  209. package/src/ruvocal/src/lib/server/abortRegistry.ts +57 -57
  210. package/src/ruvocal/src/lib/server/abortedGenerations.ts +43 -43
  211. package/src/ruvocal/src/lib/server/adminToken.ts +62 -62
  212. package/src/ruvocal/src/lib/server/api/__tests__/conversations-id.spec.ts +296 -296
  213. package/src/ruvocal/src/lib/server/api/__tests__/conversations-message.spec.ts +216 -216
  214. package/src/ruvocal/src/lib/server/api/__tests__/conversations.spec.ts +235 -235
  215. package/src/ruvocal/src/lib/server/api/__tests__/misc.spec.ts +72 -72
  216. package/src/ruvocal/src/lib/server/api/__tests__/testHelpers.ts +86 -86
  217. package/src/ruvocal/src/lib/server/api/__tests__/user-reports.spec.ts +78 -78
  218. package/src/ruvocal/src/lib/server/api/__tests__/user.spec.ts +239 -239
  219. package/src/ruvocal/src/lib/server/api/types.ts +37 -37
  220. package/src/ruvocal/src/lib/server/api/utils/requireAuth.ts +22 -22
  221. package/src/ruvocal/src/lib/server/api/utils/resolveConversation.ts +69 -69
  222. package/src/ruvocal/src/lib/server/api/utils/resolveModel.ts +27 -27
  223. package/src/ruvocal/src/lib/server/api/utils/superjsonResponse.ts +15 -15
  224. package/src/ruvocal/src/lib/server/apiToken.ts +11 -11
  225. package/src/ruvocal/src/lib/server/auth.ts +554 -554
  226. package/src/ruvocal/src/lib/server/config.ts +187 -187
  227. package/src/ruvocal/src/lib/server/conversation.ts +83 -83
  228. package/src/ruvocal/src/lib/server/database/__tests__/rvf.spec.ts +709 -709
  229. package/src/ruvocal/src/lib/server/database/postgres.ts +700 -700
  230. package/src/ruvocal/src/lib/server/database/rvf.ts +1078 -1078
  231. package/src/ruvocal/src/lib/server/database.ts +145 -145
  232. package/src/ruvocal/src/lib/server/endpoints/document.ts +68 -68
  233. package/src/ruvocal/src/lib/server/endpoints/endpoints.ts +43 -43
  234. package/src/ruvocal/src/lib/server/endpoints/images.ts +211 -211
  235. package/src/ruvocal/src/lib/server/endpoints/openai/endpointOai.ts +266 -266
  236. package/src/ruvocal/src/lib/server/endpoints/openai/openAIChatToTextGenerationStream.ts +212 -212
  237. package/src/ruvocal/src/lib/server/endpoints/openai/openAICompletionToTextGenerationStream.ts +32 -32
  238. package/src/ruvocal/src/lib/server/endpoints/preprocessMessages.ts +61 -61
  239. package/src/ruvocal/src/lib/server/exitHandler.ts +59 -59
  240. package/src/ruvocal/src/lib/server/files/downloadFile.ts +34 -34
  241. package/src/ruvocal/src/lib/server/files/uploadFile.ts +29 -29
  242. package/src/ruvocal/src/lib/server/findRepoRoot.ts +13 -13
  243. package/src/ruvocal/src/lib/server/generateFromDefaultEndpoint.ts +46 -46
  244. package/src/ruvocal/src/lib/server/hooks/error.ts +37 -37
  245. package/src/ruvocal/src/lib/server/hooks/fetch.ts +22 -22
  246. package/src/ruvocal/src/lib/server/hooks/handle.ts +250 -250
  247. package/src/ruvocal/src/lib/server/hooks/init.ts +51 -51
  248. package/src/ruvocal/src/lib/server/isURLLocal.spec.ts +31 -31
  249. package/src/ruvocal/src/lib/server/isURLLocal.ts +74 -74
  250. package/src/ruvocal/src/lib/server/logger.ts +42 -42
  251. package/src/ruvocal/src/lib/server/mcp/clientPool.spec.ts +175 -175
  252. package/src/ruvocal/src/lib/server/mcp/hf.ts +32 -32
  253. package/src/ruvocal/src/lib/server/mcp/httpClient.ts +122 -122
  254. package/src/ruvocal/src/lib/server/mcp/registry.ts +76 -76
  255. package/src/ruvocal/src/lib/server/mcp/tools.ts +196 -196
  256. package/src/ruvocal/src/lib/server/metrics.ts +255 -255
  257. package/src/ruvocal/src/lib/server/models.ts +518 -518
  258. package/src/ruvocal/src/lib/server/requestContext.ts +55 -55
  259. package/src/ruvocal/src/lib/server/router/arch.ts +230 -230
  260. package/src/ruvocal/src/lib/server/router/endpoint.ts +316 -316
  261. package/src/ruvocal/src/lib/server/router/multimodal.ts +28 -28
  262. package/src/ruvocal/src/lib/server/router/policy.ts +49 -49
  263. package/src/ruvocal/src/lib/server/router/toolsRoute.ts +51 -51
  264. package/src/ruvocal/src/lib/server/router/types.ts +21 -21
  265. package/src/ruvocal/src/lib/server/sendSlack.ts +23 -23
  266. package/src/ruvocal/src/lib/server/textGeneration/generate.ts +258 -258
  267. package/src/ruvocal/src/lib/server/textGeneration/index.ts +96 -96
  268. package/src/ruvocal/src/lib/server/textGeneration/mcp/fileRefs.ts +155 -155
  269. package/src/ruvocal/src/lib/server/textGeneration/mcp/routerResolution.ts +108 -108
  270. package/src/ruvocal/src/lib/server/textGeneration/mcp/runMcpFlow.ts +831 -831
  271. package/src/ruvocal/src/lib/server/textGeneration/mcp/toolInvocation.ts +349 -349
  272. package/src/ruvocal/src/lib/server/textGeneration/mcp/wasmTools.test.ts +633 -633
  273. package/src/ruvocal/src/lib/server/textGeneration/reasoning.ts +23 -23
  274. package/src/ruvocal/src/lib/server/textGeneration/title.ts +83 -83
  275. package/src/ruvocal/src/lib/server/textGeneration/types.ts +28 -28
  276. package/src/ruvocal/src/lib/server/textGeneration/utils/prepareFiles.ts +88 -88
  277. package/src/ruvocal/src/lib/server/textGeneration/utils/routing.ts +21 -21
  278. package/src/ruvocal/src/lib/server/textGeneration/utils/toolPrompt.ts +49 -49
  279. package/src/ruvocal/src/lib/server/urlSafety.ts +77 -77
  280. package/src/ruvocal/src/lib/server/usageLimits.ts +30 -30
  281. package/src/ruvocal/src/lib/stores/autopilotStore.svelte.ts +175 -175
  282. package/src/ruvocal/src/lib/stores/backgroundGenerations.svelte.ts +32 -32
  283. package/src/ruvocal/src/lib/stores/backgroundGenerations.ts +1 -1
  284. package/src/ruvocal/src/lib/stores/errors.ts +9 -9
  285. package/src/ruvocal/src/lib/stores/isAborted.ts +3 -3
  286. package/src/ruvocal/src/lib/stores/isPro.ts +4 -4
  287. package/src/ruvocal/src/lib/stores/loading.ts +3 -3
  288. package/src/ruvocal/src/lib/stores/mcpServers.ts +534 -534
  289. package/src/ruvocal/src/lib/stores/pendingChatInput.ts +3 -3
  290. package/src/ruvocal/src/lib/stores/pendingMessage.ts +9 -9
  291. package/src/ruvocal/src/lib/stores/settings.ts +182 -182
  292. package/src/ruvocal/src/lib/stores/shareModal.ts +13 -13
  293. package/src/ruvocal/src/lib/stores/titleUpdate.ts +8 -8
  294. package/src/ruvocal/src/lib/stores/wasmMcp.ts +472 -472
  295. package/src/ruvocal/src/lib/switchTheme.ts +124 -124
  296. package/src/ruvocal/src/lib/types/AbortedGeneration.ts +8 -8
  297. package/src/ruvocal/src/lib/types/Assistant.ts +31 -31
  298. package/src/ruvocal/src/lib/types/AssistantStats.ts +11 -11
  299. package/src/ruvocal/src/lib/types/ConfigKey.ts +4 -4
  300. package/src/ruvocal/src/lib/types/ConvSidebar.ts +9 -9
  301. package/src/ruvocal/src/lib/types/Conversation.ts +27 -27
  302. package/src/ruvocal/src/lib/types/ConversationStats.ts +13 -13
  303. package/src/ruvocal/src/lib/types/Message.ts +41 -41
  304. package/src/ruvocal/src/lib/types/MessageEvent.ts +10 -10
  305. package/src/ruvocal/src/lib/types/MessageUpdate.ts +139 -139
  306. package/src/ruvocal/src/lib/types/MigrationResult.ts +7 -7
  307. package/src/ruvocal/src/lib/types/Model.ts +23 -23
  308. package/src/ruvocal/src/lib/types/Report.ts +12 -12
  309. package/src/ruvocal/src/lib/types/Review.ts +6 -6
  310. package/src/ruvocal/src/lib/types/Semaphore.ts +19 -19
  311. package/src/ruvocal/src/lib/types/Session.ts +22 -22
  312. package/src/ruvocal/src/lib/types/Settings.ts +93 -93
  313. package/src/ruvocal/src/lib/types/SharedConversation.ts +9 -9
  314. package/src/ruvocal/src/lib/types/Template.ts +6 -6
  315. package/src/ruvocal/src/lib/types/Timestamps.ts +4 -4
  316. package/src/ruvocal/src/lib/types/TokenCache.ts +6 -6
  317. package/src/ruvocal/src/lib/types/Tool.ts +77 -77
  318. package/src/ruvocal/src/lib/types/UrlDependency.ts +5 -5
  319. package/src/ruvocal/src/lib/types/User.ts +14 -14
  320. package/src/ruvocal/src/lib/utils/PublicConfig.svelte.ts +75 -75
  321. package/src/ruvocal/src/lib/utils/auth.ts +17 -17
  322. package/src/ruvocal/src/lib/utils/chunk.ts +33 -33
  323. package/src/ruvocal/src/lib/utils/cookiesAreEnabled.ts +13 -13
  324. package/src/ruvocal/src/lib/utils/debounce.ts +17 -17
  325. package/src/ruvocal/src/lib/utils/deepestChild.ts +6 -6
  326. package/src/ruvocal/src/lib/utils/favicon.ts +21 -21
  327. package/src/ruvocal/src/lib/utils/fetchJSON.ts +23 -23
  328. package/src/ruvocal/src/lib/utils/file2base64.ts +14 -14
  329. package/src/ruvocal/src/lib/utils/formatUserCount.ts +37 -37
  330. package/src/ruvocal/src/lib/utils/generationState.spec.ts +75 -75
  331. package/src/ruvocal/src/lib/utils/generationState.ts +26 -26
  332. package/src/ruvocal/src/lib/utils/getHref.ts +41 -41
  333. package/src/ruvocal/src/lib/utils/getReturnFromGenerator.ts +7 -7
  334. package/src/ruvocal/src/lib/utils/haptics.ts +64 -64
  335. package/src/ruvocal/src/lib/utils/hashConv.ts +12 -12
  336. package/src/ruvocal/src/lib/utils/hf.ts +17 -17
  337. package/src/ruvocal/src/lib/utils/isDesktop.ts +7 -7
  338. package/src/ruvocal/src/lib/utils/isUrl.ts +8 -8
  339. package/src/ruvocal/src/lib/utils/isVirtualKeyboard.ts +16 -16
  340. package/src/ruvocal/src/lib/utils/loadAttachmentsFromUrls.ts +115 -115
  341. package/src/ruvocal/src/lib/utils/marked.spec.ts +96 -96
  342. package/src/ruvocal/src/lib/utils/marked.ts +531 -531
  343. package/src/ruvocal/src/lib/utils/mcpValidation.ts +147 -147
  344. package/src/ruvocal/src/lib/utils/mergeAsyncGenerators.ts +38 -38
  345. package/src/ruvocal/src/lib/utils/messageUpdates.spec.ts +262 -262
  346. package/src/ruvocal/src/lib/utils/messageUpdates.ts +324 -324
  347. package/src/ruvocal/src/lib/utils/mime.ts +56 -56
  348. package/src/ruvocal/src/lib/utils/models.ts +14 -14
  349. package/src/ruvocal/src/lib/utils/parseBlocks.ts +120 -120
  350. package/src/ruvocal/src/lib/utils/parseIncompleteMarkdown.ts +644 -644
  351. package/src/ruvocal/src/lib/utils/parseStringToList.ts +10 -10
  352. package/src/ruvocal/src/lib/utils/randomUuid.ts +14 -14
  353. package/src/ruvocal/src/lib/utils/searchTokens.ts +33 -33
  354. package/src/ruvocal/src/lib/utils/sha256.ts +7 -7
  355. package/src/ruvocal/src/lib/utils/stringifyError.ts +12 -12
  356. package/src/ruvocal/src/lib/utils/sum.ts +3 -3
  357. package/src/ruvocal/src/lib/utils/template.spec.ts +59 -59
  358. package/src/ruvocal/src/lib/utils/template.ts +53 -53
  359. package/src/ruvocal/src/lib/utils/timeout.ts +9 -9
  360. package/src/ruvocal/src/lib/utils/toolProgress.spec.ts +46 -46
  361. package/src/ruvocal/src/lib/utils/toolProgress.ts +11 -11
  362. package/src/ruvocal/src/lib/utils/tree/addChildren.spec.ts +102 -102
  363. package/src/ruvocal/src/lib/utils/tree/addChildren.ts +48 -48
  364. package/src/ruvocal/src/lib/utils/tree/addSibling.spec.ts +81 -81
  365. package/src/ruvocal/src/lib/utils/tree/addSibling.ts +41 -41
  366. package/src/ruvocal/src/lib/utils/tree/buildSubtree.spec.ts +110 -110
  367. package/src/ruvocal/src/lib/utils/tree/buildSubtree.ts +24 -24
  368. package/src/ruvocal/src/lib/utils/tree/convertLegacyConversation.spec.ts +31 -31
  369. package/src/ruvocal/src/lib/utils/tree/convertLegacyConversation.ts +36 -36
  370. package/src/ruvocal/src/lib/utils/tree/isMessageId.spec.ts +15 -15
  371. package/src/ruvocal/src/lib/utils/tree/isMessageId.ts +5 -5
  372. package/src/ruvocal/src/lib/utils/tree/tree.d.ts +14 -14
  373. package/src/ruvocal/src/lib/utils/tree/treeHelpers.spec.ts +167 -167
  374. package/src/ruvocal/src/lib/utils/updates.ts +39 -39
  375. package/src/ruvocal/src/lib/utils/urlParams.ts +13 -13
  376. package/src/ruvocal/src/lib/wasm/idb.ts +438 -438
  377. package/src/ruvocal/src/lib/wasm/index.ts +1213 -1213
  378. package/src/ruvocal/src/lib/wasm/tests/wasm-capabilities.test.ts +565 -565
  379. package/src/ruvocal/src/lib/wasm/wasm.worker.ts +332 -332
  380. package/src/ruvocal/src/lib/wasm/workerClient.ts +166 -166
  381. package/src/ruvocal/src/lib/workers/autopilotWorker.ts +221 -221
  382. package/src/ruvocal/src/lib/workers/detailFetchWorker.ts +100 -100
  383. package/src/ruvocal/src/lib/workers/markdownWorker.ts +61 -61
  384. package/src/ruvocal/src/routes/+error.svelte +20 -20
  385. package/src/ruvocal/src/routes/+layout.svelte +324 -324
  386. package/src/ruvocal/src/routes/+layout.ts +91 -91
  387. package/src/ruvocal/src/routes/+page.svelte +168 -168
  388. package/src/ruvocal/src/routes/.well-known/oauth-cimd/+server.ts +37 -37
  389. package/src/ruvocal/src/routes/__debug/openai/+server.ts +21 -21
  390. package/src/ruvocal/src/routes/admin/export/+server.ts +159 -159
  391. package/src/ruvocal/src/routes/admin/stats/compute/+server.ts +16 -16
  392. package/src/ruvocal/src/routes/api/conversation/[id]/+server.ts +40 -40
  393. package/src/ruvocal/src/routes/api/conversation/[id]/message/[messageId]/+server.ts +42 -42
  394. package/src/ruvocal/src/routes/api/conversations/+server.ts +48 -48
  395. package/src/ruvocal/src/routes/api/fetch-url/+server.ts +147 -147
  396. package/src/ruvocal/src/routes/api/mcp/health/+server.ts +292 -292
  397. package/src/ruvocal/src/routes/api/mcp/servers/+server.ts +32 -32
  398. package/src/ruvocal/src/routes/api/models/+server.ts +25 -25
  399. package/src/ruvocal/src/routes/api/transcribe/+server.ts +104 -104
  400. package/src/ruvocal/src/routes/api/user/+server.ts +15 -15
  401. package/src/ruvocal/src/routes/api/user/validate-token/+server.ts +20 -20
  402. package/src/ruvocal/src/routes/api/v2/conversations/+server.ts +48 -48
  403. package/src/ruvocal/src/routes/api/v2/conversations/[id]/+server.ts +94 -94
  404. package/src/ruvocal/src/routes/api/v2/conversations/[id]/message/[messageId]/+server.ts +43 -43
  405. package/src/ruvocal/src/routes/api/v2/conversations/import-share/+server.ts +23 -23
  406. package/src/ruvocal/src/routes/api/v2/debug/config/+server.ts +16 -16
  407. package/src/ruvocal/src/routes/api/v2/debug/refresh/+server.ts +30 -30
  408. package/src/ruvocal/src/routes/api/v2/export/+server.ts +196 -196
  409. package/src/ruvocal/src/routes/api/v2/feature-flags/+server.ts +14 -14
  410. package/src/ruvocal/src/routes/api/v2/models/+server.ts +38 -38
  411. package/src/ruvocal/src/routes/api/v2/models/[namespace]/+server.ts +8 -8
  412. package/src/ruvocal/src/routes/api/v2/models/[namespace]/[model]/+server.ts +8 -8
  413. package/src/ruvocal/src/routes/api/v2/models/[namespace]/[model]/subscribe/+server.ts +28 -28
  414. package/src/ruvocal/src/routes/api/v2/models/[namespace]/subscribe/+server.ts +28 -28
  415. package/src/ruvocal/src/routes/api/v2/models/old/+server.ts +7 -7
  416. package/src/ruvocal/src/routes/api/v2/models/refresh/+server.ts +33 -33
  417. package/src/ruvocal/src/routes/api/v2/public-config/+server.ts +7 -7
  418. package/src/ruvocal/src/routes/api/v2/user/+server.ts +17 -17
  419. package/src/ruvocal/src/routes/api/v2/user/billing-orgs/+server.ts +73 -73
  420. package/src/ruvocal/src/routes/api/v2/user/reports/+server.ts +17 -17
  421. package/src/ruvocal/src/routes/api/v2/user/settings/+server.ts +110 -110
  422. package/src/ruvocal/src/routes/conversation/+server.ts +115 -115
  423. package/src/ruvocal/src/routes/conversation/[id]/+page.svelte +586 -586
  424. package/src/ruvocal/src/routes/conversation/[id]/+page.ts +60 -60
  425. package/src/ruvocal/src/routes/conversation/[id]/+server.ts +740 -740
  426. package/src/ruvocal/src/routes/conversation/[id]/message/[messageId]/prompt/+server.ts +66 -66
  427. package/src/ruvocal/src/routes/conversation/[id]/share/+server.ts +69 -69
  428. package/src/ruvocal/src/routes/conversation/[id]/stop-generating/+server.ts +35 -35
  429. package/src/ruvocal/src/routes/healthcheck/+server.ts +3 -3
  430. package/src/ruvocal/src/routes/login/+server.ts +5 -5
  431. package/src/ruvocal/src/routes/login/callback/+server.ts +103 -103
  432. package/src/ruvocal/src/routes/login/callback/updateUser.spec.ts +157 -157
  433. package/src/ruvocal/src/routes/login/callback/updateUser.ts +215 -215
  434. package/src/ruvocal/src/routes/logout/+server.ts +18 -18
  435. package/src/ruvocal/src/routes/metrics/+server.ts +18 -18
  436. package/src/ruvocal/src/routes/models/+page.svelte +233 -233
  437. package/src/ruvocal/src/routes/models/[...model]/+page.svelte +161 -161
  438. package/src/ruvocal/src/routes/models/[...model]/+page.ts +14 -14
  439. package/src/ruvocal/src/routes/models/[...model]/thumbnail.png/+server.ts +64 -64
  440. package/src/ruvocal/src/routes/models/[...model]/thumbnail.png/ModelThumbnail.svelte +28 -28
  441. package/src/ruvocal/src/routes/privacy/+page.svelte +11 -11
  442. package/src/ruvocal/src/routes/r/[id]/+page.ts +34 -34
  443. package/src/ruvocal/src/routes/settings/(nav)/+layout.svelte +282 -282
  444. package/src/ruvocal/src/routes/settings/(nav)/+layout.ts +1 -1
  445. package/src/ruvocal/src/routes/settings/(nav)/+server.ts +59 -59
  446. package/src/ruvocal/src/routes/settings/(nav)/[...model]/+page.svelte +464 -464
  447. package/src/ruvocal/src/routes/settings/(nav)/[...model]/+page.ts +14 -14
  448. package/src/ruvocal/src/routes/settings/(nav)/application/+page.svelte +362 -362
  449. package/src/ruvocal/src/routes/settings/+layout.svelte +40 -40
  450. package/src/ruvocal/src/styles/highlight-js.css +195 -195
  451. package/src/ruvocal/src/styles/main.css +144 -144
  452. package/src/ruvocal/static/chatui/favicon-dark.svg +3 -3
  453. package/src/ruvocal/static/chatui/favicon-dev.svg +3 -3
  454. package/src/ruvocal/static/chatui/favicon.svg +3 -3
  455. package/src/ruvocal/static/chatui/icon.svg +3 -3
  456. package/src/ruvocal/static/chatui/logo.svg +7 -7
  457. package/src/ruvocal/static/chatui/manifest.json +54 -54
  458. package/src/ruvocal/static/chatui/welcome.js +184 -184
  459. package/src/ruvocal/static/huggingchat/favicon-dark.svg +4 -4
  460. package/src/ruvocal/static/huggingchat/favicon-dev.svg +4 -4
  461. package/src/ruvocal/static/huggingchat/favicon.svg +4 -4
  462. package/src/ruvocal/static/huggingchat/fulltext-logo.svg +1 -1
  463. package/src/ruvocal/static/huggingchat/icon.svg +4 -4
  464. package/src/ruvocal/static/huggingchat/logo.svg +4 -4
  465. package/src/ruvocal/static/huggingchat/manifest.json +54 -54
  466. package/src/ruvocal/static/huggingchat/routes.chat.json +226 -226
  467. package/src/ruvocal/static/robots.txt +10 -10
  468. package/src/ruvocal/static/wasm/rvagent_wasm.js +1539 -1539
  469. package/src/ruvocal/stub/@reflink/reflink/package.json +5 -5
  470. package/src/ruvocal/svelte.config.js +53 -53
  471. package/src/ruvocal/tailwind.config.cjs +30 -30
  472. package/src/ruvocal/tsconfig.json +19 -19
  473. package/src/ruvocal/vite.config.ts +87 -87
  474. package/src/scripts/deploy.sh +116 -116
  475. package/src/scripts/generate-config.js +245 -245
  476. package/src/scripts/generate-welcome.js +187 -187
  477. package/src/scripts/package-rvf.sh +116 -116
  478. package/src/ruvocal/.claude-flow/daemon-state.json +0 -135
  479. package/src/ruvocal/.claude-flow/data/pending-insights.jsonl +0 -0
  480. package/src/ruvocal/.claude-flow/data/ranked-context.json +0 -5
  481. package/src/ruvocal/.claude-flow/logs/daemon.log +0 -31
  482. package/src/ruvocal/.claude-flow/logs/headless/audit_1777949411822_juxau0_prompt.log +0 -989
  483. package/src/ruvocal/.claude-flow/logs/headless/audit_1777949411822_juxau0_result.log +0 -67
  484. package/src/ruvocal/.claude-flow/logs/headless/audit_1777950042278_jvj5xq_prompt.log +0 -989
  485. package/src/ruvocal/.claude-flow/logs/headless/audit_1777950042278_jvj5xq_result.log +0 -93
  486. package/src/ruvocal/.claude-flow/logs/headless/optimize_1777949531823_yt5yc2_prompt.log +0 -1498
  487. package/src/ruvocal/.claude-flow/logs/headless/optimize_1777949531823_yt5yc2_result.log +0 -93
  488. package/src/ruvocal/.claude-flow/logs/headless/testgaps_1777949771821_elw1j4_prompt.log +0 -1498
  489. package/src/ruvocal/.claude-flow/logs/headless/testgaps_1777949771821_elw1j4_result.log +0 -100
  490. package/src/ruvocal/.claude-flow/metrics/codebase-map.json +0 -11
  491. package/src/ruvocal/.claude-flow/metrics/consolidation.json +0 -6
  492. package/src/ruvocal/.claude-flow/neural/stats.json +0 -6
  493. package/src/ruvocal/.claude-flow/sessions/current.json +0 -13
  494. package/src/ruvocal/.swarm/attestation.db +0 -0
  495. package/src/ruvocal/.swarm/hnsw.index +0 -0
  496. package/src/ruvocal/.swarm/hnsw.metadata.json +0 -1
  497. package/src/ruvocal/.swarm/memory.db +0 -0
  498. package/src/ruvocal/.swarm/schema.sql +0 -305
@@ -1,534 +1,534 @@
1
- /**
2
- * MCP Servers Store
3
- * Manages base (env-configured), custom (user-added), and WASM (browser-local) MCP servers
4
- * Stores custom servers and selection state in browser localStorage
5
- * WASM servers run entirely in-browser via rvagent-wasm with IndexedDB persistence
6
- */
7
-
8
- import { writable, derived, get } from "svelte/store";
9
- import { base } from "$app/paths";
10
- import { env as publicEnv } from "$env/dynamic/public";
11
- import { browser } from "$app/environment";
12
- import type { MCPServer, ServerStatus, MCPTool } from "$lib/types/Tool";
13
- import {
14
- initWasmMcp,
15
- callMcp as callWasmMcp,
16
- listGalleryTemplates,
17
- loadGalleryTemplate,
18
- activeTemplate,
19
- } from "./wasmMcp";
20
-
21
- // Namespace storage by app identity to avoid collisions across apps
22
- function toKeyPart(s: string | undefined): string {
23
- return (s || "").toLowerCase().replace(/[^a-z0-9_-]+/g, "-");
24
- }
25
-
26
- const appLabel = toKeyPart(publicEnv.PUBLIC_APP_ASSETS || publicEnv.PUBLIC_APP_NAME);
27
- const baseLabel = toKeyPart(typeof base === "string" ? base : "");
28
- // Final prefix format requested: "huggingchat:key" (no mcp:/chat)
29
- const KEY_PREFIX = appLabel || baseLabel || "app";
30
-
31
- const STORAGE_KEYS = {
32
- CUSTOM_SERVERS: `${KEY_PREFIX}:mcp:custom-servers`,
33
- SELECTED_IDS: `${KEY_PREFIX}:mcp:selected-ids`,
34
- DISABLED_BASE_IDS: `${KEY_PREFIX}:mcp:disabled-base-ids`,
35
- } as const;
36
-
37
- // WASM MCP Server ID (constant, always available)
38
- export const WASM_SERVER_ID = "wasm-rvagent";
39
-
40
- // Create the WASM MCP server entry
41
- function createWasmServer(): MCPServer {
42
- return {
43
- id: WASM_SERVER_ID,
44
- name: "RVAgent Local (WASM)",
45
- url: "wasm://local",
46
- type: "wasm",
47
- status: "disconnected",
48
- isLocked: false,
49
- tools: [],
50
- };
51
- }
52
-
53
- // No migration needed per request — read/write only namespaced keys
54
-
55
- // Load custom servers from localStorage
56
- function loadCustomServers(): MCPServer[] {
57
- if (!browser) return [];
58
-
59
- try {
60
- const json = localStorage.getItem(STORAGE_KEYS.CUSTOM_SERVERS);
61
- return json ? JSON.parse(json) : [];
62
- } catch (error) {
63
- console.error("Failed to load custom MCP servers from localStorage:", error);
64
- return [];
65
- }
66
- }
67
-
68
- // Load selected server IDs from localStorage
69
- function loadSelectedIds(): Set<string> {
70
- if (!browser) return new Set();
71
-
72
- try {
73
- const json = localStorage.getItem(STORAGE_KEYS.SELECTED_IDS);
74
- const ids: string[] = json ? JSON.parse(json) : [];
75
- return new Set(ids);
76
- } catch (error) {
77
- console.error("Failed to load selected MCP server IDs from localStorage:", error);
78
- return new Set();
79
- }
80
- }
81
-
82
- // Save custom servers to localStorage
83
- function saveCustomServers(servers: MCPServer[]) {
84
- if (!browser) return;
85
-
86
- try {
87
- localStorage.setItem(STORAGE_KEYS.CUSTOM_SERVERS, JSON.stringify(servers));
88
- } catch (error) {
89
- console.error("Failed to save custom MCP servers to localStorage:", error);
90
- }
91
- }
92
-
93
- // Save selected IDs to localStorage
94
- function saveSelectedIds(ids: Set<string>) {
95
- if (!browser) return;
96
-
97
- try {
98
- localStorage.setItem(STORAGE_KEYS.SELECTED_IDS, JSON.stringify([...ids]));
99
- } catch (error) {
100
- console.error("Failed to save selected MCP server IDs to localStorage:", error);
101
- }
102
- }
103
-
104
- // Load disabled base server IDs from localStorage (empty set if missing or on error)
105
- function loadDisabledBaseIds(): Set<string> {
106
- if (!browser) return new Set();
107
-
108
- try {
109
- const json = localStorage.getItem(STORAGE_KEYS.DISABLED_BASE_IDS);
110
- return new Set(json ? JSON.parse(json) : []);
111
- } catch (error) {
112
- console.error("Failed to load disabled base MCP server IDs from localStorage:", error);
113
- return new Set();
114
- }
115
- }
116
-
117
- // Save disabled base server IDs to localStorage
118
- function saveDisabledBaseIds(ids: Set<string>) {
119
- if (!browser) return;
120
-
121
- try {
122
- localStorage.setItem(STORAGE_KEYS.DISABLED_BASE_IDS, JSON.stringify([...ids]));
123
- } catch (error) {
124
- console.error("Failed to save disabled base MCP server IDs to localStorage:", error);
125
- }
126
- }
127
-
128
- // Store for all servers (base + custom)
129
- export const allMcpServers = writable<MCPServer[]>([]);
130
-
131
- // Track if initial server load has completed
132
- export const mcpServersLoaded = writable<boolean>(false);
133
-
134
- // Store for selected server IDs
135
- export const selectedServerIds = writable<Set<string>>(loadSelectedIds());
136
-
137
- // Auto-persist selected IDs when they change
138
- if (browser) {
139
- selectedServerIds.subscribe((ids) => {
140
- saveSelectedIds(ids);
141
- });
142
- }
143
-
144
- // Derived store: only enabled servers
145
- export const enabledServers = derived([allMcpServers, selectedServerIds], ([$all, $selected]) =>
146
- $all.filter((s) => $selected.has(s.id))
147
- );
148
-
149
- // Derived store: count of enabled servers
150
- export const enabledServersCount = derived(enabledServers, ($enabled) => $enabled.length);
151
-
152
- // Derived store: true if all base servers are enabled
153
- export const allBaseServersEnabled = derived(
154
- [allMcpServers, selectedServerIds],
155
- ([$all, $selected]) => {
156
- const baseServers = $all.filter((s) => s.type === "base");
157
- return baseServers.length > 0 && baseServers.every((s) => $selected.has(s.id));
158
- }
159
- );
160
-
161
- // Note: Authorization overlay (with user's HF token) for the Hugging Face MCP host
162
- // is applied server-side when enabled via MCP_FORWARD_HF_USER_TOKEN.
163
-
164
- /**
165
- * Refresh base servers from API and merge with custom servers + WASM server
166
- */
167
- export async function refreshMcpServers() {
168
- try {
169
- const response = await fetch(`${base}/api/mcp/servers`);
170
- if (!response.ok) {
171
- throw new Error(`Failed to fetch base servers: ${response.statusText}`);
172
- }
173
-
174
- const baseServers: MCPServer[] = await response.json();
175
- const customServers = loadCustomServers();
176
-
177
- // Create WASM server and add to the list
178
- const wasmServer = createWasmServer();
179
-
180
- // Merge base, custom, and WASM servers
181
- const merged = [wasmServer, ...baseServers, ...customServers];
182
- allMcpServers.set(merged);
183
-
184
- // Load disabled base servers
185
- const disabledBaseIds = loadDisabledBaseIds();
186
-
187
- // Auto-enable all base servers that aren't explicitly disabled
188
- // Plus keep any custom servers that were previously selected
189
- // WASM server is auto-enabled by default
190
- const validIds = new Set(merged.map((s) => s.id));
191
- selectedServerIds.update(($currentIds) => {
192
- const newSelection = new Set<string>();
193
-
194
- // Auto-enable WASM server
195
- newSelection.add(WASM_SERVER_ID);
196
-
197
- // Add all base servers that aren't disabled
198
- for (const server of baseServers) {
199
- if (!disabledBaseIds.has(server.id)) {
200
- newSelection.add(server.id);
201
- }
202
- }
203
-
204
- // Keep custom servers that were selected and still exist
205
- for (const id of $currentIds) {
206
- if (validIds.has(id) && !id.startsWith("base-")) {
207
- newSelection.add(id);
208
- }
209
- }
210
-
211
- return newSelection;
212
- });
213
- mcpServersLoaded.set(true);
214
-
215
- // Initialize WASM MCP server in background
216
- initWasmServer();
217
- } catch (error) {
218
- console.error("Failed to refresh MCP servers:", error);
219
- // On error, use custom servers + WASM server
220
- const wasmServer = createWasmServer();
221
- allMcpServers.set([wasmServer, ...loadCustomServers()]);
222
- mcpServersLoaded.set(true);
223
-
224
- // Still try to init WASM
225
- initWasmServer();
226
- }
227
- }
228
-
229
- /**
230
- * Initialize the WASM MCP server
231
- */
232
- async function initWasmServer() {
233
- if (!browser) return;
234
-
235
- updateServerStatus(WASM_SERVER_ID, "connecting");
236
-
237
- try {
238
- const success = await initWasmMcp();
239
-
240
- if (success) {
241
- // Get tools from WASM server
242
- const toolsResponse = await callWasmMcp("tools/list");
243
- const tools: MCPTool[] = [];
244
-
245
- if (!toolsResponse.error && toolsResponse.result) {
246
- const result = toolsResponse.result as { tools: MCPTool[] };
247
- if (result.tools) {
248
- tools.push(...result.tools);
249
- }
250
- }
251
-
252
- // Get active template info
253
- const template = get(activeTemplate);
254
-
255
- updateServerStatus(WASM_SERVER_ID, "connected", undefined, tools);
256
-
257
- // Update template info
258
- allMcpServers.update(($servers) =>
259
- $servers.map((s) =>
260
- s.id === WASM_SERVER_ID
261
- ? {
262
- ...s,
263
- wasmTemplateId: template.id || undefined,
264
- wasmTemplateName: template.name || undefined,
265
- }
266
- : s
267
- )
268
- );
269
-
270
- console.log(`[MCP] WASM server initialized with ${tools.length} tools`);
271
- } else {
272
- updateServerStatus(WASM_SERVER_ID, "error", "Failed to load WASM module");
273
- }
274
- } catch (error) {
275
- const errorMessage = error instanceof Error ? error.message : "Unknown error";
276
- updateServerStatus(WASM_SERVER_ID, "error", errorMessage);
277
- console.error("[MCP] WASM server initialization failed:", error);
278
- }
279
- }
280
-
281
- /**
282
- * Toggle a server on/off
283
- */
284
- export function toggleServer(id: string) {
285
- selectedServerIds.update(($ids) => {
286
- const newSet = new Set($ids);
287
- if (newSet.has(id)) {
288
- newSet.delete(id);
289
- // Track if this is a base server being disabled
290
- if (id.startsWith("base-")) {
291
- const disabled = loadDisabledBaseIds();
292
- disabled.add(id);
293
- saveDisabledBaseIds(disabled);
294
- }
295
- } else {
296
- newSet.add(id);
297
- // Remove from disabled if re-enabling a base server
298
- if (id.startsWith("base-")) {
299
- const disabled = loadDisabledBaseIds();
300
- disabled.delete(id);
301
- saveDisabledBaseIds(disabled);
302
- }
303
- }
304
- return newSet;
305
- });
306
- }
307
-
308
- /**
309
- * Disable all MCP servers (marks all base servers as disabled)
310
- */
311
- export function disableAllServers() {
312
- // Get current base server IDs and mark them all as disabled
313
- const servers = get(allMcpServers);
314
- const baseServerIds = servers.filter((s) => s.type === "base").map((s) => s.id);
315
-
316
- // Save all base servers as disabled
317
- saveDisabledBaseIds(new Set(baseServerIds));
318
-
319
- // Clear the selection
320
- selectedServerIds.set(new Set());
321
- }
322
-
323
- /**
324
- * Add a custom MCP server
325
- */
326
- export function addCustomServer(server: Omit<MCPServer, "id" | "type" | "status">): string {
327
- const newServer: MCPServer = {
328
- ...server,
329
- id: crypto.randomUUID(),
330
- type: "custom",
331
- status: "disconnected",
332
- };
333
-
334
- const customServers = loadCustomServers();
335
- customServers.push(newServer);
336
- saveCustomServers(customServers);
337
-
338
- // Refresh all servers to include the new one
339
- refreshMcpServers();
340
-
341
- return newServer.id;
342
- }
343
-
344
- /**
345
- * Update an existing custom server
346
- */
347
- export function updateCustomServer(id: string, updates: Partial<MCPServer>) {
348
- const customServers = loadCustomServers();
349
- const index = customServers.findIndex((s) => s.id === id);
350
-
351
- if (index !== -1) {
352
- customServers[index] = { ...customServers[index], ...updates };
353
- saveCustomServers(customServers);
354
- refreshMcpServers();
355
- }
356
- }
357
-
358
- /**
359
- * Delete a custom server
360
- */
361
- export function deleteCustomServer(id: string) {
362
- const customServers = loadCustomServers();
363
- const filtered = customServers.filter((s) => s.id !== id);
364
- saveCustomServers(filtered);
365
-
366
- // Also remove from selected IDs
367
- selectedServerIds.update(($ids) => {
368
- const newSet = new Set($ids);
369
- newSet.delete(id);
370
- return newSet;
371
- });
372
-
373
- refreshMcpServers();
374
- }
375
-
376
- /**
377
- * Update server status (from health check)
378
- */
379
- export function updateServerStatus(
380
- id: string,
381
- status: ServerStatus,
382
- errorMessage?: string,
383
- tools?: MCPTool[],
384
- authRequired?: boolean
385
- ) {
386
- allMcpServers.update(($servers) =>
387
- $servers.map((s) =>
388
- s.id === id
389
- ? {
390
- ...s,
391
- status,
392
- errorMessage,
393
- tools,
394
- authRequired,
395
- }
396
- : s
397
- )
398
- );
399
- }
400
-
401
- /**
402
- * Run health check on a server
403
- */
404
- export async function healthCheckServer(
405
- server: MCPServer
406
- ): Promise<{ ready: boolean; tools?: MCPTool[]; error?: string }> {
407
- // Handle WASM servers locally
408
- if (server.type === "wasm") {
409
- return healthCheckWasmServer();
410
- }
411
-
412
- try {
413
- updateServerStatus(server.id, "connecting");
414
-
415
- const response = await fetch(`${base}/api/mcp/health`, {
416
- method: "POST",
417
- headers: { "Content-Type": "application/json" },
418
- body: JSON.stringify({ url: server.url, headers: server.headers }),
419
- });
420
-
421
- const result = await response.json();
422
-
423
- if (result.ready && result.tools) {
424
- updateServerStatus(server.id, "connected", undefined, result.tools, false);
425
- return { ready: true, tools: result.tools };
426
- } else {
427
- updateServerStatus(server.id, "error", result.error, undefined, Boolean(result.authRequired));
428
- return { ready: false, error: result.error };
429
- }
430
- } catch (error) {
431
- const errorMessage = error instanceof Error ? error.message : "Unknown error";
432
- updateServerStatus(server.id, "error", errorMessage);
433
- return { ready: false, error: errorMessage };
434
- }
435
- }
436
-
437
- /**
438
- * Health check for WASM MCP server (runs locally)
439
- */
440
- async function healthCheckWasmServer(): Promise<{ ready: boolean; tools?: MCPTool[]; error?: string }> {
441
- try {
442
- updateServerStatus(WASM_SERVER_ID, "connecting");
443
-
444
- const success = await initWasmMcp();
445
-
446
- if (!success) {
447
- updateServerStatus(WASM_SERVER_ID, "error", "Failed to load WASM module");
448
- return { ready: false, error: "Failed to load WASM module" };
449
- }
450
-
451
- // Get tools from WASM server
452
- const toolsResponse = await callWasmMcp("tools/list");
453
- const tools: MCPTool[] = [];
454
-
455
- if (!toolsResponse.error && toolsResponse.result) {
456
- const result = toolsResponse.result as { tools: MCPTool[] };
457
- if (result.tools) {
458
- tools.push(...result.tools);
459
- }
460
- }
461
-
462
- // Get active template info
463
- const template = get(activeTemplate);
464
-
465
- updateServerStatus(WASM_SERVER_ID, "connected", undefined, tools);
466
-
467
- // Update template info
468
- allMcpServers.update(($servers) =>
469
- $servers.map((s) =>
470
- s.id === WASM_SERVER_ID
471
- ? {
472
- ...s,
473
- wasmTemplateId: template.id || undefined,
474
- wasmTemplateName: template.name || undefined,
475
- }
476
- : s
477
- )
478
- );
479
-
480
- return { ready: true, tools };
481
- } catch (error) {
482
- const errorMessage = error instanceof Error ? error.message : "Unknown error";
483
- updateServerStatus(WASM_SERVER_ID, "error", errorMessage);
484
- return { ready: false, error: errorMessage };
485
- }
486
- }
487
-
488
- /**
489
- * Load a gallery template for the WASM MCP server
490
- */
491
- export async function loadWasmTemplate(templateId: string): Promise<boolean> {
492
- try {
493
- const success = await loadGalleryTemplate(templateId);
494
-
495
- if (success) {
496
- // Refresh tools after loading template
497
- await healthCheckWasmServer();
498
- return true;
499
- }
500
-
501
- return false;
502
- } catch (error) {
503
- console.error("[MCP] Failed to load WASM template:", error);
504
- return false;
505
- }
506
- }
507
-
508
- /**
509
- * Get available gallery templates for WASM server
510
- */
511
- export function getWasmGalleryTemplates() {
512
- return listGalleryTemplates();
513
- }
514
-
515
- /**
516
- * Execute a tool on the WASM MCP server
517
- */
518
- export async function executeWasmTool(
519
- name: string,
520
- args: Record<string, unknown>
521
- ): Promise<{ success: boolean; result?: unknown; error?: string }> {
522
- const response = await callWasmMcp("tools/call", { name, arguments: args });
523
-
524
- if (response.error) {
525
- return { success: false, error: response.error.message };
526
- }
527
-
528
- return { success: true, result: response.result };
529
- }
530
-
531
- // Initialize on module load
532
- if (browser) {
533
- refreshMcpServers();
534
- }
1
+ /**
2
+ * MCP Servers Store
3
+ * Manages base (env-configured), custom (user-added), and WASM (browser-local) MCP servers
4
+ * Stores custom servers and selection state in browser localStorage
5
+ * WASM servers run entirely in-browser via rvagent-wasm with IndexedDB persistence
6
+ */
7
+
8
+ import { writable, derived, get } from "svelte/store";
9
+ import { base } from "$app/paths";
10
+ import { env as publicEnv } from "$env/dynamic/public";
11
+ import { browser } from "$app/environment";
12
+ import type { MCPServer, ServerStatus, MCPTool } from "$lib/types/Tool";
13
+ import {
14
+ initWasmMcp,
15
+ callMcp as callWasmMcp,
16
+ listGalleryTemplates,
17
+ loadGalleryTemplate,
18
+ activeTemplate,
19
+ } from "./wasmMcp";
20
+
21
+ // Namespace storage by app identity to avoid collisions across apps
22
+ function toKeyPart(s: string | undefined): string {
23
+ return (s || "").toLowerCase().replace(/[^a-z0-9_-]+/g, "-");
24
+ }
25
+
26
+ const appLabel = toKeyPart(publicEnv.PUBLIC_APP_ASSETS || publicEnv.PUBLIC_APP_NAME);
27
+ const baseLabel = toKeyPart(typeof base === "string" ? base : "");
28
+ // Final prefix format requested: "huggingchat:key" (no mcp:/chat)
29
+ const KEY_PREFIX = appLabel || baseLabel || "app";
30
+
31
+ const STORAGE_KEYS = {
32
+ CUSTOM_SERVERS: `${KEY_PREFIX}:mcp:custom-servers`,
33
+ SELECTED_IDS: `${KEY_PREFIX}:mcp:selected-ids`,
34
+ DISABLED_BASE_IDS: `${KEY_PREFIX}:mcp:disabled-base-ids`,
35
+ } as const;
36
+
37
+ // WASM MCP Server ID (constant, always available)
38
+ export const WASM_SERVER_ID = "wasm-rvagent";
39
+
40
+ // Create the WASM MCP server entry
41
+ function createWasmServer(): MCPServer {
42
+ return {
43
+ id: WASM_SERVER_ID,
44
+ name: "RVAgent Local (WASM)",
45
+ url: "wasm://local",
46
+ type: "wasm",
47
+ status: "disconnected",
48
+ isLocked: false,
49
+ tools: [],
50
+ };
51
+ }
52
+
53
+ // No migration needed per request — read/write only namespaced keys
54
+
55
+ // Load custom servers from localStorage
56
+ function loadCustomServers(): MCPServer[] {
57
+ if (!browser) return [];
58
+
59
+ try {
60
+ const json = localStorage.getItem(STORAGE_KEYS.CUSTOM_SERVERS);
61
+ return json ? JSON.parse(json) : [];
62
+ } catch (error) {
63
+ console.error("Failed to load custom MCP servers from localStorage:", error);
64
+ return [];
65
+ }
66
+ }
67
+
68
+ // Load selected server IDs from localStorage
69
+ function loadSelectedIds(): Set<string> {
70
+ if (!browser) return new Set();
71
+
72
+ try {
73
+ const json = localStorage.getItem(STORAGE_KEYS.SELECTED_IDS);
74
+ const ids: string[] = json ? JSON.parse(json) : [];
75
+ return new Set(ids);
76
+ } catch (error) {
77
+ console.error("Failed to load selected MCP server IDs from localStorage:", error);
78
+ return new Set();
79
+ }
80
+ }
81
+
82
+ // Save custom servers to localStorage
83
+ function saveCustomServers(servers: MCPServer[]) {
84
+ if (!browser) return;
85
+
86
+ try {
87
+ localStorage.setItem(STORAGE_KEYS.CUSTOM_SERVERS, JSON.stringify(servers));
88
+ } catch (error) {
89
+ console.error("Failed to save custom MCP servers to localStorage:", error);
90
+ }
91
+ }
92
+
93
+ // Save selected IDs to localStorage
94
+ function saveSelectedIds(ids: Set<string>) {
95
+ if (!browser) return;
96
+
97
+ try {
98
+ localStorage.setItem(STORAGE_KEYS.SELECTED_IDS, JSON.stringify([...ids]));
99
+ } catch (error) {
100
+ console.error("Failed to save selected MCP server IDs to localStorage:", error);
101
+ }
102
+ }
103
+
104
+ // Load disabled base server IDs from localStorage (empty set if missing or on error)
105
+ function loadDisabledBaseIds(): Set<string> {
106
+ if (!browser) return new Set();
107
+
108
+ try {
109
+ const json = localStorage.getItem(STORAGE_KEYS.DISABLED_BASE_IDS);
110
+ return new Set(json ? JSON.parse(json) : []);
111
+ } catch (error) {
112
+ console.error("Failed to load disabled base MCP server IDs from localStorage:", error);
113
+ return new Set();
114
+ }
115
+ }
116
+
117
+ // Save disabled base server IDs to localStorage
118
+ function saveDisabledBaseIds(ids: Set<string>) {
119
+ if (!browser) return;
120
+
121
+ try {
122
+ localStorage.setItem(STORAGE_KEYS.DISABLED_BASE_IDS, JSON.stringify([...ids]));
123
+ } catch (error) {
124
+ console.error("Failed to save disabled base MCP server IDs to localStorage:", error);
125
+ }
126
+ }
127
+
128
+ // Store for all servers (base + custom)
129
+ export const allMcpServers = writable<MCPServer[]>([]);
130
+
131
+ // Track if initial server load has completed
132
+ export const mcpServersLoaded = writable<boolean>(false);
133
+
134
+ // Store for selected server IDs
135
+ export const selectedServerIds = writable<Set<string>>(loadSelectedIds());
136
+
137
+ // Auto-persist selected IDs when they change
138
+ if (browser) {
139
+ selectedServerIds.subscribe((ids) => {
140
+ saveSelectedIds(ids);
141
+ });
142
+ }
143
+
144
+ // Derived store: only enabled servers
145
+ export const enabledServers = derived([allMcpServers, selectedServerIds], ([$all, $selected]) =>
146
+ $all.filter((s) => $selected.has(s.id))
147
+ );
148
+
149
+ // Derived store: count of enabled servers
150
+ export const enabledServersCount = derived(enabledServers, ($enabled) => $enabled.length);
151
+
152
+ // Derived store: true if all base servers are enabled
153
+ export const allBaseServersEnabled = derived(
154
+ [allMcpServers, selectedServerIds],
155
+ ([$all, $selected]) => {
156
+ const baseServers = $all.filter((s) => s.type === "base");
157
+ return baseServers.length > 0 && baseServers.every((s) => $selected.has(s.id));
158
+ }
159
+ );
160
+
161
+ // Note: Authorization overlay (with user's HF token) for the Hugging Face MCP host
162
+ // is applied server-side when enabled via MCP_FORWARD_HF_USER_TOKEN.
163
+
164
+ /**
165
+ * Refresh base servers from API and merge with custom servers + WASM server
166
+ */
167
+ export async function refreshMcpServers() {
168
+ try {
169
+ const response = await fetch(`${base}/api/mcp/servers`);
170
+ if (!response.ok) {
171
+ throw new Error(`Failed to fetch base servers: ${response.statusText}`);
172
+ }
173
+
174
+ const baseServers: MCPServer[] = await response.json();
175
+ const customServers = loadCustomServers();
176
+
177
+ // Create WASM server and add to the list
178
+ const wasmServer = createWasmServer();
179
+
180
+ // Merge base, custom, and WASM servers
181
+ const merged = [wasmServer, ...baseServers, ...customServers];
182
+ allMcpServers.set(merged);
183
+
184
+ // Load disabled base servers
185
+ const disabledBaseIds = loadDisabledBaseIds();
186
+
187
+ // Auto-enable all base servers that aren't explicitly disabled
188
+ // Plus keep any custom servers that were previously selected
189
+ // WASM server is auto-enabled by default
190
+ const validIds = new Set(merged.map((s) => s.id));
191
+ selectedServerIds.update(($currentIds) => {
192
+ const newSelection = new Set<string>();
193
+
194
+ // Auto-enable WASM server
195
+ newSelection.add(WASM_SERVER_ID);
196
+
197
+ // Add all base servers that aren't disabled
198
+ for (const server of baseServers) {
199
+ if (!disabledBaseIds.has(server.id)) {
200
+ newSelection.add(server.id);
201
+ }
202
+ }
203
+
204
+ // Keep custom servers that were selected and still exist
205
+ for (const id of $currentIds) {
206
+ if (validIds.has(id) && !id.startsWith("base-")) {
207
+ newSelection.add(id);
208
+ }
209
+ }
210
+
211
+ return newSelection;
212
+ });
213
+ mcpServersLoaded.set(true);
214
+
215
+ // Initialize WASM MCP server in background
216
+ initWasmServer();
217
+ } catch (error) {
218
+ console.error("Failed to refresh MCP servers:", error);
219
+ // On error, use custom servers + WASM server
220
+ const wasmServer = createWasmServer();
221
+ allMcpServers.set([wasmServer, ...loadCustomServers()]);
222
+ mcpServersLoaded.set(true);
223
+
224
+ // Still try to init WASM
225
+ initWasmServer();
226
+ }
227
+ }
228
+
229
+ /**
230
+ * Initialize the WASM MCP server
231
+ */
232
+ async function initWasmServer() {
233
+ if (!browser) return;
234
+
235
+ updateServerStatus(WASM_SERVER_ID, "connecting");
236
+
237
+ try {
238
+ const success = await initWasmMcp();
239
+
240
+ if (success) {
241
+ // Get tools from WASM server
242
+ const toolsResponse = await callWasmMcp("tools/list");
243
+ const tools: MCPTool[] = [];
244
+
245
+ if (!toolsResponse.error && toolsResponse.result) {
246
+ const result = toolsResponse.result as { tools: MCPTool[] };
247
+ if (result.tools) {
248
+ tools.push(...result.tools);
249
+ }
250
+ }
251
+
252
+ // Get active template info
253
+ const template = get(activeTemplate);
254
+
255
+ updateServerStatus(WASM_SERVER_ID, "connected", undefined, tools);
256
+
257
+ // Update template info
258
+ allMcpServers.update(($servers) =>
259
+ $servers.map((s) =>
260
+ s.id === WASM_SERVER_ID
261
+ ? {
262
+ ...s,
263
+ wasmTemplateId: template.id || undefined,
264
+ wasmTemplateName: template.name || undefined,
265
+ }
266
+ : s
267
+ )
268
+ );
269
+
270
+ console.log(`[MCP] WASM server initialized with ${tools.length} tools`);
271
+ } else {
272
+ updateServerStatus(WASM_SERVER_ID, "error", "Failed to load WASM module");
273
+ }
274
+ } catch (error) {
275
+ const errorMessage = error instanceof Error ? error.message : "Unknown error";
276
+ updateServerStatus(WASM_SERVER_ID, "error", errorMessage);
277
+ console.error("[MCP] WASM server initialization failed:", error);
278
+ }
279
+ }
280
+
281
+ /**
282
+ * Toggle a server on/off
283
+ */
284
+ export function toggleServer(id: string) {
285
+ selectedServerIds.update(($ids) => {
286
+ const newSet = new Set($ids);
287
+ if (newSet.has(id)) {
288
+ newSet.delete(id);
289
+ // Track if this is a base server being disabled
290
+ if (id.startsWith("base-")) {
291
+ const disabled = loadDisabledBaseIds();
292
+ disabled.add(id);
293
+ saveDisabledBaseIds(disabled);
294
+ }
295
+ } else {
296
+ newSet.add(id);
297
+ // Remove from disabled if re-enabling a base server
298
+ if (id.startsWith("base-")) {
299
+ const disabled = loadDisabledBaseIds();
300
+ disabled.delete(id);
301
+ saveDisabledBaseIds(disabled);
302
+ }
303
+ }
304
+ return newSet;
305
+ });
306
+ }
307
+
308
+ /**
309
+ * Disable all MCP servers (marks all base servers as disabled)
310
+ */
311
+ export function disableAllServers() {
312
+ // Get current base server IDs and mark them all as disabled
313
+ const servers = get(allMcpServers);
314
+ const baseServerIds = servers.filter((s) => s.type === "base").map((s) => s.id);
315
+
316
+ // Save all base servers as disabled
317
+ saveDisabledBaseIds(new Set(baseServerIds));
318
+
319
+ // Clear the selection
320
+ selectedServerIds.set(new Set());
321
+ }
322
+
323
+ /**
324
+ * Add a custom MCP server
325
+ */
326
+ export function addCustomServer(server: Omit<MCPServer, "id" | "type" | "status">): string {
327
+ const newServer: MCPServer = {
328
+ ...server,
329
+ id: crypto.randomUUID(),
330
+ type: "custom",
331
+ status: "disconnected",
332
+ };
333
+
334
+ const customServers = loadCustomServers();
335
+ customServers.push(newServer);
336
+ saveCustomServers(customServers);
337
+
338
+ // Refresh all servers to include the new one
339
+ refreshMcpServers();
340
+
341
+ return newServer.id;
342
+ }
343
+
344
+ /**
345
+ * Update an existing custom server
346
+ */
347
+ export function updateCustomServer(id: string, updates: Partial<MCPServer>) {
348
+ const customServers = loadCustomServers();
349
+ const index = customServers.findIndex((s) => s.id === id);
350
+
351
+ if (index !== -1) {
352
+ customServers[index] = { ...customServers[index], ...updates };
353
+ saveCustomServers(customServers);
354
+ refreshMcpServers();
355
+ }
356
+ }
357
+
358
+ /**
359
+ * Delete a custom server
360
+ */
361
+ export function deleteCustomServer(id: string) {
362
+ const customServers = loadCustomServers();
363
+ const filtered = customServers.filter((s) => s.id !== id);
364
+ saveCustomServers(filtered);
365
+
366
+ // Also remove from selected IDs
367
+ selectedServerIds.update(($ids) => {
368
+ const newSet = new Set($ids);
369
+ newSet.delete(id);
370
+ return newSet;
371
+ });
372
+
373
+ refreshMcpServers();
374
+ }
375
+
376
+ /**
377
+ * Update server status (from health check)
378
+ */
379
+ export function updateServerStatus(
380
+ id: string,
381
+ status: ServerStatus,
382
+ errorMessage?: string,
383
+ tools?: MCPTool[],
384
+ authRequired?: boolean
385
+ ) {
386
+ allMcpServers.update(($servers) =>
387
+ $servers.map((s) =>
388
+ s.id === id
389
+ ? {
390
+ ...s,
391
+ status,
392
+ errorMessage,
393
+ tools,
394
+ authRequired,
395
+ }
396
+ : s
397
+ )
398
+ );
399
+ }
400
+
401
+ /**
402
+ * Run health check on a server
403
+ */
404
+ export async function healthCheckServer(
405
+ server: MCPServer
406
+ ): Promise<{ ready: boolean; tools?: MCPTool[]; error?: string }> {
407
+ // Handle WASM servers locally
408
+ if (server.type === "wasm") {
409
+ return healthCheckWasmServer();
410
+ }
411
+
412
+ try {
413
+ updateServerStatus(server.id, "connecting");
414
+
415
+ const response = await fetch(`${base}/api/mcp/health`, {
416
+ method: "POST",
417
+ headers: { "Content-Type": "application/json" },
418
+ body: JSON.stringify({ url: server.url, headers: server.headers }),
419
+ });
420
+
421
+ const result = await response.json();
422
+
423
+ if (result.ready && result.tools) {
424
+ updateServerStatus(server.id, "connected", undefined, result.tools, false);
425
+ return { ready: true, tools: result.tools };
426
+ } else {
427
+ updateServerStatus(server.id, "error", result.error, undefined, Boolean(result.authRequired));
428
+ return { ready: false, error: result.error };
429
+ }
430
+ } catch (error) {
431
+ const errorMessage = error instanceof Error ? error.message : "Unknown error";
432
+ updateServerStatus(server.id, "error", errorMessage);
433
+ return { ready: false, error: errorMessage };
434
+ }
435
+ }
436
+
437
+ /**
438
+ * Health check for WASM MCP server (runs locally)
439
+ */
440
+ async function healthCheckWasmServer(): Promise<{ ready: boolean; tools?: MCPTool[]; error?: string }> {
441
+ try {
442
+ updateServerStatus(WASM_SERVER_ID, "connecting");
443
+
444
+ const success = await initWasmMcp();
445
+
446
+ if (!success) {
447
+ updateServerStatus(WASM_SERVER_ID, "error", "Failed to load WASM module");
448
+ return { ready: false, error: "Failed to load WASM module" };
449
+ }
450
+
451
+ // Get tools from WASM server
452
+ const toolsResponse = await callWasmMcp("tools/list");
453
+ const tools: MCPTool[] = [];
454
+
455
+ if (!toolsResponse.error && toolsResponse.result) {
456
+ const result = toolsResponse.result as { tools: MCPTool[] };
457
+ if (result.tools) {
458
+ tools.push(...result.tools);
459
+ }
460
+ }
461
+
462
+ // Get active template info
463
+ const template = get(activeTemplate);
464
+
465
+ updateServerStatus(WASM_SERVER_ID, "connected", undefined, tools);
466
+
467
+ // Update template info
468
+ allMcpServers.update(($servers) =>
469
+ $servers.map((s) =>
470
+ s.id === WASM_SERVER_ID
471
+ ? {
472
+ ...s,
473
+ wasmTemplateId: template.id || undefined,
474
+ wasmTemplateName: template.name || undefined,
475
+ }
476
+ : s
477
+ )
478
+ );
479
+
480
+ return { ready: true, tools };
481
+ } catch (error) {
482
+ const errorMessage = error instanceof Error ? error.message : "Unknown error";
483
+ updateServerStatus(WASM_SERVER_ID, "error", errorMessage);
484
+ return { ready: false, error: errorMessage };
485
+ }
486
+ }
487
+
488
+ /**
489
+ * Load a gallery template for the WASM MCP server
490
+ */
491
+ export async function loadWasmTemplate(templateId: string): Promise<boolean> {
492
+ try {
493
+ const success = await loadGalleryTemplate(templateId);
494
+
495
+ if (success) {
496
+ // Refresh tools after loading template
497
+ await healthCheckWasmServer();
498
+ return true;
499
+ }
500
+
501
+ return false;
502
+ } catch (error) {
503
+ console.error("[MCP] Failed to load WASM template:", error);
504
+ return false;
505
+ }
506
+ }
507
+
508
+ /**
509
+ * Get available gallery templates for WASM server
510
+ */
511
+ export function getWasmGalleryTemplates() {
512
+ return listGalleryTemplates();
513
+ }
514
+
515
+ /**
516
+ * Execute a tool on the WASM MCP server
517
+ */
518
+ export async function executeWasmTool(
519
+ name: string,
520
+ args: Record<string, unknown>
521
+ ): Promise<{ success: boolean; result?: unknown; error?: string }> {
522
+ const response = await callWasmMcp("tools/call", { name, arguments: args });
523
+
524
+ if (response.error) {
525
+ return { success: false, error: response.error.message };
526
+ }
527
+
528
+ return { success: true, result: response.result };
529
+ }
530
+
531
+ // Initialize on module load
532
+ if (browser) {
533
+ refreshMcpServers();
534
+ }