@aigentic/ruflo 3.7.0-alpha.69

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 (524) hide show
  1. package/README.md +410 -0
  2. package/bin/ruflo.js +57 -0
  3. package/package.json +98 -0
  4. package/src/chat-ui/Dockerfile +25 -0
  5. package/src/chat-ui/patch-mcp-url-safety.sh +28 -0
  6. package/src/chat-ui/static/chatui/icon-144x144.png +0 -0
  7. package/src/chat-ui/static/chatui/omni-welcome.gif +0 -0
  8. package/src/config/config.example.json +76 -0
  9. package/src/mcp-bridge/Dockerfile +45 -0
  10. package/src/mcp-bridge/index.js +1668 -0
  11. package/src/mcp-bridge/mcp-stdio-kernel.js +159 -0
  12. package/src/mcp-bridge/package.json +17 -0
  13. package/src/mcp-bridge/test-harness.js +470 -0
  14. package/src/nginx/Dockerfile +10 -0
  15. package/src/nginx/nginx.conf +67 -0
  16. package/src/nginx/static/favicon-dark.svg +4 -0
  17. package/src/nginx/static/favicon.svg +4 -0
  18. package/src/nginx/static/icon.svg +5 -0
  19. package/src/nginx/static/logo.svg +9 -0
  20. package/src/nginx/static/manifest.json +22 -0
  21. package/src/nginx/static/welcome.js +184 -0
  22. package/src/ruvocal/.claude/skills/add-model-descriptions/SKILL.md +73 -0
  23. package/src/ruvocal/.devcontainer/Dockerfile +9 -0
  24. package/src/ruvocal/.devcontainer/devcontainer.json +36 -0
  25. package/src/ruvocal/.dockerignore +17 -0
  26. package/src/ruvocal/.eslintignore +13 -0
  27. package/src/ruvocal/.eslintrc.cjs +45 -0
  28. package/src/ruvocal/.gcloudignore +18 -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/CLAUDE.md +126 -0
  48. package/src/ruvocal/Dockerfile +96 -0
  49. package/src/ruvocal/LICENSE +203 -0
  50. package/src/ruvocal/PRIVACY.md +41 -0
  51. package/src/ruvocal/README.md +164 -0
  52. package/src/ruvocal/chart/Chart.yaml +5 -0
  53. package/src/ruvocal/chart/env/dev.yaml +260 -0
  54. package/src/ruvocal/chart/env/prod.yaml +273 -0
  55. package/src/ruvocal/chart/templates/_helpers.tpl +22 -0
  56. package/src/ruvocal/chart/templates/config.yaml +10 -0
  57. package/src/ruvocal/chart/templates/deployment.yaml +81 -0
  58. package/src/ruvocal/chart/templates/hpa.yaml +45 -0
  59. package/src/ruvocal/chart/templates/infisical.yaml +24 -0
  60. package/src/ruvocal/chart/templates/ingress-internal.yaml +32 -0
  61. package/src/ruvocal/chart/templates/ingress.yaml +32 -0
  62. package/src/ruvocal/chart/templates/network-policy.yaml +36 -0
  63. package/src/ruvocal/chart/templates/service-account.yaml +13 -0
  64. package/src/ruvocal/chart/templates/service-monitor.yaml +17 -0
  65. package/src/ruvocal/chart/templates/service.yaml +21 -0
  66. package/src/ruvocal/chart/values.yaml +73 -0
  67. package/src/ruvocal/cloudbuild.yaml +68 -0
  68. package/src/ruvocal/config/branding.env.example +19 -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/Dockerfile +45 -0
  91. package/src/ruvocal/mcp-bridge/cloudbuild.yaml +49 -0
  92. package/src/ruvocal/mcp-bridge/index.js +1878 -0
  93. package/src/ruvocal/mcp-bridge/mcp-stdio-kernel.js +159 -0
  94. package/src/ruvocal/mcp-bridge/package-lock.json +762 -0
  95. package/src/ruvocal/mcp-bridge/package.json +17 -0
  96. package/src/ruvocal/mcp-bridge/test-harness.js +470 -0
  97. package/src/ruvocal/models/add-your-models-here.txt +1 -0
  98. package/src/ruvocal/package-lock.json +11741 -0
  99. package/src/ruvocal/package.json +121 -0
  100. package/src/ruvocal/postcss.config.js +6 -0
  101. package/src/ruvocal/rvf.manifest.json +204 -0
  102. package/src/ruvocal/scripts/config.ts +64 -0
  103. package/src/ruvocal/scripts/generate-welcome.mjs +181 -0
  104. package/src/ruvocal/scripts/populate.ts +288 -0
  105. package/src/ruvocal/scripts/samples.txt +194 -0
  106. package/src/ruvocal/scripts/setups/vitest-setup-client.ts +0 -0
  107. package/src/ruvocal/scripts/setups/vitest-setup-server.ts +44 -0
  108. package/src/ruvocal/scripts/updateLocalEnv.ts +48 -0
  109. package/src/ruvocal/src/ambient.d.ts +7 -0
  110. package/src/ruvocal/src/app.d.ts +29 -0
  111. package/src/ruvocal/src/app.html +53 -0
  112. package/src/ruvocal/src/hooks.server.ts +32 -0
  113. package/src/ruvocal/src/hooks.ts +6 -0
  114. package/src/ruvocal/src/lib/APIClient.ts +148 -0
  115. package/src/ruvocal/src/lib/actions/clickOutside.ts +18 -0
  116. package/src/ruvocal/src/lib/actions/snapScrollToBottom.ts +346 -0
  117. package/src/ruvocal/src/lib/buildPrompt.ts +33 -0
  118. package/src/ruvocal/src/lib/components/AnnouncementBanner.svelte +20 -0
  119. package/src/ruvocal/src/lib/components/BackgroundGenerationPoller.svelte +168 -0
  120. package/src/ruvocal/src/lib/components/CodeBlock.svelte +73 -0
  121. package/src/ruvocal/src/lib/components/CopyToClipBoardBtn.svelte +92 -0
  122. package/src/ruvocal/src/lib/components/DeleteConversationModal.svelte +75 -0
  123. package/src/ruvocal/src/lib/components/EditConversationModal.svelte +100 -0
  124. package/src/ruvocal/src/lib/components/ExpandNavigation.svelte +22 -0
  125. package/src/ruvocal/src/lib/components/FoundationBackground.svelte +242 -0
  126. package/src/ruvocal/src/lib/components/HoverTooltip.svelte +44 -0
  127. package/src/ruvocal/src/lib/components/HtmlPreviewModal.svelte +143 -0
  128. package/src/ruvocal/src/lib/components/InfiniteScroll.svelte +50 -0
  129. package/src/ruvocal/src/lib/components/MobileNav.svelte +300 -0
  130. package/src/ruvocal/src/lib/components/Modal.svelte +115 -0
  131. package/src/ruvocal/src/lib/components/ModelCardMetadata.svelte +71 -0
  132. package/src/ruvocal/src/lib/components/NavConversationItem.svelte +151 -0
  133. package/src/ruvocal/src/lib/components/NavMenu.svelte +313 -0
  134. package/src/ruvocal/src/lib/components/Pagination.svelte +97 -0
  135. package/src/ruvocal/src/lib/components/PaginationArrow.svelte +27 -0
  136. package/src/ruvocal/src/lib/components/Portal.svelte +24 -0
  137. package/src/ruvocal/src/lib/components/RetryBtn.svelte +18 -0
  138. package/src/ruvocal/src/lib/components/RuFloUniverse.svelte +185 -0
  139. package/src/ruvocal/src/lib/components/RufloHelpModal.svelte +411 -0
  140. package/src/ruvocal/src/lib/components/ScrollToBottomBtn.svelte +47 -0
  141. package/src/ruvocal/src/lib/components/ScrollToPreviousBtn.svelte +77 -0
  142. package/src/ruvocal/src/lib/components/ShareConversationModal.svelte +182 -0
  143. package/src/ruvocal/src/lib/components/StopGeneratingBtn.svelte +69 -0
  144. package/src/ruvocal/src/lib/components/SubscribeModal.svelte +87 -0
  145. package/src/ruvocal/src/lib/components/Switch.svelte +36 -0
  146. package/src/ruvocal/src/lib/components/SystemPromptModal.svelte +44 -0
  147. package/src/ruvocal/src/lib/components/Toast.svelte +27 -0
  148. package/src/ruvocal/src/lib/components/Tooltip.svelte +30 -0
  149. package/src/ruvocal/src/lib/components/WelcomeModal.svelte +46 -0
  150. package/src/ruvocal/src/lib/components/chat/Alternatives.svelte +77 -0
  151. package/src/ruvocal/src/lib/components/chat/BlockWrapper.svelte +72 -0
  152. package/src/ruvocal/src/lib/components/chat/ChatInput.svelte +490 -0
  153. package/src/ruvocal/src/lib/components/chat/ChatIntroduction.svelte +123 -0
  154. package/src/ruvocal/src/lib/components/chat/ChatMessage.svelte +548 -0
  155. package/src/ruvocal/src/lib/components/chat/ChatWindow.svelte +1057 -0
  156. package/src/ruvocal/src/lib/components/chat/FileDropzone.svelte +92 -0
  157. package/src/ruvocal/src/lib/components/chat/ImageLightbox.svelte +66 -0
  158. package/src/ruvocal/src/lib/components/chat/MarkdownBlock.svelte +23 -0
  159. package/src/ruvocal/src/lib/components/chat/MarkdownRenderer.svelte +69 -0
  160. package/src/ruvocal/src/lib/components/chat/MarkdownRenderer.svelte.test.ts +58 -0
  161. package/src/ruvocal/src/lib/components/chat/MessageAvatar.svelte +103 -0
  162. package/src/ruvocal/src/lib/components/chat/ModelSwitch.svelte +64 -0
  163. package/src/ruvocal/src/lib/components/chat/OpenReasoningResults.svelte +81 -0
  164. package/src/ruvocal/src/lib/components/chat/TaskGroup.svelte +88 -0
  165. package/src/ruvocal/src/lib/components/chat/ToolUpdate.svelte +273 -0
  166. package/src/ruvocal/src/lib/components/chat/UploadedFile.svelte +253 -0
  167. package/src/ruvocal/src/lib/components/chat/UrlFetchModal.svelte +203 -0
  168. package/src/ruvocal/src/lib/components/chat/VoiceRecorder.svelte +214 -0
  169. package/src/ruvocal/src/lib/components/icons/IconBurger.svelte +20 -0
  170. package/src/ruvocal/src/lib/components/icons/IconCheap.svelte +20 -0
  171. package/src/ruvocal/src/lib/components/icons/IconChevron.svelte +24 -0
  172. package/src/ruvocal/src/lib/components/icons/IconDazzled.svelte +40 -0
  173. package/src/ruvocal/src/lib/components/icons/IconFast.svelte +20 -0
  174. package/src/ruvocal/src/lib/components/icons/IconLoading.svelte +22 -0
  175. package/src/ruvocal/src/lib/components/icons/IconMCP.svelte +28 -0
  176. package/src/ruvocal/src/lib/components/icons/IconMoon.svelte +21 -0
  177. package/src/ruvocal/src/lib/components/icons/IconNew.svelte +20 -0
  178. package/src/ruvocal/src/lib/components/icons/IconOmni.svelte +90 -0
  179. package/src/ruvocal/src/lib/components/icons/IconPaperclip.svelte +24 -0
  180. package/src/ruvocal/src/lib/components/icons/IconPro.svelte +37 -0
  181. package/src/ruvocal/src/lib/components/icons/IconShare.svelte +21 -0
  182. package/src/ruvocal/src/lib/components/icons/IconSun.svelte +93 -0
  183. package/src/ruvocal/src/lib/components/icons/Logo.svelte +68 -0
  184. package/src/ruvocal/src/lib/components/icons/LogoHuggingFaceBorderless.svelte +54 -0
  185. package/src/ruvocal/src/lib/components/mcp/AddServerForm.svelte +250 -0
  186. package/src/ruvocal/src/lib/components/mcp/MCPServerManager.svelte +185 -0
  187. package/src/ruvocal/src/lib/components/mcp/ServerCard.svelte +203 -0
  188. package/src/ruvocal/src/lib/components/players/AudioPlayer.svelte +82 -0
  189. package/src/ruvocal/src/lib/components/voice/AudioWaveform.svelte +96 -0
  190. package/src/ruvocal/src/lib/components/wasm/GalleryPanel.svelte +357 -0
  191. package/src/ruvocal/src/lib/constants/mcpExamples.ts +114 -0
  192. package/src/ruvocal/src/lib/constants/mime.ts +11 -0
  193. package/src/ruvocal/src/lib/constants/pagination.ts +1 -0
  194. package/src/ruvocal/src/lib/constants/publicSepToken.ts +1 -0
  195. package/src/ruvocal/src/lib/constants/routerExamples.ts +133 -0
  196. package/src/ruvocal/src/lib/constants/rvagentPresets.ts +206 -0
  197. package/src/ruvocal/src/lib/createShareLink.ts +27 -0
  198. package/src/ruvocal/src/lib/jobs/refresh-conversation-stats.ts +297 -0
  199. package/src/ruvocal/src/lib/migrations/lock.ts +56 -0
  200. package/src/ruvocal/src/lib/migrations/migrations.spec.ts +74 -0
  201. package/src/ruvocal/src/lib/migrations/migrations.ts +109 -0
  202. package/src/ruvocal/src/lib/migrations/routines/01-update-search-assistants.ts +50 -0
  203. package/src/ruvocal/src/lib/migrations/routines/02-update-assistants-models.ts +48 -0
  204. package/src/ruvocal/src/lib/migrations/routines/04-update-message-updates.ts +151 -0
  205. package/src/ruvocal/src/lib/migrations/routines/05-update-message-files.ts +56 -0
  206. package/src/ruvocal/src/lib/migrations/routines/06-trim-message-updates.ts +56 -0
  207. package/src/ruvocal/src/lib/migrations/routines/08-update-featured-to-review.ts +32 -0
  208. package/src/ruvocal/src/lib/migrations/routines/09-delete-empty-conversations.spec.ts +214 -0
  209. package/src/ruvocal/src/lib/migrations/routines/09-delete-empty-conversations.ts +88 -0
  210. package/src/ruvocal/src/lib/migrations/routines/10-update-reports-assistantid.ts +29 -0
  211. package/src/ruvocal/src/lib/migrations/routines/index.ts +15 -0
  212. package/src/ruvocal/src/lib/server/__tests__/conversation-stop-generating.spec.ts +103 -0
  213. package/src/ruvocal/src/lib/server/abortRegistry.ts +57 -0
  214. package/src/ruvocal/src/lib/server/abortedGenerations.ts +43 -0
  215. package/src/ruvocal/src/lib/server/adminToken.ts +62 -0
  216. package/src/ruvocal/src/lib/server/api/__tests__/conversations-id.spec.ts +296 -0
  217. package/src/ruvocal/src/lib/server/api/__tests__/conversations-message.spec.ts +216 -0
  218. package/src/ruvocal/src/lib/server/api/__tests__/conversations.spec.ts +235 -0
  219. package/src/ruvocal/src/lib/server/api/__tests__/misc.spec.ts +72 -0
  220. package/src/ruvocal/src/lib/server/api/__tests__/testHelpers.ts +86 -0
  221. package/src/ruvocal/src/lib/server/api/__tests__/user-reports.spec.ts +78 -0
  222. package/src/ruvocal/src/lib/server/api/__tests__/user.spec.ts +239 -0
  223. package/src/ruvocal/src/lib/server/api/types.ts +37 -0
  224. package/src/ruvocal/src/lib/server/api/utils/requireAuth.ts +22 -0
  225. package/src/ruvocal/src/lib/server/api/utils/resolveConversation.ts +69 -0
  226. package/src/ruvocal/src/lib/server/api/utils/resolveModel.ts +27 -0
  227. package/src/ruvocal/src/lib/server/api/utils/superjsonResponse.ts +15 -0
  228. package/src/ruvocal/src/lib/server/apiToken.ts +11 -0
  229. package/src/ruvocal/src/lib/server/auth.ts +554 -0
  230. package/src/ruvocal/src/lib/server/config.ts +187 -0
  231. package/src/ruvocal/src/lib/server/conversation.ts +83 -0
  232. package/src/ruvocal/src/lib/server/database/__tests__/rvf.spec.ts +709 -0
  233. package/src/ruvocal/src/lib/server/database/postgres.ts +700 -0
  234. package/src/ruvocal/src/lib/server/database/rvf.ts +1078 -0
  235. package/src/ruvocal/src/lib/server/database.ts +145 -0
  236. package/src/ruvocal/src/lib/server/endpoints/document.ts +68 -0
  237. package/src/ruvocal/src/lib/server/endpoints/endpoints.ts +43 -0
  238. package/src/ruvocal/src/lib/server/endpoints/images.ts +211 -0
  239. package/src/ruvocal/src/lib/server/endpoints/openai/endpointOai.ts +266 -0
  240. package/src/ruvocal/src/lib/server/endpoints/openai/openAIChatToTextGenerationStream.ts +212 -0
  241. package/src/ruvocal/src/lib/server/endpoints/openai/openAICompletionToTextGenerationStream.ts +32 -0
  242. package/src/ruvocal/src/lib/server/endpoints/preprocessMessages.ts +61 -0
  243. package/src/ruvocal/src/lib/server/exitHandler.ts +59 -0
  244. package/src/ruvocal/src/lib/server/files/downloadFile.ts +34 -0
  245. package/src/ruvocal/src/lib/server/files/uploadFile.ts +29 -0
  246. package/src/ruvocal/src/lib/server/findRepoRoot.ts +13 -0
  247. package/src/ruvocal/src/lib/server/fonts/Inter-Black.ttf +0 -0
  248. package/src/ruvocal/src/lib/server/fonts/Inter-Bold.ttf +0 -0
  249. package/src/ruvocal/src/lib/server/fonts/Inter-ExtraBold.ttf +0 -0
  250. package/src/ruvocal/src/lib/server/fonts/Inter-ExtraLight.ttf +0 -0
  251. package/src/ruvocal/src/lib/server/fonts/Inter-Light.ttf +0 -0
  252. package/src/ruvocal/src/lib/server/fonts/Inter-Medium.ttf +0 -0
  253. package/src/ruvocal/src/lib/server/fonts/Inter-Regular.ttf +0 -0
  254. package/src/ruvocal/src/lib/server/fonts/Inter-SemiBold.ttf +0 -0
  255. package/src/ruvocal/src/lib/server/fonts/Inter-Thin.ttf +0 -0
  256. package/src/ruvocal/src/lib/server/generateFromDefaultEndpoint.ts +46 -0
  257. package/src/ruvocal/src/lib/server/hooks/error.ts +37 -0
  258. package/src/ruvocal/src/lib/server/hooks/fetch.ts +22 -0
  259. package/src/ruvocal/src/lib/server/hooks/handle.ts +250 -0
  260. package/src/ruvocal/src/lib/server/hooks/init.ts +51 -0
  261. package/src/ruvocal/src/lib/server/isURLLocal.spec.ts +31 -0
  262. package/src/ruvocal/src/lib/server/isURLLocal.ts +74 -0
  263. package/src/ruvocal/src/lib/server/logger.ts +42 -0
  264. package/src/ruvocal/src/lib/server/mcp/clientPool.spec.ts +175 -0
  265. package/src/ruvocal/src/lib/server/mcp/clientPool.ts +0 -0
  266. package/src/ruvocal/src/lib/server/mcp/hf.ts +32 -0
  267. package/src/ruvocal/src/lib/server/mcp/httpClient.ts +122 -0
  268. package/src/ruvocal/src/lib/server/mcp/registry.ts +76 -0
  269. package/src/ruvocal/src/lib/server/mcp/tools.ts +196 -0
  270. package/src/ruvocal/src/lib/server/metrics.ts +255 -0
  271. package/src/ruvocal/src/lib/server/models.ts +518 -0
  272. package/src/ruvocal/src/lib/server/requestContext.ts +55 -0
  273. package/src/ruvocal/src/lib/server/router/arch.ts +230 -0
  274. package/src/ruvocal/src/lib/server/router/endpoint.ts +316 -0
  275. package/src/ruvocal/src/lib/server/router/multimodal.ts +28 -0
  276. package/src/ruvocal/src/lib/server/router/policy.ts +49 -0
  277. package/src/ruvocal/src/lib/server/router/toolsRoute.ts +51 -0
  278. package/src/ruvocal/src/lib/server/router/types.ts +21 -0
  279. package/src/ruvocal/src/lib/server/sendSlack.ts +23 -0
  280. package/src/ruvocal/src/lib/server/textGeneration/generate.ts +258 -0
  281. package/src/ruvocal/src/lib/server/textGeneration/index.ts +96 -0
  282. package/src/ruvocal/src/lib/server/textGeneration/mcp/fileRefs.ts +155 -0
  283. package/src/ruvocal/src/lib/server/textGeneration/mcp/routerResolution.ts +108 -0
  284. package/src/ruvocal/src/lib/server/textGeneration/mcp/runMcpFlow.ts +831 -0
  285. package/src/ruvocal/src/lib/server/textGeneration/mcp/toolInvocation.ts +349 -0
  286. package/src/ruvocal/src/lib/server/textGeneration/mcp/wasmTools.test.ts +633 -0
  287. package/src/ruvocal/src/lib/server/textGeneration/reasoning.ts +23 -0
  288. package/src/ruvocal/src/lib/server/textGeneration/title.ts +83 -0
  289. package/src/ruvocal/src/lib/server/textGeneration/types.ts +28 -0
  290. package/src/ruvocal/src/lib/server/textGeneration/utils/prepareFiles.ts +88 -0
  291. package/src/ruvocal/src/lib/server/textGeneration/utils/routing.ts +21 -0
  292. package/src/ruvocal/src/lib/server/textGeneration/utils/toolPrompt.ts +49 -0
  293. package/src/ruvocal/src/lib/server/urlSafety.ts +77 -0
  294. package/src/ruvocal/src/lib/server/usageLimits.ts +30 -0
  295. package/src/ruvocal/src/lib/stores/autopilotStore.svelte.ts +175 -0
  296. package/src/ruvocal/src/lib/stores/backgroundGenerations.svelte.ts +32 -0
  297. package/src/ruvocal/src/lib/stores/backgroundGenerations.ts +1 -0
  298. package/src/ruvocal/src/lib/stores/errors.ts +9 -0
  299. package/src/ruvocal/src/lib/stores/isAborted.ts +3 -0
  300. package/src/ruvocal/src/lib/stores/isPro.ts +4 -0
  301. package/src/ruvocal/src/lib/stores/loading.ts +3 -0
  302. package/src/ruvocal/src/lib/stores/mcpServers.ts +534 -0
  303. package/src/ruvocal/src/lib/stores/pendingChatInput.ts +3 -0
  304. package/src/ruvocal/src/lib/stores/pendingMessage.ts +9 -0
  305. package/src/ruvocal/src/lib/stores/settings.ts +182 -0
  306. package/src/ruvocal/src/lib/stores/shareModal.ts +13 -0
  307. package/src/ruvocal/src/lib/stores/titleUpdate.ts +8 -0
  308. package/src/ruvocal/src/lib/stores/wasmMcp.ts +472 -0
  309. package/src/ruvocal/src/lib/switchTheme.ts +124 -0
  310. package/src/ruvocal/src/lib/types/AbortedGeneration.ts +8 -0
  311. package/src/ruvocal/src/lib/types/Assistant.ts +31 -0
  312. package/src/ruvocal/src/lib/types/AssistantStats.ts +11 -0
  313. package/src/ruvocal/src/lib/types/ConfigKey.ts +4 -0
  314. package/src/ruvocal/src/lib/types/ConvSidebar.ts +9 -0
  315. package/src/ruvocal/src/lib/types/Conversation.ts +27 -0
  316. package/src/ruvocal/src/lib/types/ConversationStats.ts +13 -0
  317. package/src/ruvocal/src/lib/types/Message.ts +41 -0
  318. package/src/ruvocal/src/lib/types/MessageEvent.ts +10 -0
  319. package/src/ruvocal/src/lib/types/MessageUpdate.ts +139 -0
  320. package/src/ruvocal/src/lib/types/MigrationResult.ts +7 -0
  321. package/src/ruvocal/src/lib/types/Model.ts +23 -0
  322. package/src/ruvocal/src/lib/types/Report.ts +12 -0
  323. package/src/ruvocal/src/lib/types/Review.ts +6 -0
  324. package/src/ruvocal/src/lib/types/Semaphore.ts +19 -0
  325. package/src/ruvocal/src/lib/types/Session.ts +22 -0
  326. package/src/ruvocal/src/lib/types/Settings.ts +93 -0
  327. package/src/ruvocal/src/lib/types/SharedConversation.ts +9 -0
  328. package/src/ruvocal/src/lib/types/Template.ts +6 -0
  329. package/src/ruvocal/src/lib/types/Timestamps.ts +4 -0
  330. package/src/ruvocal/src/lib/types/TokenCache.ts +6 -0
  331. package/src/ruvocal/src/lib/types/Tool.ts +77 -0
  332. package/src/ruvocal/src/lib/types/UrlDependency.ts +5 -0
  333. package/src/ruvocal/src/lib/types/User.ts +14 -0
  334. package/src/ruvocal/src/lib/utils/PublicConfig.svelte.ts +75 -0
  335. package/src/ruvocal/src/lib/utils/auth.ts +17 -0
  336. package/src/ruvocal/src/lib/utils/chunk.ts +33 -0
  337. package/src/ruvocal/src/lib/utils/cookiesAreEnabled.ts +13 -0
  338. package/src/ruvocal/src/lib/utils/debounce.ts +17 -0
  339. package/src/ruvocal/src/lib/utils/deepestChild.ts +6 -0
  340. package/src/ruvocal/src/lib/utils/favicon.ts +21 -0
  341. package/src/ruvocal/src/lib/utils/fetchJSON.ts +23 -0
  342. package/src/ruvocal/src/lib/utils/file2base64.ts +14 -0
  343. package/src/ruvocal/src/lib/utils/formatUserCount.ts +37 -0
  344. package/src/ruvocal/src/lib/utils/generationState.spec.ts +75 -0
  345. package/src/ruvocal/src/lib/utils/generationState.ts +26 -0
  346. package/src/ruvocal/src/lib/utils/getHref.ts +41 -0
  347. package/src/ruvocal/src/lib/utils/getReturnFromGenerator.ts +7 -0
  348. package/src/ruvocal/src/lib/utils/haptics.ts +64 -0
  349. package/src/ruvocal/src/lib/utils/hashConv.ts +12 -0
  350. package/src/ruvocal/src/lib/utils/hf.ts +17 -0
  351. package/src/ruvocal/src/lib/utils/isDesktop.ts +7 -0
  352. package/src/ruvocal/src/lib/utils/isUrl.ts +8 -0
  353. package/src/ruvocal/src/lib/utils/isVirtualKeyboard.ts +16 -0
  354. package/src/ruvocal/src/lib/utils/loadAttachmentsFromUrls.ts +115 -0
  355. package/src/ruvocal/src/lib/utils/marked.spec.ts +96 -0
  356. package/src/ruvocal/src/lib/utils/marked.ts +531 -0
  357. package/src/ruvocal/src/lib/utils/mcpValidation.ts +147 -0
  358. package/src/ruvocal/src/lib/utils/mergeAsyncGenerators.ts +38 -0
  359. package/src/ruvocal/src/lib/utils/messageUpdates.spec.ts +262 -0
  360. package/src/ruvocal/src/lib/utils/messageUpdates.ts +324 -0
  361. package/src/ruvocal/src/lib/utils/mime.ts +56 -0
  362. package/src/ruvocal/src/lib/utils/models.ts +14 -0
  363. package/src/ruvocal/src/lib/utils/parseBlocks.ts +120 -0
  364. package/src/ruvocal/src/lib/utils/parseIncompleteMarkdown.ts +644 -0
  365. package/src/ruvocal/src/lib/utils/parseStringToList.ts +10 -0
  366. package/src/ruvocal/src/lib/utils/randomUuid.ts +14 -0
  367. package/src/ruvocal/src/lib/utils/searchTokens.ts +33 -0
  368. package/src/ruvocal/src/lib/utils/sha256.ts +7 -0
  369. package/src/ruvocal/src/lib/utils/stringifyError.ts +12 -0
  370. package/src/ruvocal/src/lib/utils/sum.ts +3 -0
  371. package/src/ruvocal/src/lib/utils/template.spec.ts +59 -0
  372. package/src/ruvocal/src/lib/utils/template.ts +53 -0
  373. package/src/ruvocal/src/lib/utils/timeout.ts +9 -0
  374. package/src/ruvocal/src/lib/utils/toolProgress.spec.ts +46 -0
  375. package/src/ruvocal/src/lib/utils/toolProgress.ts +11 -0
  376. package/src/ruvocal/src/lib/utils/tree/addChildren.spec.ts +102 -0
  377. package/src/ruvocal/src/lib/utils/tree/addChildren.ts +48 -0
  378. package/src/ruvocal/src/lib/utils/tree/addSibling.spec.ts +81 -0
  379. package/src/ruvocal/src/lib/utils/tree/addSibling.ts +41 -0
  380. package/src/ruvocal/src/lib/utils/tree/buildSubtree.spec.ts +110 -0
  381. package/src/ruvocal/src/lib/utils/tree/buildSubtree.ts +24 -0
  382. package/src/ruvocal/src/lib/utils/tree/convertLegacyConversation.spec.ts +31 -0
  383. package/src/ruvocal/src/lib/utils/tree/convertLegacyConversation.ts +36 -0
  384. package/src/ruvocal/src/lib/utils/tree/isMessageId.spec.ts +15 -0
  385. package/src/ruvocal/src/lib/utils/tree/isMessageId.ts +5 -0
  386. package/src/ruvocal/src/lib/utils/tree/tree.d.ts +14 -0
  387. package/src/ruvocal/src/lib/utils/tree/treeHelpers.spec.ts +167 -0
  388. package/src/ruvocal/src/lib/utils/updates.ts +39 -0
  389. package/src/ruvocal/src/lib/utils/urlParams.ts +13 -0
  390. package/src/ruvocal/src/lib/wasm/idb.ts +438 -0
  391. package/src/ruvocal/src/lib/wasm/index.ts +1213 -0
  392. package/src/ruvocal/src/lib/wasm/tests/wasm-capabilities.test.ts +565 -0
  393. package/src/ruvocal/src/lib/wasm/wasm.worker.ts +332 -0
  394. package/src/ruvocal/src/lib/wasm/workerClient.ts +166 -0
  395. package/src/ruvocal/src/lib/workers/autopilotWorker.ts +221 -0
  396. package/src/ruvocal/src/lib/workers/detailFetchWorker.ts +100 -0
  397. package/src/ruvocal/src/lib/workers/markdownWorker.ts +61 -0
  398. package/src/ruvocal/src/routes/+error.svelte +20 -0
  399. package/src/ruvocal/src/routes/+layout.svelte +324 -0
  400. package/src/ruvocal/src/routes/+layout.ts +91 -0
  401. package/src/ruvocal/src/routes/+page.svelte +168 -0
  402. package/src/ruvocal/src/routes/.well-known/oauth-cimd/+server.ts +37 -0
  403. package/src/ruvocal/src/routes/__debug/openai/+server.ts +21 -0
  404. package/src/ruvocal/src/routes/admin/export/+server.ts +159 -0
  405. package/src/ruvocal/src/routes/admin/stats/compute/+server.ts +16 -0
  406. package/src/ruvocal/src/routes/api/conversation/[id]/+server.ts +40 -0
  407. package/src/ruvocal/src/routes/api/conversation/[id]/message/[messageId]/+server.ts +42 -0
  408. package/src/ruvocal/src/routes/api/conversations/+server.ts +48 -0
  409. package/src/ruvocal/src/routes/api/fetch-url/+server.ts +147 -0
  410. package/src/ruvocal/src/routes/api/mcp/health/+server.ts +292 -0
  411. package/src/ruvocal/src/routes/api/mcp/servers/+server.ts +32 -0
  412. package/src/ruvocal/src/routes/api/models/+server.ts +25 -0
  413. package/src/ruvocal/src/routes/api/transcribe/+server.ts +104 -0
  414. package/src/ruvocal/src/routes/api/user/+server.ts +15 -0
  415. package/src/ruvocal/src/routes/api/user/validate-token/+server.ts +20 -0
  416. package/src/ruvocal/src/routes/api/v2/conversations/+server.ts +48 -0
  417. package/src/ruvocal/src/routes/api/v2/conversations/[id]/+server.ts +94 -0
  418. package/src/ruvocal/src/routes/api/v2/conversations/[id]/message/[messageId]/+server.ts +43 -0
  419. package/src/ruvocal/src/routes/api/v2/conversations/import-share/+server.ts +23 -0
  420. package/src/ruvocal/src/routes/api/v2/debug/config/+server.ts +16 -0
  421. package/src/ruvocal/src/routes/api/v2/debug/refresh/+server.ts +30 -0
  422. package/src/ruvocal/src/routes/api/v2/export/+server.ts +196 -0
  423. package/src/ruvocal/src/routes/api/v2/feature-flags/+server.ts +14 -0
  424. package/src/ruvocal/src/routes/api/v2/models/+server.ts +38 -0
  425. package/src/ruvocal/src/routes/api/v2/models/[namespace]/+server.ts +8 -0
  426. package/src/ruvocal/src/routes/api/v2/models/[namespace]/[model]/+server.ts +8 -0
  427. package/src/ruvocal/src/routes/api/v2/models/[namespace]/[model]/subscribe/+server.ts +28 -0
  428. package/src/ruvocal/src/routes/api/v2/models/[namespace]/subscribe/+server.ts +28 -0
  429. package/src/ruvocal/src/routes/api/v2/models/old/+server.ts +7 -0
  430. package/src/ruvocal/src/routes/api/v2/models/refresh/+server.ts +33 -0
  431. package/src/ruvocal/src/routes/api/v2/public-config/+server.ts +7 -0
  432. package/src/ruvocal/src/routes/api/v2/user/+server.ts +17 -0
  433. package/src/ruvocal/src/routes/api/v2/user/billing-orgs/+server.ts +73 -0
  434. package/src/ruvocal/src/routes/api/v2/user/reports/+server.ts +17 -0
  435. package/src/ruvocal/src/routes/api/v2/user/settings/+server.ts +110 -0
  436. package/src/ruvocal/src/routes/conversation/+server.ts +115 -0
  437. package/src/ruvocal/src/routes/conversation/[id]/+page.svelte +586 -0
  438. package/src/ruvocal/src/routes/conversation/[id]/+page.ts +60 -0
  439. package/src/ruvocal/src/routes/conversation/[id]/+server.ts +740 -0
  440. package/src/ruvocal/src/routes/conversation/[id]/message/[messageId]/prompt/+server.ts +66 -0
  441. package/src/ruvocal/src/routes/conversation/[id]/share/+server.ts +69 -0
  442. package/src/ruvocal/src/routes/conversation/[id]/stop-generating/+server.ts +35 -0
  443. package/src/ruvocal/src/routes/healthcheck/+server.ts +3 -0
  444. package/src/ruvocal/src/routes/login/+server.ts +5 -0
  445. package/src/ruvocal/src/routes/login/callback/+server.ts +103 -0
  446. package/src/ruvocal/src/routes/login/callback/updateUser.spec.ts +157 -0
  447. package/src/ruvocal/src/routes/login/callback/updateUser.ts +215 -0
  448. package/src/ruvocal/src/routes/logout/+server.ts +18 -0
  449. package/src/ruvocal/src/routes/metrics/+server.ts +18 -0
  450. package/src/ruvocal/src/routes/models/+page.svelte +233 -0
  451. package/src/ruvocal/src/routes/models/[...model]/+page.svelte +161 -0
  452. package/src/ruvocal/src/routes/models/[...model]/+page.ts +14 -0
  453. package/src/ruvocal/src/routes/models/[...model]/thumbnail.png/+server.ts +64 -0
  454. package/src/ruvocal/src/routes/models/[...model]/thumbnail.png/ModelThumbnail.svelte +28 -0
  455. package/src/ruvocal/src/routes/privacy/+page.svelte +11 -0
  456. package/src/ruvocal/src/routes/r/[id]/+page.ts +34 -0
  457. package/src/ruvocal/src/routes/settings/(nav)/+layout.svelte +282 -0
  458. package/src/ruvocal/src/routes/settings/(nav)/+layout.ts +1 -0
  459. package/src/ruvocal/src/routes/settings/(nav)/+page.svelte +0 -0
  460. package/src/ruvocal/src/routes/settings/(nav)/+server.ts +59 -0
  461. package/src/ruvocal/src/routes/settings/(nav)/[...model]/+page.svelte +464 -0
  462. package/src/ruvocal/src/routes/settings/(nav)/[...model]/+page.ts +14 -0
  463. package/src/ruvocal/src/routes/settings/(nav)/application/+page.svelte +362 -0
  464. package/src/ruvocal/src/routes/settings/+layout.svelte +40 -0
  465. package/src/ruvocal/src/styles/highlight-js.css +195 -0
  466. package/src/ruvocal/src/styles/main.css +144 -0
  467. package/src/ruvocal/static/chatui/apple-touch-icon.png +0 -0
  468. package/src/ruvocal/static/chatui/favicon-dark.svg +3 -0
  469. package/src/ruvocal/static/chatui/favicon-dev.svg +3 -0
  470. package/src/ruvocal/static/chatui/favicon.ico +0 -0
  471. package/src/ruvocal/static/chatui/favicon.svg +3 -0
  472. package/src/ruvocal/static/chatui/icon-128x128.png +0 -0
  473. package/src/ruvocal/static/chatui/icon-144x144.png +0 -0
  474. package/src/ruvocal/static/chatui/icon-192x192.png +0 -0
  475. package/src/ruvocal/static/chatui/icon-256x256.png +0 -0
  476. package/src/ruvocal/static/chatui/icon-36x36.png +0 -0
  477. package/src/ruvocal/static/chatui/icon-48x48.png +0 -0
  478. package/src/ruvocal/static/chatui/icon-512x512.png +0 -0
  479. package/src/ruvocal/static/chatui/icon-72x72.png +0 -0
  480. package/src/ruvocal/static/chatui/icon-96x96.png +0 -0
  481. package/src/ruvocal/static/chatui/icon.svg +3 -0
  482. package/src/ruvocal/static/chatui/logo.svg +7 -0
  483. package/src/ruvocal/static/chatui/manifest.json +54 -0
  484. package/src/ruvocal/static/chatui/omni-welcome.gif +0 -0
  485. package/src/ruvocal/static/chatui/omni-welcome.png +0 -0
  486. package/src/ruvocal/static/chatui/welcome.js +184 -0
  487. package/src/ruvocal/static/chatui/welcome.svg +1 -0
  488. package/src/ruvocal/static/huggingchat/apple-touch-icon.png +0 -0
  489. package/src/ruvocal/static/huggingchat/assistants-thumbnail.png +0 -0
  490. package/src/ruvocal/static/huggingchat/castle-example.jpg +0 -0
  491. package/src/ruvocal/static/huggingchat/favicon-dark.svg +4 -0
  492. package/src/ruvocal/static/huggingchat/favicon-dev.svg +4 -0
  493. package/src/ruvocal/static/huggingchat/favicon.ico +0 -0
  494. package/src/ruvocal/static/huggingchat/favicon.svg +4 -0
  495. package/src/ruvocal/static/huggingchat/fulltext-logo.svg +2 -0
  496. package/src/ruvocal/static/huggingchat/icon-128x128.png +0 -0
  497. package/src/ruvocal/static/huggingchat/icon-144x144.png +0 -0
  498. package/src/ruvocal/static/huggingchat/icon-192x192.png +0 -0
  499. package/src/ruvocal/static/huggingchat/icon-256x256.png +0 -0
  500. package/src/ruvocal/static/huggingchat/icon-36x36.png +0 -0
  501. package/src/ruvocal/static/huggingchat/icon-48x48.png +0 -0
  502. package/src/ruvocal/static/huggingchat/icon-512x512.png +0 -0
  503. package/src/ruvocal/static/huggingchat/icon-72x72.png +0 -0
  504. package/src/ruvocal/static/huggingchat/icon-96x96.png +0 -0
  505. package/src/ruvocal/static/huggingchat/icon.svg +4 -0
  506. package/src/ruvocal/static/huggingchat/logo.svg +4 -0
  507. package/src/ruvocal/static/huggingchat/manifest.json +54 -0
  508. package/src/ruvocal/static/huggingchat/omni-welcome.gif +0 -0
  509. package/src/ruvocal/static/huggingchat/routes.chat.json +226 -0
  510. package/src/ruvocal/static/huggingchat/thumbnail.png +0 -0
  511. package/src/ruvocal/static/huggingchat/tools-thumbnail.png +0 -0
  512. package/src/ruvocal/static/robots.txt +10 -0
  513. package/src/ruvocal/static/wasm/rvagent_wasm.js +1539 -0
  514. package/src/ruvocal/static/wasm/rvagent_wasm_bg.wasm +0 -0
  515. package/src/ruvocal/stub/@reflink/reflink/index.js +0 -0
  516. package/src/ruvocal/stub/@reflink/reflink/package.json +5 -0
  517. package/src/ruvocal/svelte.config.js +53 -0
  518. package/src/ruvocal/tailwind.config.cjs +30 -0
  519. package/src/ruvocal/tsconfig.json +19 -0
  520. package/src/ruvocal/vite.config.ts +87 -0
  521. package/src/scripts/deploy.sh +116 -0
  522. package/src/scripts/generate-config.js +245 -0
  523. package/src/scripts/generate-welcome.js +187 -0
  524. package/src/scripts/package-rvf.sh +116 -0
