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,518 @@
1
+ import { config } from "$lib/server/config";
2
+ import type { ChatTemplateInput } from "$lib/types/Template";
3
+ import { z } from "zod";
4
+ import endpoints, { endpointSchema, type Endpoint } from "./endpoints/endpoints";
5
+
6
+ import JSON5 from "json5";
7
+ import { logger } from "$lib/server/logger";
8
+ import { makeRouterEndpoint } from "$lib/server/router/endpoint";
9
+
10
+ type Optional<T, K extends keyof T> = Pick<Partial<T>, K> & Omit<T, K>;
11
+
12
+ const sanitizeJSONEnv = (val: string, fallback: string) => {
13
+ const raw = (val ?? "").trim();
14
+ const unquoted = raw.startsWith("`") && raw.endsWith("`") ? raw.slice(1, -1) : raw;
15
+ return unquoted || fallback;
16
+ };
17
+
18
+ const modelConfig = z.object({
19
+ /** Used as an identifier in DB */
20
+ id: z.string().optional(),
21
+ /** Used to link to the model page, and for inference */
22
+ name: z.string().default(""),
23
+ displayName: z.string().min(1).optional(),
24
+ description: z.string().min(1).optional(),
25
+ logoUrl: z.string().url().optional(),
26
+ websiteUrl: z.string().url().optional(),
27
+ modelUrl: z.string().url().optional(),
28
+ tokenizer: z.never().optional(),
29
+ datasetName: z.string().min(1).optional(),
30
+ datasetUrl: z.string().url().optional(),
31
+ preprompt: z.string().default(""),
32
+ prepromptUrl: z.string().url().optional(),
33
+ chatPromptTemplate: z.never().optional(),
34
+ promptExamples: z
35
+ .array(
36
+ z.object({
37
+ title: z.string().min(1),
38
+ prompt: z.string().min(1),
39
+ })
40
+ )
41
+ .optional(),
42
+ endpoints: z.array(endpointSchema).optional(),
43
+ providers: z.array(z.object({ supports_tools: z.boolean().optional() }).passthrough()).optional(),
44
+ parameters: z
45
+ .object({
46
+ temperature: z.number().min(0).max(2).optional(),
47
+ truncate: z.number().int().positive().optional(),
48
+ max_tokens: z.number().int().positive().optional(),
49
+ stop: z.array(z.string()).optional(),
50
+ top_p: z.number().positive().optional(),
51
+ top_k: z.number().positive().optional(),
52
+ frequency_penalty: z.number().min(-2).max(2).optional(),
53
+ presence_penalty: z.number().min(-2).max(2).optional(),
54
+ })
55
+ .passthrough()
56
+ .optional(),
57
+ multimodal: z.boolean().default(false),
58
+ multimodalAcceptedMimetypes: z.array(z.string()).optional(),
59
+ // Aggregated tool-calling capability across providers (HF router)
60
+ supportsTools: z.boolean().default(false),
61
+ unlisted: z.boolean().default(false),
62
+ embeddingModel: z.never().optional(),
63
+ /** Used to enable/disable system prompt usage */
64
+ systemRoleSupported: z.boolean().default(true),
65
+ });
66
+
67
+ type ModelConfig = z.infer<typeof modelConfig>;
68
+
69
+ const overrideEntrySchema = modelConfig
70
+ .partial()
71
+ .extend({
72
+ id: z.string().optional(),
73
+ name: z.string().optional(),
74
+ })
75
+ .refine((value) => Boolean((value.id ?? value.name)?.trim()), {
76
+ message: "Model override entry must provide an id or name",
77
+ });
78
+
79
+ type ModelOverride = z.infer<typeof overrideEntrySchema>;
80
+
81
+ const openaiBaseUrl = config.OPENAI_BASE_URL
82
+ ? config.OPENAI_BASE_URL.replace(/\/$/, "")
83
+ : undefined;
84
+ const isHFRouter = openaiBaseUrl === "https://router.huggingface.co/v1";
85
+
86
+ const listSchema = z
87
+ .object({
88
+ data: z.array(
89
+ z.object({
90
+ id: z.string(),
91
+ description: z.string().optional(),
92
+ providers: z
93
+ .array(z.object({ supports_tools: z.boolean().optional() }).passthrough())
94
+ .optional(),
95
+ architecture: z
96
+ .object({
97
+ input_modalities: z.array(z.string()).optional(),
98
+ })
99
+ .passthrough()
100
+ .optional(),
101
+ })
102
+ ),
103
+ })
104
+ .passthrough();
105
+
106
+ function getChatPromptRender(_m: ModelConfig): (inputs: ChatTemplateInput) => string {
107
+ // Minimal template to support legacy "completions" flow if ever used.
108
+ // We avoid any tokenizer/Jinja usage in this build.
109
+ return ({ messages, preprompt }) => {
110
+ const parts: string[] = [];
111
+ if (preprompt) parts.push(`[SYSTEM]\n${preprompt}`);
112
+ for (const msg of messages) {
113
+ const role = msg.from === "assistant" ? "ASSISTANT" : msg.from.toUpperCase();
114
+ parts.push(`[${role}]\n${msg.content}`);
115
+ }
116
+ parts.push(`[ASSISTANT]`);
117
+ return parts.join("\n\n");
118
+ };
119
+ }
120
+
121
+ const processModel = async (m: ModelConfig) => ({
122
+ ...m,
123
+ chatPromptRender: await getChatPromptRender(m),
124
+ id: m.id || m.name,
125
+ displayName: m.displayName || m.name,
126
+ preprompt: m.prepromptUrl ? await fetch(m.prepromptUrl).then((r) => r.text()) : m.preprompt,
127
+ parameters: { ...m.parameters, stop_sequences: m.parameters?.stop },
128
+ unlisted: m.unlisted ?? false,
129
+ });
130
+
131
+ const addEndpoint = (m: Awaited<ReturnType<typeof processModel>>) => ({
132
+ ...m,
133
+ getEndpoint: async (): Promise<Endpoint> => {
134
+ if (!m.endpoints || m.endpoints.length === 0) {
135
+ throw new Error("No endpoints configured. This build requires OpenAI-compatible endpoints.");
136
+ }
137
+ // Only support OpenAI-compatible endpoints in this build
138
+ const endpoint = m.endpoints[0];
139
+ if (endpoint.type !== "openai") {
140
+ throw new Error("Only 'openai' endpoint type is supported in this build");
141
+ }
142
+ return await endpoints.openai({ ...endpoint, model: m });
143
+ },
144
+ });
145
+
146
+ type InternalProcessedModel = Awaited<ReturnType<typeof addEndpoint>> & {
147
+ isRouter: boolean;
148
+ hasInferenceAPI: boolean;
149
+ };
150
+
151
+ const inferenceApiIds: string[] = [];
152
+
153
+ const getModelOverrides = (): ModelOverride[] => {
154
+ const overridesEnv = (Reflect.get(config, "MODELS") as string | undefined) ?? "";
155
+
156
+ if (!overridesEnv.trim()) {
157
+ return [];
158
+ }
159
+
160
+ try {
161
+ return z.array(overrideEntrySchema).parse(JSON5.parse(sanitizeJSONEnv(overridesEnv, "[]")));
162
+ } catch (error) {
163
+ logger.error(error, "[models] Failed to parse MODELS overrides");
164
+ return [];
165
+ }
166
+ };
167
+
168
+ export type ModelsRefreshSummary = {
169
+ refreshedAt: Date;
170
+ durationMs: number;
171
+ added: string[];
172
+ removed: string[];
173
+ changed: string[];
174
+ total: number;
175
+ };
176
+
177
+ export type ProcessedModel = InternalProcessedModel;
178
+
179
+ export let models: ProcessedModel[] = [];
180
+ export let defaultModel!: ProcessedModel;
181
+ export let taskModel!: ProcessedModel;
182
+ export let validModelIdSchema: z.ZodType<string> = z.string();
183
+ export let lastModelRefresh = new Date(0);
184
+ export let lastModelRefreshDurationMs = 0;
185
+ export let lastModelRefreshSummary: ModelsRefreshSummary = {
186
+ refreshedAt: new Date(0),
187
+ durationMs: 0,
188
+ added: [],
189
+ removed: [],
190
+ changed: [],
191
+ total: 0,
192
+ };
193
+
194
+ let inflightRefresh: Promise<ModelsRefreshSummary> | null = null;
195
+
196
+ const createValidModelIdSchema = (modelList: ProcessedModel[]): z.ZodType<string> => {
197
+ if (modelList.length === 0) {
198
+ throw new Error("No models available to build validation schema");
199
+ }
200
+ const ids = new Set(modelList.map((m) => m.id));
201
+ return z.string().refine((value) => ids.has(value), "Invalid model id");
202
+ };
203
+
204
+ const resolveTaskModel = (modelList: ProcessedModel[]) => {
205
+ if (modelList.length === 0) {
206
+ throw new Error("No models available to select task model");
207
+ }
208
+
209
+ if (config.TASK_MODEL) {
210
+ const preferred = modelList.find(
211
+ (m) => m.name === config.TASK_MODEL || m.id === config.TASK_MODEL
212
+ );
213
+ if (preferred) {
214
+ return preferred;
215
+ }
216
+ }
217
+
218
+ return modelList[0];
219
+ };
220
+
221
+ const signatureForModel = (model: ProcessedModel) =>
222
+ JSON.stringify({
223
+ description: model.description,
224
+ displayName: model.displayName,
225
+ providers: model.providers,
226
+ parameters: model.parameters,
227
+ preprompt: model.preprompt,
228
+ prepromptUrl: model.prepromptUrl,
229
+ endpoints:
230
+ model.endpoints?.map((endpoint) => {
231
+ if (endpoint.type === "openai") {
232
+ const { type, baseURL } = endpoint;
233
+ return { type, baseURL };
234
+ }
235
+ return { type: endpoint.type };
236
+ }) ?? null,
237
+ multimodal: model.multimodal,
238
+ multimodalAcceptedMimetypes: model.multimodalAcceptedMimetypes,
239
+ supportsTools: (model as unknown as { supportsTools?: boolean }).supportsTools ?? false,
240
+ isRouter: model.isRouter,
241
+ hasInferenceAPI: model.hasInferenceAPI,
242
+ });
243
+
244
+ const applyModelState = (newModels: ProcessedModel[], startedAt: number): ModelsRefreshSummary => {
245
+ if (newModels.length === 0) {
246
+ throw new Error("Failed to load any models from upstream");
247
+ }
248
+
249
+ const previousIds = new Set(models.map((m) => m.id));
250
+ const previousSignatures = new Map(models.map((m) => [m.id, signatureForModel(m)]));
251
+ const refreshedAt = new Date();
252
+ const durationMs = Date.now() - startedAt;
253
+
254
+ models = newModels;
255
+ defaultModel = models[0];
256
+ taskModel = resolveTaskModel(models);
257
+ validModelIdSchema = createValidModelIdSchema(models);
258
+ lastModelRefresh = refreshedAt;
259
+ lastModelRefreshDurationMs = durationMs;
260
+
261
+ const added = newModels.map((m) => m.id).filter((id) => !previousIds.has(id));
262
+ const removed = Array.from(previousIds).filter(
263
+ (id) => !newModels.some((model) => model.id === id)
264
+ );
265
+ const changed = newModels
266
+ .filter((model) => {
267
+ const previousSignature = previousSignatures.get(model.id);
268
+ return previousSignature !== undefined && previousSignature !== signatureForModel(model);
269
+ })
270
+ .map((model) => model.id);
271
+
272
+ const summary: ModelsRefreshSummary = {
273
+ refreshedAt,
274
+ durationMs,
275
+ added,
276
+ removed,
277
+ changed,
278
+ total: models.length,
279
+ };
280
+
281
+ lastModelRefreshSummary = summary;
282
+
283
+ logger.info(
284
+ {
285
+ total: summary.total,
286
+ added: summary.added,
287
+ removed: summary.removed,
288
+ changed: summary.changed,
289
+ durationMs: summary.durationMs,
290
+ },
291
+ "[models] Model cache refreshed"
292
+ );
293
+
294
+ return summary;
295
+ };
296
+
297
+ const buildModels = async (): Promise<ProcessedModel[]> => {
298
+ if (!openaiBaseUrl) {
299
+ logger.error(
300
+ "OPENAI_BASE_URL is required. Set it to an OpenAI-compatible base (e.g., https://router.huggingface.co/v1)."
301
+ );
302
+ throw new Error("OPENAI_BASE_URL not set");
303
+ }
304
+
305
+ try {
306
+ const baseURL = openaiBaseUrl;
307
+ logger.info({ baseURL }, "[models] Using OpenAI-compatible base URL");
308
+
309
+ // Canonical auth token is OPENAI_API_KEY; keep HF_TOKEN as legacy alias
310
+ const authToken = config.OPENAI_API_KEY || config.HF_TOKEN;
311
+
312
+ // Use auth token from the start if available to avoid rate limiting issues
313
+ // Some APIs rate-limit unauthenticated requests more aggressively
314
+ const response = await fetch(`${baseURL}/models`, {
315
+ headers: authToken ? { Authorization: `Bearer ${authToken}` } : undefined,
316
+ });
317
+ logger.info({ status: response.status }, "[models] First fetch status");
318
+ if (!response.ok && response.status === 401 && !authToken) {
319
+ // If we get 401 and didn't have a token, there's nothing we can do
320
+ throw new Error(
321
+ `Failed to fetch ${baseURL}/models: ${response.status} ${response.statusText} (no auth token available)`
322
+ );
323
+ }
324
+ if (!response.ok) {
325
+ throw new Error(
326
+ `Failed to fetch ${baseURL}/models: ${response.status} ${response.statusText}`
327
+ );
328
+ }
329
+ const json = await response.json();
330
+ logger.info({ keys: Object.keys(json || {}) }, "[models] Response keys");
331
+
332
+ const parsed = listSchema.parse(json);
333
+ logger.info({ count: parsed.data.length }, "[models] Parsed models count");
334
+
335
+ let modelsRaw = parsed.data.map((m) => {
336
+ let logoUrl: string | undefined = undefined;
337
+ if (isHFRouter && m.id.includes("/")) {
338
+ const org = m.id.split("/")[0];
339
+ logoUrl = `https://huggingface.co/api/avatars/${encodeURIComponent(org)}`;
340
+ }
341
+
342
+ const inputModalities = (m.architecture?.input_modalities ?? []).map((modality) =>
343
+ modality.toLowerCase()
344
+ );
345
+ const supportsImageInput =
346
+ inputModalities.includes("image") || inputModalities.includes("vision");
347
+
348
+ // If any provider supports tools, consider the model as supporting tools
349
+ const supportsTools = Boolean((m.providers ?? []).some((p) => p?.supports_tools === true));
350
+ return {
351
+ id: m.id,
352
+ name: m.id,
353
+ displayName: m.id,
354
+ description: m.description,
355
+ logoUrl,
356
+ providers: m.providers,
357
+ multimodal: supportsImageInput,
358
+ multimodalAcceptedMimetypes: supportsImageInput ? ["image/*"] : undefined,
359
+ supportsTools,
360
+ endpoints: [
361
+ {
362
+ type: "openai" as const,
363
+ baseURL,
364
+ // apiKey will be taken from OPENAI_API_KEY or HF_TOKEN automatically
365
+ },
366
+ ],
367
+ } as ModelConfig;
368
+ }) as ModelConfig[];
369
+
370
+ const overrides = getModelOverrides();
371
+
372
+ if (overrides.length) {
373
+ const overrideMap = new Map<string, ModelOverride>();
374
+ for (const override of overrides) {
375
+ for (const key of [override.id, override.name]) {
376
+ const trimmed = key?.trim();
377
+ if (trimmed) overrideMap.set(trimmed, override);
378
+ }
379
+ }
380
+
381
+ // Filter to only configured models and apply overrides, preserving MODELS order
382
+ const filteredAndOrdered: ModelConfig[] = [];
383
+ for (const override of overrides) {
384
+ const matchKey = override.name?.trim() || override.id?.trim() || "";
385
+ const found = modelsRaw.find(
386
+ (model) => model.id === matchKey || model.name === matchKey
387
+ );
388
+ if (found) {
389
+ const { id, name, ...rest } = override;
390
+ void id;
391
+ void name;
392
+ filteredAndOrdered.push({ ...found, ...rest });
393
+ }
394
+ }
395
+
396
+ // If we matched at least one, use filtered list; otherwise fall back to all models with overrides
397
+ if (filteredAndOrdered.length > 0) {
398
+ modelsRaw = filteredAndOrdered;
399
+ } else {
400
+ modelsRaw = modelsRaw.map((model) => {
401
+ const override = overrideMap.get(model.id ?? "") ?? overrideMap.get(model.name ?? "");
402
+ if (!override) return model;
403
+
404
+ const { id, name, ...rest } = override;
405
+ void id;
406
+ void name;
407
+
408
+ return {
409
+ ...model,
410
+ ...rest,
411
+ };
412
+ });
413
+ }
414
+ }
415
+
416
+ const builtModels = await Promise.all(
417
+ modelsRaw.map((e) =>
418
+ processModel(e)
419
+ .then(addEndpoint)
420
+ .then(async (m) => ({
421
+ ...m,
422
+ hasInferenceAPI: inferenceApiIds.includes(m.id ?? m.name),
423
+ // router decoration added later
424
+ isRouter: false as boolean,
425
+ }))
426
+ )
427
+ );
428
+
429
+ const archBase = (config.LLM_ROUTER_ARCH_BASE_URL || "").trim();
430
+ const routerLabel = (config.PUBLIC_LLM_ROUTER_DISPLAY_NAME || "Omni").trim() || "Omni";
431
+ const routerLogo = (config.PUBLIC_LLM_ROUTER_LOGO_URL || "").trim();
432
+ const routerAliasId = (config.PUBLIC_LLM_ROUTER_ALIAS_ID || "omni").trim() || "omni";
433
+ const routerMultimodalEnabled =
434
+ (config.LLM_ROUTER_ENABLE_MULTIMODAL || "").toLowerCase() === "true";
435
+ const routerToolsEnabled = (config.LLM_ROUTER_ENABLE_TOOLS || "").toLowerCase() === "true";
436
+
437
+ let decorated = builtModels as ProcessedModel[];
438
+
439
+ if (archBase) {
440
+ // Build a minimal model config for the alias
441
+ const aliasRaw = {
442
+ id: routerAliasId,
443
+ name: routerAliasId,
444
+ displayName: routerLabel,
445
+ description: "Automatically routes your messages to the best model for your request.",
446
+ logoUrl: routerLogo || undefined,
447
+ preprompt: "",
448
+ endpoints: [
449
+ {
450
+ type: "openai" as const,
451
+ baseURL: openaiBaseUrl,
452
+ },
453
+ ],
454
+ // Keep the alias visible
455
+ unlisted: false,
456
+ } as ModelConfig;
457
+
458
+ if (routerMultimodalEnabled) {
459
+ aliasRaw.multimodal = true;
460
+ aliasRaw.multimodalAcceptedMimetypes = ["image/*"];
461
+ }
462
+
463
+ if (routerToolsEnabled) {
464
+ aliasRaw.supportsTools = true;
465
+ }
466
+
467
+ const aliasBase = await processModel(aliasRaw);
468
+ // Create a self-referential ProcessedModel for the router endpoint
469
+ const aliasModel: ProcessedModel = {
470
+ ...aliasBase,
471
+ isRouter: true,
472
+ hasInferenceAPI: false,
473
+ // getEndpoint uses the router wrapper regardless of the endpoints array
474
+ getEndpoint: async (): Promise<Endpoint> => makeRouterEndpoint(aliasModel),
475
+ } as ProcessedModel;
476
+
477
+ // Put alias first
478
+ decorated = [aliasModel, ...decorated];
479
+ }
480
+
481
+ return decorated;
482
+ } catch (e) {
483
+ logger.error(e, "Failed to load models from OpenAI base URL");
484
+ throw e;
485
+ }
486
+ };
487
+
488
+ const rebuildModels = async (): Promise<ModelsRefreshSummary> => {
489
+ const startedAt = Date.now();
490
+ const newModels = await buildModels();
491
+ return applyModelState(newModels, startedAt);
492
+ };
493
+
494
+ await rebuildModels();
495
+
496
+ export const refreshModels = async (): Promise<ModelsRefreshSummary> => {
497
+ if (inflightRefresh) {
498
+ return inflightRefresh;
499
+ }
500
+
501
+ inflightRefresh = rebuildModels().finally(() => {
502
+ inflightRefresh = null;
503
+ });
504
+
505
+ return inflightRefresh;
506
+ };
507
+
508
+ export const validateModel = (_models: BackendModel[]) => {
509
+ // Zod enum function requires 2 parameters
510
+ return z.enum([_models[0].id, ..._models.slice(1).map((m) => m.id)]);
511
+ };
512
+
513
+ // if `TASK_MODEL` is string & name of a model in `MODELS`, then we use `MODELS[TASK_MODEL]`, else we try to parse `TASK_MODEL` as a model config itself
514
+
515
+ export type BackendModel = Optional<
516
+ typeof defaultModel,
517
+ "preprompt" | "parameters" | "multimodal" | "unlisted" | "hasInferenceAPI"
518
+ >;
@@ -0,0 +1,55 @@
1
+ import { AsyncLocalStorage } from "node:async_hooks";
2
+ import { randomUUID } from "node:crypto";
3
+
4
+ export interface RequestContext {
5
+ requestId: string;
6
+ url?: string;
7
+ ip?: string;
8
+ user?: string;
9
+ statusCode?: number;
10
+ }
11
+
12
+ const asyncLocalStorage = new AsyncLocalStorage<RequestContext>();
13
+
14
+ /**
15
+ * Run a function within a request context.
16
+ * All logs within this context will automatically include the requestId.
17
+ */
18
+ export function runWithRequestContext<T>(
19
+ fn: () => T,
20
+ context: Partial<RequestContext> & { requestId?: string } = {}
21
+ ): T {
22
+ const fullContext: RequestContext = {
23
+ requestId: context.requestId ?? randomUUID(),
24
+ url: context.url,
25
+ ip: context.ip,
26
+ user: context.user,
27
+ statusCode: context.statusCode,
28
+ };
29
+ return asyncLocalStorage.run(fullContext, fn);
30
+ }
31
+
32
+ /**
33
+ * Update the current request context with additional information.
34
+ * Useful for adding user information after authentication.
35
+ */
36
+ export function updateRequestContext(updates: Partial<Omit<RequestContext, "requestId">>): void {
37
+ const store = asyncLocalStorage.getStore();
38
+ if (store) {
39
+ Object.assign(store, updates);
40
+ }
41
+ }
42
+
43
+ /**
44
+ * Get the current request context, if any.
45
+ */
46
+ export function getRequestContext(): RequestContext | undefined {
47
+ return asyncLocalStorage.getStore();
48
+ }
49
+
50
+ /**
51
+ * Get the current request ID, or undefined if not in a request context.
52
+ */
53
+ export function getRequestId(): string | undefined {
54
+ return asyncLocalStorage.getStore()?.requestId;
55
+ }