ruflo 3.5.2 → 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,32 @@
1
+ // Minimal shared helpers for HF MCP token forwarding
2
+
3
+ export const hasAuthHeader = (h?: Record<string, string>) =>
4
+ !!h && Object.keys(h).some((k) => k.toLowerCase() === "authorization");
5
+
6
+ export const isStrictHfMcpLogin = (urlString: string) => {
7
+ try {
8
+ const u = new URL(urlString);
9
+ const host = u.hostname.toLowerCase();
10
+ const allowedHosts = new Set(["hf.co", "huggingface.co"]);
11
+ return (
12
+ u.protocol === "https:" &&
13
+ allowedHosts.has(host) &&
14
+ u.pathname === "/mcp" &&
15
+ u.search === "?login"
16
+ );
17
+ } catch {
18
+ return false;
19
+ }
20
+ };
21
+
22
+ export const hasNonEmptyToken = (tok: unknown): tok is string =>
23
+ typeof tok === "string" && tok.trim().length > 0;
24
+
25
+ export const isExaMcpServer = (urlString: string): boolean => {
26
+ try {
27
+ const u = new URL(urlString);
28
+ return u.protocol === "https:" && u.hostname.toLowerCase() === "mcp.exa.ai";
29
+ } catch {
30
+ return false;
31
+ }
32
+ };
@@ -0,0 +1,122 @@
1
+ import { Client } from "@modelcontextprotocol/sdk/client";
2
+ import { getClient, evictFromPool } from "./clientPool";
3
+ import { config } from "$lib/server/config";
4
+
5
+ function isConnectionClosedError(err: unknown): boolean {
6
+ const message = err instanceof Error ? err.message : String(err);
7
+ return message.includes("-32000") || message.toLowerCase().includes("connection closed");
8
+ }
9
+
10
+ export interface McpServerConfig {
11
+ name: string;
12
+ url: string;
13
+ headers?: Record<string, string>;
14
+ }
15
+
16
+ const DEFAULT_TIMEOUT_MS = 120_000;
17
+
18
+ export function getMcpToolTimeoutMs(): number {
19
+ const envValue = config.MCP_TOOL_TIMEOUT_MS;
20
+ if (envValue) {
21
+ const parsed = parseInt(envValue, 10);
22
+ if (!isNaN(parsed) && parsed > 0) {
23
+ return parsed;
24
+ }
25
+ }
26
+ return DEFAULT_TIMEOUT_MS;
27
+ }
28
+
29
+ export type McpToolTextResponse = {
30
+ text: string;
31
+ /** If the server returned structuredContent, include it raw */
32
+ structured?: unknown;
33
+ /** Raw content blocks returned by the server, if any */
34
+ content?: unknown[];
35
+ };
36
+
37
+ export type McpToolProgress = {
38
+ progress: number;
39
+ total?: number;
40
+ message?: string;
41
+ };
42
+
43
+ export async function callMcpTool(
44
+ server: McpServerConfig,
45
+ tool: string,
46
+ args: unknown = {},
47
+ {
48
+ timeoutMs = DEFAULT_TIMEOUT_MS,
49
+ signal,
50
+ client,
51
+ onProgress,
52
+ }: {
53
+ timeoutMs?: number;
54
+ signal?: AbortSignal;
55
+ client?: Client;
56
+ onProgress?: (progress: McpToolProgress) => void;
57
+ } = {}
58
+ ): Promise<McpToolTextResponse> {
59
+ const normalizedArgs =
60
+ typeof args === "object" && args !== null && !Array.isArray(args)
61
+ ? (args as Record<string, unknown>)
62
+ : undefined;
63
+
64
+ // Get a (possibly pooled) client. The client itself was connected with a signal
65
+ // that already composes outer cancellation. We still enforce a per-call timeout here.
66
+ let activeClient = client ?? (await getClient(server, signal));
67
+
68
+ const callToolOptions = {
69
+ signal,
70
+ timeout: timeoutMs,
71
+ // Enable progress tokens so long-running tools keep extending the timeout.
72
+ onprogress: (progress: McpToolProgress) => {
73
+ onProgress?.({
74
+ progress: progress.progress,
75
+ total: progress.total,
76
+ message: progress.message,
77
+ });
78
+ },
79
+ resetTimeoutOnProgress: true,
80
+ };
81
+
82
+ let response;
83
+ try {
84
+ response = await activeClient.callTool(
85
+ { name: tool, arguments: normalizedArgs },
86
+ undefined,
87
+ callToolOptions
88
+ );
89
+ } catch (err) {
90
+ if (!isConnectionClosedError(err)) {
91
+ throw err;
92
+ }
93
+
94
+ // Evict stale client and close it
95
+ const stale = evictFromPool(server);
96
+ stale?.close?.().catch(() => {});
97
+
98
+ // Retry with fresh client
99
+ activeClient = await getClient(server, signal);
100
+ response = await activeClient.callTool(
101
+ { name: tool, arguments: normalizedArgs },
102
+ undefined,
103
+ callToolOptions
104
+ );
105
+ }
106
+
107
+ const parts = Array.isArray(response?.content) ? (response.content as Array<unknown>) : [];
108
+ const textParts = parts
109
+ .filter((part): part is { type: "text"; text: string } => {
110
+ if (typeof part !== "object" || part === null) return false;
111
+ const obj = part as Record<string, unknown>;
112
+ return obj["type"] === "text" && typeof obj["text"] === "string";
113
+ })
114
+ .map((p) => p.text);
115
+
116
+ const text = textParts.join("\n");
117
+ const structured = (response as unknown as { structuredContent?: unknown })?.structuredContent;
118
+ const contentBlocks = Array.isArray(response?.content)
119
+ ? (response.content as unknown[])
120
+ : undefined;
121
+ return { text, structured, content: contentBlocks };
122
+ }
@@ -0,0 +1,76 @@
1
+ import { config } from "$lib/server/config";
2
+ import { logger } from "$lib/server/logger";
3
+ import type { McpServerConfig } from "./httpClient";
4
+ import { resetMcpToolsCache } from "./tools";
5
+
6
+ let cachedRaw: string | null = null;
7
+ let cachedServers: McpServerConfig[] = [];
8
+
9
+ function parseServers(raw: string): McpServerConfig[] {
10
+ if (!raw) return [];
11
+
12
+ try {
13
+ const parsed = JSON.parse(raw);
14
+ if (!Array.isArray(parsed)) return [];
15
+
16
+ return parsed
17
+ .map((entry) => {
18
+ if (!entry || typeof entry !== "object") return undefined;
19
+ const name = (entry as Record<string, unknown>).name;
20
+ const url = (entry as Record<string, unknown>).url;
21
+ if (typeof name !== "string" || !name.trim()) return undefined;
22
+ if (typeof url !== "string" || !url.trim()) return undefined;
23
+
24
+ const headersRaw = (entry as Record<string, unknown>).headers;
25
+ let headers: Record<string, string> | undefined;
26
+ if (headersRaw && typeof headersRaw === "object" && !Array.isArray(headersRaw)) {
27
+ const headerEntries = Object.entries(headersRaw as Record<string, unknown>).filter(
28
+ (entry): entry is [string, string] => typeof entry[1] === "string"
29
+ );
30
+ headers = Object.fromEntries(headerEntries);
31
+ }
32
+
33
+ return headers ? { name, url, headers } : { name, url };
34
+ })
35
+ .filter((server): server is McpServerConfig => Boolean(server));
36
+ } catch (error) {
37
+ logger.warn({ err: error }, "[mcp] failed to parse MCP_SERVERS env");
38
+ return [];
39
+ }
40
+ }
41
+
42
+ function setServers(raw: string) {
43
+ cachedServers = parseServers(raw);
44
+ cachedRaw = raw;
45
+ resetMcpToolsCache();
46
+ logger.debug({ count: cachedServers.length }, "[mcp] loaded server configuration");
47
+ console.log(
48
+ `[MCP] Loaded ${cachedServers.length} server(s):`,
49
+ cachedServers.map((s) => s.name).join(", ") || "none"
50
+ );
51
+ }
52
+
53
+ export function loadMcpServersOnStartup(): McpServerConfig[] {
54
+ const raw = config.MCP_SERVERS || "[]";
55
+ setServers(raw);
56
+ return cachedServers;
57
+ }
58
+
59
+ export function refreshMcpServersIfChanged(): void {
60
+ const currentRaw = config.MCP_SERVERS || "[]";
61
+ if (cachedRaw === null) {
62
+ setServers(currentRaw);
63
+ return;
64
+ }
65
+
66
+ if (currentRaw !== cachedRaw) {
67
+ setServers(currentRaw);
68
+ }
69
+ }
70
+
71
+ export function getMcpServers(): McpServerConfig[] {
72
+ if (cachedRaw === null) {
73
+ loadMcpServersOnStartup();
74
+ }
75
+ return cachedServers;
76
+ }
@@ -0,0 +1,196 @@
1
+ import { Client } from "@modelcontextprotocol/sdk/client";
2
+ import { StreamableHTTPClientTransport } from "@modelcontextprotocol/sdk/client/streamableHttp.js";
3
+ import { SSEClientTransport } from "@modelcontextprotocol/sdk/client/sse.js";
4
+ import type { McpServerConfig } from "./httpClient";
5
+ import { logger } from "$lib/server/logger";
6
+ // use console.* for lightweight diagnostics in production logs
7
+
8
+ export type OpenAiTool = {
9
+ type: "function";
10
+ function: { name: string; description?: string; parameters?: Record<string, unknown> };
11
+ };
12
+
13
+ export interface McpToolMapping {
14
+ fnName: string;
15
+ server: string;
16
+ tool: string;
17
+ }
18
+
19
+ interface CacheEntry {
20
+ fetchedAt: number;
21
+ ttlMs: number;
22
+ tools: OpenAiTool[];
23
+ mapping: Record<string, McpToolMapping>;
24
+ }
25
+
26
+ const DEFAULT_TTL_MS = 60_000;
27
+ const cache = new Map<string, CacheEntry>();
28
+
29
+ // Per OpenAI tool/function name guidelines most providers enforce:
30
+ // ^[a-zA-Z0-9_-]{1,64}$
31
+ // Dots are not universally accepted (e.g., MiniMax via HF router rejects them).
32
+ // Normalize any disallowed characters (including ".") to underscore and trim to 64 chars.
33
+ function sanitizeName(name: string) {
34
+ return name.replace(/[^a-zA-Z0-9_-]/g, "_").slice(0, 64);
35
+ }
36
+
37
+ function buildCacheKey(servers: McpServerConfig[]): string {
38
+ const normalized = servers
39
+ .map((server) => ({
40
+ name: server.name,
41
+ url: server.url,
42
+ headers: server.headers
43
+ ? Object.entries(server.headers)
44
+ .sort(([a], [b]) => a.localeCompare(b))
45
+ .map(([key, value]) => [key, value])
46
+ : [],
47
+ }))
48
+ .sort((a, b) => {
49
+ const byName = a.name.localeCompare(b.name);
50
+ if (byName !== 0) return byName;
51
+ return a.url.localeCompare(b.url);
52
+ });
53
+
54
+ return JSON.stringify(normalized);
55
+ }
56
+
57
+ type ListedTool = {
58
+ name?: string;
59
+ inputSchema?: Record<string, unknown>;
60
+ description?: string;
61
+ annotations?: { title?: string };
62
+ };
63
+
64
+ async function listServerTools(
65
+ server: McpServerConfig,
66
+ opts: { signal?: AbortSignal } = {}
67
+ ): Promise<ListedTool[]> {
68
+ const url = new URL(server.url);
69
+ const client = new Client({ name: "chat-ui-mcp", version: "0.1.0" });
70
+ try {
71
+ try {
72
+ const transport = new StreamableHTTPClientTransport(url, {
73
+ requestInit: { headers: server.headers, signal: opts.signal },
74
+ });
75
+ await client.connect(transport);
76
+ } catch {
77
+ const transport = new SSEClientTransport(url, {
78
+ requestInit: { headers: server.headers, signal: opts.signal },
79
+ });
80
+ await client.connect(transport);
81
+ }
82
+
83
+ const response = await client.listTools({});
84
+ const tools = Array.isArray(response?.tools) ? (response.tools as ListedTool[]) : [];
85
+ try {
86
+ logger.debug(
87
+ {
88
+ server: server.name,
89
+ url: server.url,
90
+ count: tools.length,
91
+ toolNames: tools.map((t) => t?.name).filter(Boolean),
92
+ },
93
+ "[mcp] listed tools from server"
94
+ );
95
+ } catch {}
96
+ return tools;
97
+ } finally {
98
+ try {
99
+ await client.close?.();
100
+ } catch {
101
+ // ignore close errors
102
+ }
103
+ }
104
+ }
105
+
106
+ export async function getOpenAiToolsForMcp(
107
+ servers: McpServerConfig[],
108
+ { ttlMs = DEFAULT_TTL_MS, signal }: { ttlMs?: number; signal?: AbortSignal } = {}
109
+ ): Promise<{ tools: OpenAiTool[]; mapping: Record<string, McpToolMapping> }> {
110
+ const now = Date.now();
111
+ const cacheKey = buildCacheKey(servers);
112
+ const cached = cache.get(cacheKey);
113
+ if (cached && now - cached.fetchedAt < cached.ttlMs) {
114
+ return { tools: cached.tools, mapping: cached.mapping };
115
+ }
116
+
117
+ const tools: OpenAiTool[] = [];
118
+ const mapping: Record<string, McpToolMapping> = {};
119
+
120
+ const seenNames = new Set<string>();
121
+
122
+ const pushToolDefinition = (
123
+ name: string,
124
+ description: string | undefined,
125
+ parameters: Record<string, unknown> | undefined
126
+ ) => {
127
+ if (seenNames.has(name)) return;
128
+ tools.push({
129
+ type: "function",
130
+ function: {
131
+ name,
132
+ description,
133
+ parameters,
134
+ },
135
+ });
136
+ seenNames.add(name);
137
+ };
138
+
139
+ // Fetch tools in parallel; tolerate individual failures
140
+ const tasks = servers.map((server) => listServerTools(server, { signal }));
141
+ const results = await Promise.allSettled(tasks);
142
+
143
+ for (let i = 0; i < results.length; i++) {
144
+ const server = servers[i];
145
+ const r = results[i];
146
+ if (r.status === "fulfilled") {
147
+ const serverTools = r.value;
148
+ for (const tool of serverTools) {
149
+ if (typeof tool.name !== "string" || tool.name.trim().length === 0) {
150
+ continue;
151
+ }
152
+
153
+ const parameters =
154
+ tool.inputSchema && typeof tool.inputSchema === "object" ? tool.inputSchema : undefined;
155
+ const description = tool.description ?? tool.annotations?.title;
156
+ const toolName = tool.name;
157
+
158
+ // Emit a collision-aware function name.
159
+ // Prefer the plain tool name; on conflict, suffix with server name.
160
+ let plainName = sanitizeName(toolName);
161
+ if (plainName in mapping) {
162
+ const suffix = sanitizeName(server.name);
163
+ const candidate = `${plainName}_${suffix}`.slice(0, 64);
164
+ if (!(candidate in mapping)) {
165
+ plainName = candidate;
166
+ } else {
167
+ let i = 2;
168
+ let next = `${candidate}_${i}`;
169
+ while (i < 10 && next in mapping) {
170
+ i += 1;
171
+ next = `${candidate}_${i}`;
172
+ }
173
+ plainName = next.slice(0, 64);
174
+ }
175
+ }
176
+
177
+ pushToolDefinition(plainName, description, parameters);
178
+ mapping[plainName] = {
179
+ fnName: plainName,
180
+ server: server.name,
181
+ tool: toolName,
182
+ };
183
+ }
184
+ } else {
185
+ // ignore failure for this server
186
+ continue;
187
+ }
188
+ }
189
+
190
+ cache.set(cacheKey, { fetchedAt: now, ttlMs, tools, mapping });
191
+ return { tools, mapping };
192
+ }
193
+
194
+ export function resetMcpToolsCache() {
195
+ cache.clear();
196
+ }
@@ -0,0 +1,255 @@
1
+ import { collectDefaultMetrics, Counter, Registry, Summary } from "prom-client";
2
+ import { logger } from "$lib/server/logger";
3
+ import { config } from "$lib/server/config";
4
+ import { createServer, type Server as HttpServer } from "http";
5
+ import { onExit } from "./exitHandler";
6
+
7
+ type ModelLabel = "model";
8
+ type ToolLabel = "tool";
9
+
10
+ interface Metrics {
11
+ model: {
12
+ conversationsTotal: Counter<ModelLabel>;
13
+ messagesTotal: Counter<ModelLabel>;
14
+ tokenCountTotal: Counter<ModelLabel>;
15
+ timePerOutputToken: Summary<ModelLabel>;
16
+ timeToFirstToken: Summary<ModelLabel>;
17
+ latency: Summary<ModelLabel>;
18
+ votesPositive: Counter<ModelLabel>;
19
+ votesNegative: Counter<ModelLabel>;
20
+ };
21
+ webSearch: {
22
+ requestCount: Counter;
23
+ pageFetchCount: Counter;
24
+ pageFetchCountError: Counter;
25
+ pageFetchDuration: Summary;
26
+ embeddingDuration: Summary;
27
+ };
28
+ tool: {
29
+ toolUseCount: Counter<ToolLabel>;
30
+ toolUseCountError: Counter<ToolLabel>;
31
+ toolUseDuration: Summary<ToolLabel>;
32
+ timeToChooseTools: Summary<ModelLabel>;
33
+ };
34
+ }
35
+
36
+ export class MetricsServer {
37
+ private static instance: MetricsServer | undefined;
38
+ private readonly enabled: boolean;
39
+ private readonly register: Registry;
40
+ private readonly metrics: Metrics;
41
+ private httpServer: HttpServer | undefined;
42
+
43
+ private constructor() {
44
+ this.enabled = config.METRICS_ENABLED === "true";
45
+ this.register = new Registry();
46
+
47
+ if (this.enabled) {
48
+ collectDefaultMetrics({ register: this.register });
49
+ }
50
+
51
+ this.metrics = this.createMetrics();
52
+
53
+ if (this.enabled) {
54
+ this.startStandaloneServer();
55
+ }
56
+ }
57
+
58
+ public static getInstance(): MetricsServer {
59
+ if (!MetricsServer.instance) {
60
+ MetricsServer.instance = new MetricsServer();
61
+ }
62
+ return MetricsServer.instance;
63
+ }
64
+
65
+ public static getMetrics(): Metrics {
66
+ return MetricsServer.getInstance().metrics;
67
+ }
68
+
69
+ public static isEnabled(): boolean {
70
+ return config.METRICS_ENABLED === "true";
71
+ }
72
+
73
+ public async render(): Promise<string> {
74
+ if (!this.enabled) {
75
+ return "";
76
+ }
77
+
78
+ return this.register.metrics();
79
+ }
80
+
81
+ private createMetrics(): Metrics {
82
+ const labelNames: ModelLabel[] = ["model"];
83
+ const toolLabelNames: ToolLabel[] = ["tool"];
84
+
85
+ const noopRegistry = new Registry();
86
+
87
+ const registry = this.enabled ? this.register : noopRegistry;
88
+
89
+ return {
90
+ model: {
91
+ conversationsTotal: new Counter<ModelLabel>({
92
+ name: "model_conversations_total",
93
+ help: "Total number of conversations",
94
+ labelNames,
95
+ registers: [registry],
96
+ }),
97
+ messagesTotal: new Counter<ModelLabel>({
98
+ name: "model_messages_total",
99
+ help: "Total number of messages",
100
+ labelNames,
101
+ registers: [registry],
102
+ }),
103
+ tokenCountTotal: new Counter<ModelLabel>({
104
+ name: "model_token_count_total",
105
+ help: "Total number of tokens emitted by the model",
106
+ labelNames,
107
+ registers: [registry],
108
+ }),
109
+ timePerOutputToken: new Summary<ModelLabel>({
110
+ name: "model_time_per_output_token_ms",
111
+ help: "Per-token latency in milliseconds",
112
+ labelNames,
113
+ registers: [registry],
114
+ maxAgeSeconds: 5 * 60,
115
+ ageBuckets: 5,
116
+ }),
117
+ timeToFirstToken: new Summary<ModelLabel>({
118
+ name: "model_time_to_first_token_ms",
119
+ help: "Time to first token in milliseconds",
120
+ labelNames,
121
+ registers: [registry],
122
+ maxAgeSeconds: 5 * 60,
123
+ ageBuckets: 5,
124
+ }),
125
+ latency: new Summary<ModelLabel>({
126
+ name: "model_latency_ms",
127
+ help: "Total time to complete a response in milliseconds",
128
+ labelNames,
129
+ registers: [registry],
130
+ maxAgeSeconds: 5 * 60,
131
+ ageBuckets: 5,
132
+ }),
133
+ votesPositive: new Counter<ModelLabel>({
134
+ name: "model_votes_positive_total",
135
+ help: "Total number of positive votes on model messages",
136
+ labelNames,
137
+ registers: [registry],
138
+ }),
139
+ votesNegative: new Counter<ModelLabel>({
140
+ name: "model_votes_negative_total",
141
+ help: "Total number of negative votes on model messages",
142
+ labelNames,
143
+ registers: [registry],
144
+ }),
145
+ },
146
+ webSearch: {
147
+ requestCount: new Counter({
148
+ name: "web_search_request_count",
149
+ help: "Total number of web search requests",
150
+ registers: [registry],
151
+ }),
152
+ pageFetchCount: new Counter({
153
+ name: "web_search_page_fetch_count",
154
+ help: "Total number of web search page fetches",
155
+ registers: [registry],
156
+ }),
157
+ pageFetchCountError: new Counter({
158
+ name: "web_search_page_fetch_count_error",
159
+ help: "Total number of web search page fetch errors",
160
+ registers: [registry],
161
+ }),
162
+ pageFetchDuration: new Summary({
163
+ name: "web_search_page_fetch_duration_ms",
164
+ help: "Duration of web search page fetches in milliseconds",
165
+ registers: [registry],
166
+ maxAgeSeconds: 5 * 60,
167
+ ageBuckets: 5,
168
+ }),
169
+ embeddingDuration: new Summary({
170
+ name: "web_search_embedding_duration_ms",
171
+ help: "Duration of web search embeddings in milliseconds",
172
+ registers: [registry],
173
+ maxAgeSeconds: 5 * 60,
174
+ ageBuckets: 5,
175
+ }),
176
+ },
177
+ tool: {
178
+ toolUseCount: new Counter<ToolLabel>({
179
+ name: "tool_use_count",
180
+ help: "Total number of tool invocations",
181
+ labelNames: toolLabelNames,
182
+ registers: [registry],
183
+ }),
184
+ toolUseCountError: new Counter<ToolLabel>({
185
+ name: "tool_use_count_error",
186
+ help: "Total number of tool invocation errors",
187
+ labelNames: toolLabelNames,
188
+ registers: [registry],
189
+ }),
190
+ toolUseDuration: new Summary<ToolLabel>({
191
+ name: "tool_use_duration_ms",
192
+ help: "Duration of tool invocations in milliseconds",
193
+ labelNames: toolLabelNames,
194
+ registers: [registry],
195
+ maxAgeSeconds: 30 * 60,
196
+ ageBuckets: 5,
197
+ }),
198
+ timeToChooseTools: new Summary<ModelLabel>({
199
+ name: "time_to_choose_tools_ms",
200
+ help: "Time spent selecting tools in milliseconds",
201
+ labelNames,
202
+ registers: [registry],
203
+ maxAgeSeconds: 5 * 60,
204
+ ageBuckets: 5,
205
+ }),
206
+ },
207
+ };
208
+ }
209
+
210
+ private startStandaloneServer() {
211
+ const port = Number(config.METRICS_PORT || "5565");
212
+
213
+ if (!Number.isInteger(port) || port < 0 || port > 65535) {
214
+ logger.warn(`Invalid METRICS_PORT value: ${config.METRICS_PORT}`);
215
+ return;
216
+ }
217
+
218
+ this.httpServer = createServer(async (req, res) => {
219
+ if (req.method !== "GET") {
220
+ res.statusCode = 405;
221
+ res.end("Method Not Allowed");
222
+ return;
223
+ }
224
+
225
+ try {
226
+ const payload = await this.render();
227
+ res.setHeader("Content-Type", "text/plain; version=0.0.4");
228
+ res.end(payload);
229
+ } catch (error) {
230
+ logger.error(error, "Failed to render metrics");
231
+ res.statusCode = 500;
232
+ res.end("Failed to render metrics");
233
+ }
234
+ });
235
+
236
+ this.httpServer.listen(port, () => {
237
+ logger.info(`Metrics server listening on port ${port}`);
238
+ });
239
+
240
+ onExit(async () => {
241
+ if (!this.httpServer) return;
242
+ logger.info("Shutting down metrics server...");
243
+ await new Promise<void>((resolve, reject) => {
244
+ this.httpServer?.close((err) => {
245
+ if (err) {
246
+ reject(err);
247
+ return;
248
+ }
249
+ resolve();
250
+ });
251
+ }).catch((error) => logger.error(error, "Failed to close metrics server"));
252
+ this.httpServer = undefined;
253
+ });
254
+ }
255
+ }