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,262 @@
1
+ import { describe, expect, it } from "vitest";
2
+ import {
3
+ MessageUpdateStatus,
4
+ MessageUpdateType,
5
+ type MessageUpdate,
6
+ } from "$lib/types/MessageUpdate";
7
+ import { applyStreamingMode, resolveStreamingMode, smoothStreamUpdates } from "./messageUpdates";
8
+
9
+ async function* fromArray<T>(values: T[]): AsyncGenerator<T> {
10
+ for (const value of values) {
11
+ yield value;
12
+ }
13
+ }
14
+
15
+ async function collect(iter: AsyncGenerator<MessageUpdate>) {
16
+ const updates: MessageUpdate[] = [];
17
+ for await (const update of iter) {
18
+ updates.push(update);
19
+ }
20
+ return updates;
21
+ }
22
+
23
+ const streamText = (updates: MessageUpdate[]) =>
24
+ updates
25
+ .filter((u) => u.type === MessageUpdateType.Stream)
26
+ .map((u) => u.token)
27
+ .join("");
28
+
29
+ describe("smoothStreamUpdates", () => {
30
+ it("merges partial words and preserves final text", async () => {
31
+ const source: MessageUpdate[] = [
32
+ { type: MessageUpdateType.Stream, token: "Hel" },
33
+ { type: MessageUpdateType.Stream, token: "lo " },
34
+ { type: MessageUpdateType.Stream, token: "wor" },
35
+ { type: MessageUpdateType.Stream, token: "ld!" },
36
+ { type: MessageUpdateType.Status, status: MessageUpdateStatus.Finished },
37
+ ];
38
+
39
+ const updates = await collect(
40
+ smoothStreamUpdates(fromArray(source), {
41
+ minDelayMs: 0,
42
+ maxDelayMs: 0,
43
+ _internal: { detectChunk: (buffer) => /\S+\s+/.exec(buffer)?.[0] ?? null },
44
+ })
45
+ );
46
+
47
+ const streamedChunks = updates.filter((u) => u.type === MessageUpdateType.Stream);
48
+ expect(streamedChunks.map((u) => u.token)).toEqual(["Hello ", "world!"]);
49
+ expect(streamText(updates)).toBe("Hello world!");
50
+ });
51
+
52
+ it("flushes buffered stream text before non-stream updates", async () => {
53
+ const source: MessageUpdate[] = [
54
+ { type: MessageUpdateType.Stream, token: "hello" },
55
+ { type: MessageUpdateType.Stream, token: " world" },
56
+ { type: MessageUpdateType.Title, title: "done" },
57
+ ];
58
+
59
+ const updates = await collect(
60
+ smoothStreamUpdates(fromArray(source), { minDelayMs: 0, maxDelayMs: 0 })
61
+ );
62
+ expect(updates[0]).toMatchObject({ type: MessageUpdateType.Stream });
63
+ expect(updates[1]).toMatchObject({ type: MessageUpdateType.Stream });
64
+ expect(updates[2]).toEqual({ type: MessageUpdateType.Title, title: "done" });
65
+ expect(streamText(updates)).toBe("hello world");
66
+ });
67
+
68
+ it("spreads burst tokens over time", async () => {
69
+ const bigToken = "word ".repeat(40); // 200 chars, 40 words
70
+ const source: MessageUpdate[] = [{ type: MessageUpdateType.Stream, token: bigToken }];
71
+ let nowMs = 0;
72
+ const emitTimes: number[] = [];
73
+
74
+ const iter = smoothStreamUpdates(fromArray(source), {
75
+ minDelayMs: 5,
76
+ maxDelayMs: 80,
77
+ minRateCharsPerMs: 0.3,
78
+ _internal: {
79
+ now: () => nowMs,
80
+ sleep: async (ms: number) => {
81
+ nowMs += ms;
82
+ },
83
+ detectChunk: (buffer) => /\S+\s+/.exec(buffer)?.[0] ?? null,
84
+ },
85
+ });
86
+
87
+ for await (const update of iter) {
88
+ if (update.type === MessageUpdateType.Stream) {
89
+ emitTimes.push(nowMs);
90
+ }
91
+ }
92
+
93
+ // Should have multiple emissions
94
+ expect(emitTimes.length).toBeGreaterThan(5);
95
+ // Gap between first and last emission should be significant (not instant dump)
96
+ const totalSpread = (emitTimes.at(-1) ?? 0) - (emitTimes[0] ?? 0);
97
+ expect(totalSpread).toBeGreaterThan(100);
98
+ });
99
+
100
+ it("keeps delays within configured bounds", async () => {
101
+ const source: MessageUpdate[] = [
102
+ {
103
+ type: MessageUpdateType.Stream,
104
+ token: "one two three four five six seven eight nine ten ",
105
+ },
106
+ ];
107
+ const delays: number[] = [];
108
+ let nowMs = 0;
109
+
110
+ await collect(
111
+ smoothStreamUpdates(fromArray(source), {
112
+ minDelayMs: 5,
113
+ maxDelayMs: 80,
114
+ minRateCharsPerMs: 0.3,
115
+ _internal: {
116
+ now: () => nowMs,
117
+ sleep: async (ms: number) => {
118
+ delays.push(ms);
119
+ nowMs += ms;
120
+ },
121
+ detectChunk: (buffer) => /\S+\s+/.exec(buffer)?.[0] ?? null,
122
+ },
123
+ })
124
+ );
125
+
126
+ expect(delays.length).toBeGreaterThan(2);
127
+ expect(delays.every((d) => d >= 5 && d <= 80)).toBe(true);
128
+ // First delay should be >= later delays (rate floor dominates initially)
129
+ expect(delays[0]).toBeGreaterThanOrEqual(delays.at(-1) ?? 0);
130
+ });
131
+
132
+ it("handles CJK text correctly", async () => {
133
+ const source: MessageUpdate[] = [{ type: MessageUpdateType.Stream, token: "你好,世界!" }];
134
+
135
+ const updates = await collect(
136
+ smoothStreamUpdates(fromArray(source), { minDelayMs: 0, maxDelayMs: 0 })
137
+ );
138
+
139
+ expect(streamText(updates)).toBe("你好,世界!");
140
+ });
141
+
142
+ it("propagates source errors to consumer", async () => {
143
+ async function* failingSource(): AsyncGenerator<MessageUpdate> {
144
+ yield { type: MessageUpdateType.Stream, token: "hello " };
145
+ throw new Error("source failed");
146
+ }
147
+
148
+ await expect(
149
+ collect(smoothStreamUpdates(failingSource(), { minDelayMs: 0, maxDelayMs: 0 }))
150
+ ).rejects.toThrow("source failed");
151
+ });
152
+
153
+ it("propagates source errors even when no full chunk was emitted yet", async () => {
154
+ async function* failingSource(): AsyncGenerator<MessageUpdate> {
155
+ yield { type: MessageUpdateType.Stream, token: "hel" };
156
+ throw new Error("source failed");
157
+ }
158
+
159
+ await expect(
160
+ collect(
161
+ smoothStreamUpdates(failingSource(), {
162
+ minDelayMs: 0,
163
+ maxDelayMs: 0,
164
+ _internal: { detectChunk: (buffer) => /\S+\s+/.exec(buffer)?.[0] ?? null },
165
+ })
166
+ )
167
+ ).rejects.toThrow("source failed");
168
+ });
169
+
170
+ it("drains queued stream chunks before throwing source errors", async () => {
171
+ async function* failingSource(): AsyncGenerator<MessageUpdate> {
172
+ yield { type: MessageUpdateType.Stream, token: "a " };
173
+ yield { type: MessageUpdateType.Stream, token: "b " };
174
+ yield { type: MessageUpdateType.Stream, token: "c " };
175
+ throw new Error("source failed");
176
+ }
177
+
178
+ const seen: MessageUpdate[] = [];
179
+ let seenError: Error | null = null;
180
+ try {
181
+ for await (const update of smoothStreamUpdates(failingSource(), {
182
+ minDelayMs: 0,
183
+ maxDelayMs: 0,
184
+ _internal: { detectChunk: (buffer) => /\S+\s+/.exec(buffer)?.[0] ?? null },
185
+ })) {
186
+ seen.push(update);
187
+ }
188
+ } catch (error) {
189
+ seenError = error as Error;
190
+ }
191
+
192
+ expect(streamText(seen)).toBe("a b c ");
193
+ expect(seenError?.message).toBe("source failed");
194
+ });
195
+
196
+ it("caps burst tail latency with backlog acceleration", async () => {
197
+ const source: MessageUpdate[] = [
198
+ { type: MessageUpdateType.Stream, token: "word ".repeat(500) },
199
+ ];
200
+ let nowMs = 0;
201
+ await collect(
202
+ smoothStreamUpdates(fromArray(source), {
203
+ minDelayMs: 5,
204
+ maxDelayMs: 80,
205
+ minRateCharsPerMs: 0.3,
206
+ maxBufferedMs: 400,
207
+ _internal: {
208
+ now: () => nowMs,
209
+ sleep: async (ms: number) => {
210
+ nowMs += ms;
211
+ },
212
+ detectChunk: (buffer) => /\S+\s+/.exec(buffer)?.[0] ?? null,
213
+ },
214
+ })
215
+ );
216
+
217
+ expect(nowMs).toBeLessThan(1500);
218
+ });
219
+
220
+ it("skips empty tokens gracefully", async () => {
221
+ const source: MessageUpdate[] = [
222
+ { type: MessageUpdateType.Stream, token: "" },
223
+ { type: MessageUpdateType.Stream, token: "hello " },
224
+ { type: MessageUpdateType.Stream, token: "" },
225
+ { type: MessageUpdateType.Stream, token: "world!" },
226
+ { type: MessageUpdateType.Status, status: MessageUpdateStatus.Finished },
227
+ ];
228
+
229
+ const updates = await collect(
230
+ smoothStreamUpdates(fromArray(source), { minDelayMs: 0, maxDelayMs: 0 })
231
+ );
232
+ expect(streamText(updates)).toBe("hello world!");
233
+ });
234
+ });
235
+
236
+ describe("applyStreamingMode", () => {
237
+ it("keeps stream unchanged for raw mode", async () => {
238
+ const source: MessageUpdate[] = [
239
+ { type: MessageUpdateType.Stream, token: "Hello" },
240
+ { type: MessageUpdateType.Status, status: MessageUpdateStatus.Finished },
241
+ ];
242
+
243
+ const raw = await collect(applyStreamingMode(fromArray(source), "raw"));
244
+
245
+ expect(raw).toEqual(source);
246
+ });
247
+ });
248
+
249
+ describe("resolveStreamingMode", () => {
250
+ it("returns explicit streamingMode when set", () => {
251
+ expect(resolveStreamingMode({ streamingMode: "raw" })).toBe("raw");
252
+ expect(resolveStreamingMode({ streamingMode: "smooth" })).toBe("smooth");
253
+ });
254
+
255
+ it("defaults to smooth when unset", () => {
256
+ expect(resolveStreamingMode({})).toBe("smooth");
257
+ });
258
+
259
+ it("maps unsupported legacy values to smooth", () => {
260
+ expect(resolveStreamingMode({ streamingMode: "final" })).toBe("smooth");
261
+ });
262
+ });
@@ -0,0 +1,324 @@
1
+ import type { MessageFile } from "$lib/types/Message";
2
+ import {
3
+ type MessageUpdate,
4
+ type MessageToolUpdate,
5
+ type MessageToolCallUpdate,
6
+ type MessageToolResultUpdate,
7
+ type MessageToolErrorUpdate,
8
+ type MessageToolProgressUpdate,
9
+ MessageUpdateType,
10
+ MessageToolUpdateType,
11
+ } from "$lib/types/MessageUpdate";
12
+ import type { StreamingMode } from "$lib/types/Settings";
13
+ import type { KeyValuePair } from "$lib/types/Tool";
14
+
15
+ type MessageUpdateRequestOptions = {
16
+ base: string;
17
+ inputs?: string;
18
+ messageId?: string;
19
+ isRetry: boolean;
20
+ isContinue?: boolean;
21
+ files?: MessageFile[];
22
+ // Optional: pass selected MCP server names (client-side selection)
23
+ selectedMcpServerNames?: string[];
24
+ // Optional: pass selected MCP server configs (for custom client-defined servers)
25
+ selectedMcpServers?: Array<{ name: string; url: string; headers?: KeyValuePair[] }>;
26
+ streamingMode?: StreamingMode;
27
+ autopilot?: boolean;
28
+ };
29
+
30
+ type ChunkDetector = (buffer: string) => string | null;
31
+
32
+ type SmoothStreamConfig = {
33
+ minDelayMs?: number;
34
+ maxDelayMs?: number;
35
+ minRateCharsPerMs?: number;
36
+ maxBufferedMs?: number;
37
+ _internal?: {
38
+ now?: () => number;
39
+ sleep?: (ms: number) => Promise<void>;
40
+ detectChunk?: ChunkDetector;
41
+ };
42
+ };
43
+
44
+ export async function fetchMessageUpdates(
45
+ conversationId: string,
46
+ opts: MessageUpdateRequestOptions,
47
+ abortSignal: AbortSignal
48
+ ): Promise<AsyncGenerator<MessageUpdate>> {
49
+ const abortController = new AbortController();
50
+ abortSignal.addEventListener("abort", () => abortController.abort());
51
+
52
+ const form = new FormData();
53
+
54
+ const optsJSON = JSON.stringify({
55
+ inputs: opts.inputs,
56
+ id: opts.messageId,
57
+ is_retry: opts.isRetry,
58
+ is_continue: Boolean(opts.isContinue),
59
+ // Will be ignored server-side if unsupported
60
+ selectedMcpServerNames: opts.selectedMcpServerNames,
61
+ selectedMcpServers: opts.selectedMcpServers,
62
+ autopilot: opts.autopilot,
63
+ });
64
+
65
+ opts.files?.forEach((file) => {
66
+ const name = file.type + ";" + file.name;
67
+
68
+ form.append("files", new File([file.value], name, { type: file.mime }));
69
+ });
70
+
71
+ form.append("data", optsJSON);
72
+
73
+ const response = await fetch(`${opts.base}/conversation/${conversationId}`, {
74
+ method: "POST",
75
+ body: form,
76
+ signal: abortController.signal,
77
+ });
78
+
79
+ if (!response.ok) {
80
+ const errorMessage = await response
81
+ .json()
82
+ .then((obj) => obj.message)
83
+ .catch(() => `Request failed with status code ${response.status}: ${response.statusText}`);
84
+ throw Error(errorMessage);
85
+ }
86
+ if (!response.body) {
87
+ throw Error("Body not defined");
88
+ }
89
+
90
+ return applyStreamingMode(
91
+ endpointStreamToIterator(response, abortController),
92
+ opts.streamingMode ?? "smooth"
93
+ );
94
+ }
95
+
96
+ export function applyStreamingMode(
97
+ iterator: AsyncGenerator<MessageUpdate>,
98
+ streamingMode: StreamingMode
99
+ ): AsyncGenerator<MessageUpdate> {
100
+ if (streamingMode === "smooth") {
101
+ return smoothStreamUpdates(iterator);
102
+ }
103
+
104
+ // "raw" keeps source stream intact.
105
+ return iterator;
106
+ }
107
+
108
+ export function resolveStreamingMode(s: { streamingMode?: unknown }): StreamingMode {
109
+ return s.streamingMode === "raw" || s.streamingMode === "smooth" ? s.streamingMode : "smooth";
110
+ }
111
+
112
+ async function* endpointStreamToIterator(
113
+ response: Response,
114
+ abortController: AbortController
115
+ ): AsyncGenerator<MessageUpdate> {
116
+ const reader = response.body?.pipeThrough(new TextDecoderStream()).getReader();
117
+ if (!reader) throw Error("Response for endpoint had no body");
118
+
119
+ // Handle any cases where we must abort
120
+ reader.closed.then(() => abortController.abort());
121
+
122
+ // Handle logic for aborting
123
+ abortController.signal.addEventListener("abort", () => reader.cancel());
124
+
125
+ // ex) If the last response is => {"type": "stream", "token":
126
+ // It should be => {"type": "stream", "token": "Hello"} = prev_input_chunk + "Hello"}
127
+ let prevChunk = "";
128
+ while (!abortController.signal.aborted) {
129
+ const { done, value } = await reader.read();
130
+ if (done) {
131
+ abortController.abort();
132
+ break;
133
+ }
134
+ if (!value) continue;
135
+
136
+ const { messageUpdates, remainingText } = parseMessageUpdates(prevChunk + value);
137
+ prevChunk = remainingText;
138
+ for (const messageUpdate of messageUpdates) yield messageUpdate;
139
+ }
140
+ }
141
+
142
+ function parseMessageUpdates(value: string): {
143
+ messageUpdates: MessageUpdate[];
144
+ remainingText: string;
145
+ } {
146
+ const inputs = value.split("\n");
147
+ const messageUpdates: MessageUpdate[] = [];
148
+ for (const input of inputs) {
149
+ try {
150
+ messageUpdates.push(JSON.parse(input) as MessageUpdate);
151
+ } catch (error) {
152
+ // in case of parsing error, we return what we were able to parse
153
+ if (error instanceof SyntaxError) {
154
+ return {
155
+ messageUpdates,
156
+ remainingText: inputs.at(-1) ?? "",
157
+ };
158
+ }
159
+ }
160
+ }
161
+ return { messageUpdates, remainingText: "" };
162
+ }
163
+
164
+ export async function* smoothStreamUpdates(
165
+ iterator: AsyncGenerator<MessageUpdate>,
166
+ {
167
+ minDelayMs = 5,
168
+ maxDelayMs = 80,
169
+ minRateCharsPerMs = 0.3,
170
+ maxBufferedMs = 400,
171
+ _internal: { now = () => performance.now(), sleep = defaultSleep, detectChunk } = {},
172
+ }: SmoothStreamConfig = {}
173
+ ): AsyncGenerator<MessageUpdate> {
174
+ const chunkDetector = detectChunk ?? createWordChunkDetector();
175
+ const eventTarget = new EventTarget();
176
+ const outputQueue: Array<{ update: MessageUpdate }> = [];
177
+ let producerDone = false;
178
+ let producerError: unknown = null;
179
+ let pendingBuffer = "";
180
+ let queuedStreamChars = 0;
181
+
182
+ const enqueue = (update: MessageUpdate) => {
183
+ if (update.type === MessageUpdateType.Stream) {
184
+ queuedStreamChars += update.token.length;
185
+ }
186
+ outputQueue.push({ update });
187
+ eventTarget.dispatchEvent(new Event("next"));
188
+ };
189
+
190
+ const flushPendingBuffer = () => {
191
+ if (pendingBuffer.length === 0) return;
192
+ enqueue({ type: MessageUpdateType.Stream, token: pendingBuffer });
193
+ pendingBuffer = "";
194
+ };
195
+
196
+ const producer = (async () => {
197
+ for await (const messageUpdate of iterator) {
198
+ if (messageUpdate.type !== MessageUpdateType.Stream) {
199
+ flushPendingBuffer();
200
+ enqueue(messageUpdate);
201
+ continue;
202
+ }
203
+
204
+ if (!messageUpdate.token) continue;
205
+
206
+ pendingBuffer += messageUpdate.token;
207
+ let chunk: string | null;
208
+ while ((chunk = chunkDetector(pendingBuffer)) !== null) {
209
+ if (chunk.length === 0) break;
210
+ enqueue({ type: MessageUpdateType.Stream, token: chunk });
211
+ pendingBuffer = pendingBuffer.slice(chunk.length);
212
+ }
213
+ }
214
+ flushPendingBuffer();
215
+ })()
216
+ .catch((error) => {
217
+ producerError = error;
218
+ })
219
+ .finally(() => {
220
+ producerDone = true;
221
+ eventTarget.dispatchEvent(new Event("next"));
222
+ });
223
+
224
+ // Character-rate targeting consumer
225
+ let totalCharsEmitted = 0;
226
+ let firstEmitAt: number | null = null;
227
+
228
+ while (!producerDone || outputQueue.length > 0) {
229
+ if (outputQueue.length === 0) {
230
+ await waitForEvent(eventTarget, "next");
231
+ continue;
232
+ }
233
+
234
+ const next = outputQueue.shift();
235
+ if (!next) continue;
236
+
237
+ if (next.update.type === MessageUpdateType.Stream) {
238
+ const tokenLen = next.update.token.length;
239
+ queuedStreamChars = Math.max(0, queuedStreamChars - tokenLen);
240
+ totalCharsEmitted += tokenLen;
241
+
242
+ if (firstEmitAt === null) firstEmitAt = now();
243
+
244
+ const elapsedMs = now() - firstEmitAt;
245
+ const currentRate = elapsedMs > 0 ? totalCharsEmitted / elapsedMs : 0;
246
+ const backlogChars = tokenLen + queuedStreamChars;
247
+ const backlogRate = maxBufferedMs > 0 ? backlogChars / maxBufferedMs : 0;
248
+ const targetRate = Math.max(currentRate, minRateCharsPerMs, backlogRate);
249
+ const rawDelay = tokenLen / targetRate;
250
+ const underBacklogPressure = backlogRate > minRateCharsPerMs;
251
+ const effectiveMinDelayMs = underBacklogPressure ? 0 : minDelayMs;
252
+ const delayMs = Math.round(Math.max(effectiveMinDelayMs, Math.min(maxDelayMs, rawDelay)));
253
+
254
+ if (delayMs > 0) {
255
+ await sleep(delayMs);
256
+ }
257
+ }
258
+
259
+ yield next.update;
260
+ }
261
+
262
+ await producer;
263
+ if (producerError) throw producerError;
264
+ }
265
+
266
+ function createWordChunkDetector(): ChunkDetector {
267
+ if (typeof Intl !== "undefined" && typeof Intl.Segmenter === "function") {
268
+ const segmenter = new Intl.Segmenter(undefined, { granularity: "word" });
269
+ return (buffer: string): string | null => {
270
+ if (buffer.length === 0) return null;
271
+ let cursor = 0;
272
+ let boundary = 0;
273
+ let sawWordLike = false;
274
+
275
+ for (const part of segmenter.segment(buffer)) {
276
+ cursor += part.segment.length;
277
+ if (part.isWordLike) {
278
+ sawWordLike = true;
279
+ continue;
280
+ }
281
+ if (sawWordLike) {
282
+ boundary = cursor;
283
+ break;
284
+ }
285
+ }
286
+
287
+ return boundary > 0 ? buffer.slice(0, boundary) : null;
288
+ };
289
+ }
290
+
291
+ const wordWithTrailingBoundary = /\S+\s+/m;
292
+ return (buffer: string): string | null => {
293
+ const match = wordWithTrailingBoundary.exec(buffer);
294
+ if (!match) return null;
295
+ return buffer.slice(0, match.index) + match[0];
296
+ };
297
+ }
298
+
299
+ // Tool update type guards for UI rendering
300
+ export const isMessageToolUpdate = (update: MessageUpdate): update is MessageToolUpdate =>
301
+ update.type === MessageUpdateType.Tool;
302
+
303
+ export const isMessageToolCallUpdate = (update: MessageUpdate): update is MessageToolCallUpdate =>
304
+ isMessageToolUpdate(update) && update.subtype === MessageToolUpdateType.Call;
305
+
306
+ export const isMessageToolResultUpdate = (
307
+ update: MessageUpdate
308
+ ): update is MessageToolResultUpdate =>
309
+ isMessageToolUpdate(update) && update.subtype === MessageToolUpdateType.Result;
310
+
311
+ export const isMessageToolErrorUpdate = (update: MessageUpdate): update is MessageToolErrorUpdate =>
312
+ isMessageToolUpdate(update) && update.subtype === MessageToolUpdateType.Error;
313
+
314
+ export const isMessageToolProgressUpdate = (
315
+ update: MessageUpdate
316
+ ): update is MessageToolProgressUpdate =>
317
+ isMessageToolUpdate(update) && update.subtype === MessageToolUpdateType.Progress;
318
+
319
+ const defaultSleep = (ms: number): Promise<void> =>
320
+ new Promise((resolve) => setTimeout(resolve, ms));
321
+ const waitForEvent = (eventTarget: EventTarget, eventName: string) =>
322
+ new Promise<boolean>((resolve) =>
323
+ eventTarget.addEventListener(eventName, () => resolve(true), { once: true })
324
+ );
@@ -0,0 +1,56 @@
1
+ // Lightweight MIME helpers to avoid new dependencies.
2
+
3
+ const EXTENSION_TO_MIME: Record<string, string> = {
4
+ png: "image/png",
5
+ jpg: "image/jpeg",
6
+ jpe: "image/jpeg",
7
+ jpeg: "image/jpeg",
8
+ gif: "image/gif",
9
+ webp: "image/webp",
10
+ svg: "image/svg+xml",
11
+ pdf: "application/pdf",
12
+ txt: "text/plain",
13
+ csv: "text/csv",
14
+ json: "application/json",
15
+ mp3: "audio/mpeg",
16
+ wav: "audio/wav",
17
+ ogg: "audio/ogg",
18
+ mp4: "video/mp4",
19
+ mov: "video/quicktime",
20
+ webm: "video/webm",
21
+ zip: "application/zip",
22
+ gz: "application/gzip",
23
+ tgz: "application/gzip",
24
+ tar: "application/x-tar",
25
+ html: "text/html",
26
+ htm: "text/html",
27
+ md: "text/markdown",
28
+ };
29
+
30
+ export function guessMimeFromUrl(url: string): string | undefined {
31
+ try {
32
+ const pathname = new URL(url).pathname;
33
+ const ext = pathname.split(".").pop()?.toLowerCase();
34
+ if (ext && EXTENSION_TO_MIME[ext]) return EXTENSION_TO_MIME[ext];
35
+ } catch {
36
+ /* ignore */
37
+ }
38
+ return undefined;
39
+ }
40
+
41
+ export function pickSafeMime(
42
+ forwardedType: string | null,
43
+ blobType: string | undefined,
44
+ url: string
45
+ ): string {
46
+ const inferred = guessMimeFromUrl(url);
47
+ if (forwardedType) return forwardedType;
48
+ if (
49
+ inferred &&
50
+ (!blobType || blobType === "application/octet-stream" || blobType.startsWith("text/plain"))
51
+ ) {
52
+ return inferred;
53
+ }
54
+ if (blobType) return blobType;
55
+ return inferred || "application/octet-stream";
56
+ }
@@ -0,0 +1,14 @@
1
+ import type { Model } from "$lib/types/Model";
2
+
3
+ export const findCurrentModel = (
4
+ models: Model[],
5
+ _oldModels: { id: string; transferTo?: string }[] = [],
6
+ id?: string
7
+ ): Model => {
8
+ if (id) {
9
+ const direct = models.find((m) => m.id === id);
10
+ if (direct) return direct;
11
+ }
12
+
13
+ return models[0];
14
+ };