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,19 @@
1
+ import type { Timestamps } from "./Timestamps";
2
+
3
+ export interface Semaphore extends Timestamps {
4
+ key: string;
5
+ deleteAt: Date;
6
+ }
7
+
8
+ export enum Semaphores {
9
+ CONVERSATION_STATS = "conversation.stats",
10
+ CONFIG_UPDATE = "config.update",
11
+ MIGRATION = "migration",
12
+ TEST_MIGRATION = "test.migration",
13
+ /**
14
+ * Note this lock name is used as `${Semaphores.OAUTH_TOKEN_REFRESH}:${sessionId}`
15
+ *
16
+ * not a global lock, but a lock for each session
17
+ */
18
+ OAUTH_TOKEN_REFRESH = "oauth.token.refresh",
19
+ }
@@ -0,0 +1,22 @@
1
+ import type { ObjectId } from "bson";
2
+ import type { Timestamps } from "./Timestamps";
3
+ import type { User } from "./User";
4
+
5
+ export interface Session extends Timestamps {
6
+ _id: ObjectId;
7
+ sessionId: string;
8
+ userId: User["_id"];
9
+ userAgent?: string;
10
+ ip?: string;
11
+ expiresAt: Date;
12
+ admin?: boolean;
13
+ coupledCookieHash?: string;
14
+
15
+ oauth?: {
16
+ token: {
17
+ value: string;
18
+ expiresAt: Date;
19
+ };
20
+ refreshToken?: string;
21
+ };
22
+ }
@@ -0,0 +1,86 @@
1
+ import { defaultModel } from "$lib/server/models";
2
+ import type { Timestamps } from "./Timestamps";
3
+ import type { User } from "./User";
4
+
5
+ export type StreamingMode = "raw" | "smooth";
6
+
7
+ export interface Settings extends Timestamps {
8
+ userId?: User["_id"];
9
+ sessionId?: string;
10
+
11
+ shareConversationsWithModelAuthors: boolean;
12
+ /** One-time welcome modal acknowledgement */
13
+ welcomeModalSeenAt?: Date | null;
14
+ activeModel: string;
15
+
16
+ // model name and system prompts
17
+ customPrompts?: Record<string, string>;
18
+
19
+ /**
20
+ * Per‑model overrides to enable multimodal (image) support
21
+ * even when not advertised by the provider/model list.
22
+ * Only the `true` value is meaningful (enables images).
23
+ */
24
+ multimodalOverrides?: Record<string, boolean>;
25
+
26
+ /**
27
+ * Per‑model overrides to enable tool calling (OpenAI tools/function calling)
28
+ * even when not advertised by the provider list. Only `true` is meaningful.
29
+ */
30
+ toolsOverrides?: Record<string, boolean>;
31
+
32
+ /**
33
+ * Per-model toggle to hide Omni prompt suggestions shown near the composer.
34
+ * When set to `true`, prompt examples for that model are suppressed.
35
+ */
36
+ hidePromptExamples?: Record<string, boolean>;
37
+
38
+ /**
39
+ * Per-model inference provider preference.
40
+ * Values: "auto" (default), "fastest", "cheapest", or a specific provider name (e.g., "together", "sambanova").
41
+ * The value is appended to the model ID when making inference requests (e.g., "model:fastest").
42
+ */
43
+ providerOverrides?: Record<string, string>;
44
+
45
+ /**
46
+ * Preferred assistant output behavior in the chat UI.
47
+ * - "raw": show provider-native stream chunks
48
+ * - "smooth": show smoothed stream chunks
49
+ */
50
+ streamingMode: StreamingMode;
51
+ directPaste: boolean;
52
+
53
+ /**
54
+ * Whether haptic feedback is enabled on supported touch devices.
55
+ * Uses the ios-haptics library for cross-platform vibration.
56
+ */
57
+ hapticsEnabled: boolean;
58
+
59
+ /**
60
+ * Autopilot mode — AI auto-continues after tool calls without user intervention.
61
+ * When enabled, the model loops through tool calls automatically up to maxSteps.
62
+ */
63
+ autopilotEnabled: boolean;
64
+
65
+ /**
66
+ * Organization to bill inference requests to (HuggingChat only).
67
+ * Stores the org's preferred_username. If empty/undefined, bills to personal account.
68
+ */
69
+ billingOrganization?: string;
70
+ }
71
+
72
+ export type SettingsEditable = Omit<Settings, "welcomeModalSeenAt" | "createdAt" | "updatedAt">;
73
+ // TODO: move this to a constant file along with other constants
74
+ export const DEFAULT_SETTINGS = {
75
+ shareConversationsWithModelAuthors: true,
76
+ activeModel: defaultModel.id,
77
+ customPrompts: {},
78
+ multimodalOverrides: {},
79
+ toolsOverrides: {},
80
+ hidePromptExamples: {},
81
+ providerOverrides: {},
82
+ streamingMode: "smooth",
83
+ directPaste: false,
84
+ hapticsEnabled: true,
85
+ autopilotEnabled: true,
86
+ } satisfies SettingsEditable;
@@ -0,0 +1,9 @@
1
+ import type { Conversation } from "./Conversation";
2
+
3
+ export type SharedConversation = Pick<
4
+ Conversation,
5
+ "model" | "title" | "rootMessageId" | "messages" | "preprompt" | "createdAt" | "updatedAt"
6
+ > & {
7
+ _id: string;
8
+ hash: string;
9
+ };
@@ -0,0 +1,6 @@
1
+ import type { Message } from "./Message";
2
+
3
+ export type ChatTemplateInput = {
4
+ messages: Pick<Message, "from" | "content" | "files">[];
5
+ preprompt?: string;
6
+ };
@@ -0,0 +1,4 @@
1
+ export interface Timestamps {
2
+ createdAt: Date;
3
+ updatedAt: Date;
4
+ }
@@ -0,0 +1,6 @@
1
+ import type { Timestamps } from "./Timestamps";
2
+
3
+ export interface TokenCache extends Timestamps {
4
+ tokenHash: string; // sha256 of the bearer token
5
+ userId: string; // the matching hf user id
6
+ }
@@ -0,0 +1,74 @@
1
+ export enum ToolResultStatus {
2
+ Success = "success",
3
+ Error = "error",
4
+ }
5
+
6
+ export interface ToolCall {
7
+ name: string;
8
+ parameters: Record<string, string | number | boolean>;
9
+ toolId?: string;
10
+ }
11
+
12
+ export interface ToolResultSuccess {
13
+ status: ToolResultStatus.Success;
14
+ call: ToolCall;
15
+ outputs: Record<string, unknown>[];
16
+ display?: boolean;
17
+ }
18
+
19
+ export interface ToolResultError {
20
+ status: ToolResultStatus.Error;
21
+ call: ToolCall;
22
+ message: string;
23
+ display?: boolean;
24
+ }
25
+
26
+ export type ToolResult = ToolResultSuccess | ToolResultError;
27
+
28
+ export interface ToolFront {
29
+ _id: string;
30
+ name: string;
31
+ displayName?: string;
32
+ description?: string;
33
+ color?: string;
34
+ icon?: string;
35
+ type?: "config" | "community";
36
+ isOnByDefault?: boolean;
37
+ isLocked?: boolean;
38
+ mimeTypes?: string[];
39
+ timeToUseMS?: number;
40
+ }
41
+
42
+ // MCP Server types
43
+ export interface KeyValuePair {
44
+ key: string;
45
+ value: string;
46
+ }
47
+
48
+ export type ServerStatus = "connected" | "connecting" | "disconnected" | "error";
49
+
50
+ export interface MCPTool {
51
+ name: string;
52
+ description?: string;
53
+ inputSchema?: unknown;
54
+ }
55
+
56
+ export interface MCPServer {
57
+ id: string;
58
+ name: string;
59
+ url: string;
60
+ type: "base" | "custom";
61
+ headers?: KeyValuePair[];
62
+ env?: KeyValuePair[];
63
+ status?: ServerStatus;
64
+ isLocked?: boolean;
65
+ tools?: MCPTool[];
66
+ errorMessage?: string;
67
+ // Indicates server reports or appears to require OAuth or other auth
68
+ authRequired?: boolean;
69
+ }
70
+
71
+ export interface MCPServerApi {
72
+ url: string;
73
+ headers?: KeyValuePair[];
74
+ }
@@ -0,0 +1,5 @@
1
+ /* eslint-disable no-shadow */
2
+ export enum UrlDependency {
3
+ ConversationList = "conversation:list",
4
+ Conversation = "conversation:id",
5
+ }
@@ -0,0 +1,14 @@
1
+ import type { ObjectId } from "mongodb";
2
+ import type { Timestamps } from "./Timestamps";
3
+
4
+ export interface User extends Timestamps {
5
+ _id: ObjectId;
6
+
7
+ username?: string;
8
+ name: string;
9
+ email?: string;
10
+ avatarUrl: string | undefined;
11
+ hfUserId: string;
12
+ isAdmin?: boolean;
13
+ isEarlyAccess?: boolean;
14
+ }
@@ -0,0 +1,75 @@
1
+ import type { env as publicEnv } from "$env/dynamic/public";
2
+ import { page } from "$app/state";
3
+ import { base } from "$app/paths";
4
+
5
+ import type { Transporter } from "@sveltejs/kit";
6
+ import { getContext } from "svelte";
7
+
8
+ type PublicConfigKey = keyof typeof publicEnv;
9
+
10
+ class PublicConfigManager {
11
+ #configStore = $state<Record<PublicConfigKey, string>>({});
12
+
13
+ constructor(initialConfig?: Record<PublicConfigKey, string>) {
14
+ this.init = this.init.bind(this);
15
+ this.getPublicConfig = this.getPublicConfig.bind(this);
16
+ if (initialConfig) {
17
+ this.init(initialConfig);
18
+ }
19
+ }
20
+
21
+ init(publicConfig: Record<PublicConfigKey, string>) {
22
+ this.#configStore = publicConfig;
23
+ }
24
+
25
+ get(key: PublicConfigKey) {
26
+ return this.#configStore[key];
27
+ }
28
+
29
+ getPublicConfig() {
30
+ return this.#configStore;
31
+ }
32
+
33
+ get isHuggingChat() {
34
+ return this.#configStore.PUBLIC_APP_ASSETS === "huggingchat";
35
+ }
36
+
37
+ get assetPath() {
38
+ // Use relative path when PUBLIC_ORIGIN is empty (avoids cross-origin issues
39
+ // when accessed via port-forwards or reverse proxies)
40
+ const origin = this.#configStore.PUBLIC_ORIGIN || "";
41
+ return origin + base + "/" + (this.#configStore.PUBLIC_APP_ASSETS || "chatui");
42
+ }
43
+ }
44
+ type ConfigProxy = PublicConfigManager & { [K in PublicConfigKey]: string };
45
+
46
+ export function getConfigManager(initialConfig?: Record<PublicConfigKey, string>) {
47
+ const publicConfigManager = new PublicConfigManager(initialConfig);
48
+
49
+ const publicConfig: ConfigProxy = new Proxy(publicConfigManager, {
50
+ get(target, prop) {
51
+ if (prop in target) {
52
+ return Reflect.get(target, prop);
53
+ }
54
+ if (typeof prop === "string") {
55
+ return target.get(prop as PublicConfigKey);
56
+ }
57
+ return undefined;
58
+ },
59
+ set(target, prop, value, receiver) {
60
+ if (prop in target) {
61
+ return Reflect.set(target, prop, value, receiver);
62
+ }
63
+ return false;
64
+ },
65
+ }) as ConfigProxy;
66
+ return publicConfig;
67
+ }
68
+
69
+ export const publicConfigTransporter: Transporter = {
70
+ encode: (value) =>
71
+ value instanceof PublicConfigManager ? JSON.stringify(value.getPublicConfig()) : false,
72
+ decode: (value) => getConfigManager(JSON.parse(value)),
73
+ };
74
+
75
+ export const usePublicConfig = () => getContext<ConfigProxy>("publicConfig");
@@ -0,0 +1,17 @@
1
+ import { goto } from "$app/navigation";
2
+ import { base } from "$app/paths";
3
+ import { page } from "$app/state";
4
+
5
+ /**
6
+ * Redirects to the login page if the user is not authenticated
7
+ * and the login feature is enabled.
8
+ */
9
+ export function requireAuthUser(): boolean {
10
+ if (page.data.loginEnabled && !page.data.user) {
11
+ const next = page.url.pathname + page.url.search;
12
+ const url = `${base}/login?next=${encodeURIComponent(next)}`;
13
+ goto(url, { invalidateAll: true });
14
+ return true;
15
+ }
16
+ return false;
17
+ }
@@ -0,0 +1,33 @@
1
+ /**
2
+ * Chunk array into arrays of length at most `chunkSize`
3
+ *
4
+ * @param chunkSize must be greater than or equal to 1
5
+ */
6
+ export function chunk<T extends unknown[] | string>(arr: T, chunkSize: number): T[] {
7
+ if (isNaN(chunkSize) || chunkSize < 1) {
8
+ throw new RangeError("Invalid chunk size: " + chunkSize);
9
+ }
10
+
11
+ if (!arr.length) {
12
+ return [];
13
+ }
14
+
15
+ /// Small optimization to not chunk buffers unless needed
16
+ if (arr.length <= chunkSize) {
17
+ return [arr];
18
+ }
19
+
20
+ return range(Math.ceil(arr.length / chunkSize)).map((i) => {
21
+ return arr.slice(i * chunkSize, (i + 1) * chunkSize);
22
+ }) as T[];
23
+ }
24
+
25
+ function range(n: number, b?: number): number[] {
26
+ return b
27
+ ? Array(b - n)
28
+ .fill(0)
29
+ .map((_, i) => n + i)
30
+ : Array(n)
31
+ .fill(0)
32
+ .map((_, i) => i);
33
+ }
@@ -0,0 +1,13 @@
1
+ import { browser } from "$app/environment";
2
+
3
+ export function cookiesAreEnabled(): boolean {
4
+ if (!browser) return false;
5
+ if (navigator.cookieEnabled) return navigator.cookieEnabled;
6
+
7
+ // Create cookie
8
+ document.cookie = "cookietest=1";
9
+ const ret = document.cookie.indexOf("cookietest=") != -1;
10
+ // Delete cookie
11
+ document.cookie = "cookietest=1; expires=Thu, 01-Jan-1970 00:00:01 GMT";
12
+ return ret;
13
+ }
@@ -0,0 +1,17 @@
1
+ /**
2
+ * A debounce function that works in both browser and Nodejs.
3
+ * For pure Nodejs work, prefer the `Debouncer` class.
4
+ */
5
+ export function debounce<T extends unknown[]>(
6
+ callback: (...rest: T) => unknown,
7
+ limit: number
8
+ ): (...rest: T) => void {
9
+ let timer: ReturnType<typeof setTimeout>;
10
+
11
+ return function (...rest) {
12
+ clearTimeout(timer);
13
+ timer = setTimeout(() => {
14
+ callback(...rest);
15
+ }, limit);
16
+ };
17
+ }
@@ -0,0 +1,6 @@
1
+ export function deepestChild(el: HTMLElement): HTMLElement {
2
+ if (el.lastElementChild && el.lastElementChild.nodeType !== Node.TEXT_NODE) {
3
+ return deepestChild(el.lastElementChild as HTMLElement);
4
+ }
5
+ return el;
6
+ }
@@ -0,0 +1,21 @@
1
+ /**
2
+ * Generates a Google favicon URL for the given server URL
3
+ * @param serverUrl - The MCP server URL (e.g., "https://mcp.exa.ai/mcp")
4
+ * @param size - The size of the favicon in pixels (default: 64)
5
+ * @returns The Google favicon service URL
6
+ */
7
+ export function getMcpServerFaviconUrl(serverUrl: string, size: number = 64): string {
8
+ try {
9
+ const parsed = new URL(serverUrl);
10
+ // Extract root domain (e.g., "exa.ai" from "mcp.exa.ai")
11
+ // Google's favicon service needs the root domain, not subdomains
12
+ const hostnameParts = parsed.hostname.split(".");
13
+ const rootDomain =
14
+ hostnameParts.length >= 2 ? hostnameParts.slice(-2).join(".") : parsed.hostname;
15
+ const domain = `${parsed.protocol}//${rootDomain}`;
16
+ return `https://www.google.com/s2/favicons?sz=${size}&domain_url=${encodeURIComponent(domain)}`;
17
+ } catch {
18
+ // If URL parsing fails, just use the raw serverUrl - Google will handle it
19
+ return `https://www.google.com/s2/favicons?sz=${size}&domain_url=${encodeURIComponent(serverUrl)}`;
20
+ }
21
+ }
@@ -0,0 +1,23 @@
1
+ export async function fetchJSON<T>(
2
+ url: string,
3
+ options?: {
4
+ fetch?: typeof window.fetch;
5
+ allowNull?: boolean;
6
+ }
7
+ ): Promise<T> {
8
+ const response = await (options?.fetch ?? fetch)(url);
9
+ if (!response.ok) {
10
+ throw new Error(`Failed to fetch ${url}: ${response.status} ${response.statusText}`);
11
+ }
12
+
13
+ // Handle empty responses (which parse to null)
14
+ const text = await response.text();
15
+ if (!text || text.trim() === "") {
16
+ if (options?.allowNull) {
17
+ return null as T;
18
+ }
19
+ throw new Error(`Received empty response from ${url} but allowNull is not set to true`);
20
+ }
21
+
22
+ return JSON.parse(text);
23
+ }
@@ -0,0 +1,14 @@
1
+ const file2base64 = (file: File): Promise<string> => {
2
+ return new Promise<string>((resolve, reject) => {
3
+ const reader = new FileReader();
4
+ reader.readAsDataURL(file);
5
+ reader.onload = () => {
6
+ const dataUrl = reader.result as string;
7
+ const base64 = dataUrl.split(",")[1];
8
+ resolve(base64);
9
+ };
10
+ reader.onerror = (error) => reject(error);
11
+ });
12
+ };
13
+
14
+ export default file2base64;
@@ -0,0 +1,37 @@
1
+ export function formatUserCount(userCount: number): string {
2
+ const userCountRanges: { min: number; max: number; label: string }[] = [
3
+ { min: 0, max: 1, label: "1" },
4
+ { min: 2, max: 9, label: "1-10" },
5
+ { min: 10, max: 49, label: "10+" },
6
+ { min: 50, max: 99, label: "50+" },
7
+ { min: 100, max: 299, label: "100+" },
8
+ { min: 300, max: 499, label: "300+" },
9
+ { min: 500, max: 999, label: "500+" },
10
+ { min: 1_000, max: 2_999, label: "1k+" },
11
+ { min: 3_000, max: 4_999, label: "3k+" },
12
+ { min: 5_000, max: 9_999, label: "5k+" },
13
+ { min: 10_000, max: 19_999, label: "10k+" },
14
+ { min: 20_000, max: 29_999, label: "20k+" },
15
+ { min: 30_000, max: 39_999, label: "30k+" },
16
+ { min: 40_000, max: 49_999, label: "40k+" },
17
+ { min: 50_000, max: 59_999, label: "50k+" },
18
+ { min: 60_000, max: 69_999, label: "60k+" },
19
+ { min: 70_000, max: 79_999, label: "70k+" },
20
+ { min: 80_000, max: 89_999, label: "80k+" },
21
+ { min: 90_000, max: 99_999, label: "90k+" },
22
+ { min: 100_000, max: 109_999, label: "100k+" },
23
+ { min: 110_000, max: 119_999, label: "110k+" },
24
+ { min: 120_000, max: 129_999, label: "120k+" },
25
+ { min: 130_000, max: 139_999, label: "130k+" },
26
+ { min: 140_000, max: 149_999, label: "140k+" },
27
+ { min: 150_000, max: 199_999, label: "150k+" },
28
+ { min: 200_000, max: 299_999, label: "200k+" },
29
+ { min: 300_000, max: 499_999, label: "300k+" },
30
+ { min: 500_000, max: 749_999, label: "500k+" },
31
+ { min: 750_000, max: 999_999, label: "750k+" },
32
+ { min: 1_000_000, max: Infinity, label: "1M+" },
33
+ ];
34
+
35
+ const range = userCountRanges.find(({ min, max }) => userCount >= min && userCount <= max);
36
+ return range?.label ?? "";
37
+ }
@@ -0,0 +1,75 @@
1
+ import { describe, expect, test } from "vitest";
2
+
3
+ import type { Message } from "$lib/types/Message";
4
+ import { MessageUpdateStatus, MessageUpdateType } from "$lib/types/MessageUpdate";
5
+ import { isAssistantGenerationTerminal, isConversationGenerationActive } from "./generationState";
6
+
7
+ function assistantMessage(overrides: Partial<Message> = {}): Message {
8
+ return {
9
+ from: "assistant",
10
+ id: "assistant-1" as Message["id"],
11
+ content: "",
12
+ children: [],
13
+ ...overrides,
14
+ };
15
+ }
16
+
17
+ describe("generationState", () => {
18
+ test("returns active when assistant has no terminal update", () => {
19
+ const messages = [
20
+ assistantMessage({
21
+ updates: [{ type: MessageUpdateType.Stream, token: "Hello" }],
22
+ }),
23
+ ];
24
+
25
+ expect(isConversationGenerationActive(messages)).toBe(true);
26
+ });
27
+
28
+ test("treats final answer update as terminal", () => {
29
+ const message = assistantMessage({
30
+ updates: [{ type: MessageUpdateType.FinalAnswer, text: "Done", interrupted: false }],
31
+ });
32
+
33
+ expect(isAssistantGenerationTerminal(message)).toBe(true);
34
+ expect(isConversationGenerationActive([message])).toBe(false);
35
+ });
36
+
37
+ test("treats error status update as terminal", () => {
38
+ const message = assistantMessage({
39
+ updates: [
40
+ {
41
+ type: MessageUpdateType.Status,
42
+ status: MessageUpdateStatus.Error,
43
+ message: "Something went wrong",
44
+ },
45
+ ],
46
+ });
47
+
48
+ expect(isAssistantGenerationTerminal(message)).toBe(true);
49
+ expect(isConversationGenerationActive([message])).toBe(false);
50
+ });
51
+
52
+ test("treats finished status update as terminal", () => {
53
+ const message = assistantMessage({
54
+ updates: [
55
+ {
56
+ type: MessageUpdateType.Status,
57
+ status: MessageUpdateStatus.Finished,
58
+ },
59
+ ],
60
+ });
61
+
62
+ expect(isAssistantGenerationTerminal(message)).toBe(true);
63
+ expect(isConversationGenerationActive([message])).toBe(false);
64
+ });
65
+
66
+ test("treats interrupted assistant message as terminal", () => {
67
+ const message = assistantMessage({
68
+ interrupted: true,
69
+ updates: [{ type: MessageUpdateType.Stream, token: "partial" }],
70
+ });
71
+
72
+ expect(isAssistantGenerationTerminal(message)).toBe(true);
73
+ expect(isConversationGenerationActive([message])).toBe(false);
74
+ });
75
+ });
@@ -0,0 +1,26 @@
1
+ import type { Message } from "$lib/types/Message";
2
+ import { MessageUpdateStatus, MessageUpdateType } from "$lib/types/MessageUpdate";
3
+
4
+ export function isAssistantGenerationTerminal(message?: Message): boolean {
5
+ if (!message || message.from !== "assistant") return true;
6
+
7
+ if (message.interrupted === true) return true;
8
+
9
+ const updates = message.updates ?? [];
10
+ const hasFinalAnswer = updates.some((update) => update.type === MessageUpdateType.FinalAnswer);
11
+ if (hasFinalAnswer) return true;
12
+
13
+ return updates.some(
14
+ (update) =>
15
+ update.type === MessageUpdateType.Status &&
16
+ (update.status === MessageUpdateStatus.Error ||
17
+ update.status === MessageUpdateStatus.Finished)
18
+ );
19
+ }
20
+
21
+ export function isConversationGenerationActive(messages: Message[]): boolean {
22
+ const lastAssistant = [...messages].reverse().find((message) => message.from === "assistant");
23
+ if (!lastAssistant) return false;
24
+
25
+ return !isAssistantGenerationTerminal(lastAssistant);
26
+ }
@@ -0,0 +1,41 @@
1
+ export function getHref(
2
+ url: URL | string,
3
+ modifications: {
4
+ newKeys?: Record<string, string | undefined | null>;
5
+ existingKeys?: { behaviour: "delete_except" | "delete"; keys: string[] };
6
+ }
7
+ ) {
8
+ const newUrl = new URL(url);
9
+ const { newKeys, existingKeys } = modifications;
10
+
11
+ // exsiting keys logic
12
+ if (existingKeys) {
13
+ const { behaviour, keys } = existingKeys;
14
+ if (behaviour === "delete") {
15
+ for (const key of keys) {
16
+ newUrl.searchParams.delete(key);
17
+ }
18
+ } else {
19
+ // delete_except
20
+ const keysToPreserve = keys;
21
+ for (const key of [...newUrl.searchParams.keys()]) {
22
+ if (!keysToPreserve.includes(key)) {
23
+ newUrl.searchParams.delete(key);
24
+ }
25
+ }
26
+ }
27
+ }
28
+
29
+ // new keys logic
30
+ if (newKeys) {
31
+ for (const [key, val] of Object.entries(newKeys)) {
32
+ if (val) {
33
+ newUrl.searchParams.set(key, val);
34
+ } else {
35
+ newUrl.searchParams.delete(key);
36
+ }
37
+ }
38
+ }
39
+
40
+ return newUrl.toString();
41
+ }
@@ -0,0 +1,7 @@
1
+ export async function getReturnFromGenerator<T, R>(generator: AsyncGenerator<T, R>): Promise<R> {
2
+ let result: IteratorResult<T, R>;
3
+ do {
4
+ result = await generator.next();
5
+ } while (!result.done); // Keep calling `next()` until `done` is true
6
+ return result.value; // Return the final value
7
+ }