@@ -0,0 +1,77 @@
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" | "wasm";
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
+ // For WASM servers: active template info
70
+ wasmTemplateId?: string;
71
+ wasmTemplateName?: string;
72
+ }
73
+
74
+ export interface MCPServerApi {
75
+ url: string;
76
+ headers?: KeyValuePair[];
77
+ }
@@ -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
+ }
@@ -0,0 +1,64 @@
1
+ import { browser } from "$app/environment";
2
+ import type { WebHaptics } from "web-haptics";
3
+
4
+ let instance: WebHaptics | null = null;
5
+ let enabled = true;
6
+
7
+ /**
8
+ * Lazily initializes the WebHaptics instance on first use.
9
+ * Avoids importing at module level so SSR doesn't break.
10
+ */
11
+ async function getInstance(): Promise<WebHaptics | null> {
12
+ if (!browser || !supportsHaptics()) return null;
13
+ if (instance) return instance;
14
+
15
+ try {
16
+ const { WebHaptics: WH } = await import("web-haptics");
17
+ instance = new WH();
18
+ return instance;
19
+ } catch {
20
+ return null;
21
+ }
22
+ }
23
+
24
+ /** Call from the settings store to keep haptics in sync with user preference. */
25
+ export function setHapticsEnabled(value: boolean) {
26
+ enabled = value;
27
+ }
28
+
29
+ /** Whether the device likely supports haptic feedback (touch screen present). */
30
+ export function supportsHaptics(): boolean {
31
+ return browser && navigator.maxTouchPoints > 0;
32
+ }
33
+
34
+ // ── Internals ────────────────────────────────────────────────────────
35
+
36
+ /** Fire a haptic pattern, swallowing errors so callers can safely fire-and-forget. */
37
+ function fire(pattern: string): void {
38
+ if (!enabled) return;
39
+ Promise.resolve(getInstance())
40
+ .then((h) => h?.trigger(pattern))
41
+ .catch(() => {});
42
+ }
43
+
44
+ // ── Semantic haptic actions ──────────────────────────────────────────
45
+
46
+ /** Light tap — for routine actions (send message, toggle, navigate). */
47
+ export function tap() {
48
+ fire("light");
49
+ }
50
+
51
+ /** Success confirmation — double-tap pattern (copy, share, save). */
52
+ export function confirm() {
53
+ fire("success");
54
+ }
55
+
56
+ /** Error / destructive warning — three rapid taps (delete, stop generation). */
57
+ export function error() {
58
+ fire("error");
59
+ }
60
+
61
+ /** Selection change — subtle tap for pickers and selections. */
62
+ export function selection() {
63
+ fire("selection");
64
+ }
@@ -0,0 +1,12 @@
1
+ import type { Conversation } from "$lib/types/Conversation";
2
+ import { sha256 } from "./sha256";
3
+
4
+ export async function hashConv(conv: Conversation) {
5
+ // messages contains the conversation message but only the immutable part
6
+ const messages = conv.messages.map((message) => {
7
+ return (({ from, id, content }) => ({ from, id, content }))(message);
8
+ });
9
+
10
+ const hash = await sha256(JSON.stringify(messages));
11
+ return hash;
12
+ }
@@ -0,0 +1,17 @@
1
+ // Client-safe HF utilities used in UI components
2
+
3
+ export function isStrictHfMcpLogin(urlString: string): boolean {
4
+ try {
5
+ const u = new URL(urlString);
6
+ const host = u.hostname.toLowerCase();
7
+ const allowedHosts = new Set(["hf.co", "huggingface.co"]);
8
+ return (
9
+ u.protocol === "https:" &&
10
+ allowedHosts.has(host) &&
11
+ u.pathname === "/mcp" &&
12
+ u.search === "?login"
13
+ );
14
+ } catch {
15
+ return false;
16
+ }
17
+ }
@@ -0,0 +1,7 @@
1
+ // Approximate width from which we disable autofocus
2
+ const TABLET_VIEWPORT_WIDTH = 768;
3
+
4
+ export function isDesktop(window: Window) {
5
+ const { innerWidth } = window;
6
+ return innerWidth > TABLET_VIEWPORT_WIDTH;
7
+ }
@@ -0,0 +1,8 @@
1
+ export function isURL(url: string) {
2
+ try {
3
+ new URL(url);
4
+ return true;
5
+ } catch (e) {
6
+ return false;
7
+ }
8
+ }
@@ -0,0 +1,16 @@
1
+ import { browser } from "$app/environment";
2
+
3
+ export function isVirtualKeyboard(): boolean {
4
+ if (!browser) return false;
5
+
6
+ // Check for touch capability
7
+ if (navigator.maxTouchPoints > 0 && screen.width <= 768) return true;
8
+
9
+ // Check for touch events
10
+ if ("ontouchstart" in window) return true;
11
+
12
+ // Fallback to user agent string check
13
+ const userAgent = navigator.userAgent.toLowerCase();
14
+
15
+ return /android|webos|iphone|ipad|ipod|blackberry|iemobile|opera mini/i.test(userAgent);
16
+ }