@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,66 @@
1
+ import { buildPrompt } from "$lib/buildPrompt";
2
+ import { authCondition } from "$lib/server/auth";
3
+ import { collections } from "$lib/server/database";
4
+ import { models } from "$lib/server/models";
5
+ import { buildSubtree } from "$lib/utils/tree/buildSubtree";
6
+ import { isMessageId } from "$lib/utils/tree/isMessageId";
7
+ import { error } from "@sveltejs/kit";
8
+ import { ObjectId } from "mongodb";
9
+
10
+ export async function GET({ params, locals }) {
11
+ const conv =
12
+ params.id.length === 7
13
+ ? await collections.sharedConversations.findOne({
14
+ _id: params.id,
15
+ })
16
+ : await collections.conversations.findOne({
17
+ _id: new ObjectId(params.id),
18
+ ...authCondition(locals),
19
+ });
20
+
21
+ if (conv === null) {
22
+ error(404, "Conversation not found");
23
+ }
24
+
25
+ const messageId = params.messageId;
26
+
27
+ const messageIndex = conv.messages.findIndex((msg) => msg.id === messageId);
28
+
29
+ if (!isMessageId(messageId) || messageIndex === -1) {
30
+ error(404, "Message not found");
31
+ }
32
+
33
+ const model = models.find((m) => m.id === conv.model);
34
+
35
+ if (!model) {
36
+ error(404, "Conversation model not found");
37
+ }
38
+
39
+ const messagesUpTo = buildSubtree(conv, messageId);
40
+
41
+ const prompt = await buildPrompt({
42
+ preprompt: conv.preprompt,
43
+ messages: messagesUpTo,
44
+ model,
45
+ }).catch((err) => {
46
+ console.error(err);
47
+ return "Prompt generation failed";
48
+ });
49
+
50
+ return Response.json({
51
+ prompt,
52
+ model: model.name,
53
+ parameters: {
54
+ ...model.parameters,
55
+ return_full_text: false,
56
+ },
57
+ messages: messagesUpTo.map((msg) => ({
58
+ role: msg.from,
59
+ content: msg.content,
60
+ createdAt: msg.createdAt,
61
+ updatedAt: msg.updatedAt,
62
+ updates: msg.updates?.filter((u) => u.type === "title"),
63
+ files: msg.files,
64
+ })),
65
+ });
66
+ }
@@ -0,0 +1,69 @@
1
+ import { authCondition } from "$lib/server/auth";
2
+ import { collections } from "$lib/server/database";
3
+ import type { SharedConversation } from "$lib/types/SharedConversation";
4
+ import { hashConv } from "$lib/utils/hashConv";
5
+ import { error } from "@sveltejs/kit";
6
+ import { ObjectId } from "mongodb";
7
+ import { nanoid } from "nanoid";
8
+
9
+ export async function POST({ params, locals }) {
10
+ const conversation = await collections.conversations.findOne({
11
+ _id: new ObjectId(params.id),
12
+ ...authCondition(locals),
13
+ });
14
+
15
+ if (!conversation) {
16
+ error(404, "Conversation not found");
17
+ }
18
+
19
+ const hash = await hashConv(conversation);
20
+
21
+ const existingShare = await collections.sharedConversations.findOne({ hash });
22
+
23
+ if (existingShare) {
24
+ return new Response(
25
+ JSON.stringify({
26
+ shareId: existingShare._id,
27
+ }),
28
+ { headers: { "Content-Type": "application/json" } }
29
+ );
30
+ }
31
+
32
+ const shared: SharedConversation = {
33
+ _id: nanoid(7),
34
+ hash,
35
+ createdAt: new Date(),
36
+ updatedAt: new Date(),
37
+ rootMessageId: conversation.rootMessageId,
38
+ messages: conversation.messages,
39
+ title: conversation.title,
40
+ model: conversation.model,
41
+ preprompt: conversation.preprompt,
42
+ };
43
+
44
+ await collections.sharedConversations.insertOne(shared);
45
+
46
+ // copy files from `${conversation._id}-` to `${shared._id}-`
47
+ const files = await collections.bucket
48
+ .find({ filename: { $regex: `^${conversation._id}-` } })
49
+ .toArray();
50
+
51
+ await Promise.all(
52
+ files.map(async (file) => {
53
+ const newFilename = file.filename.replace(`${conversation._id}-`, `${shared._id}-`);
54
+ // copy files from `${conversation._id}-` to `${shared._id}-` by downloading and reuploaidng
55
+ const downloadStream = collections.bucket.openDownloadStream(file._id);
56
+ const uploadStream = collections.bucket.openUploadStream(newFilename, {
57
+ metadata: { ...file.metadata, conversation: shared._id.toString() },
58
+ });
59
+ downloadStream.pipe(uploadStream);
60
+ })
61
+ );
62
+
63
+ return new Response(
64
+ JSON.stringify({
65
+ shareId: shared._id,
66
+ }),
67
+ { headers: { "Content-Type": "application/json" } }
68
+ );
69
+ }
@@ -0,0 +1,35 @@
1
+ import { authCondition } from "$lib/server/auth";
2
+ import { collections } from "$lib/server/database";
3
+ import { AbortRegistry } from "$lib/server/abortRegistry";
4
+ import { error } from "@sveltejs/kit";
5
+ import { ObjectId } from "mongodb";
6
+
7
+ /**
8
+ * Ideally, we'd be able to detect the client-side abort, see https://github.com/huggingface/chat-ui/pull/88#issuecomment-1523173850
9
+ */
10
+ export async function POST({ params, locals }) {
11
+ if (!locals.user && !locals.sessionId) {
12
+ error(401, "Unauthorized");
13
+ }
14
+
15
+ const conversationId = new ObjectId(params.id);
16
+
17
+ const conversation = await collections.conversations.findOne({
18
+ _id: conversationId,
19
+ ...authCondition(locals),
20
+ });
21
+
22
+ if (!conversation) {
23
+ error(404, "Conversation not found");
24
+ }
25
+
26
+ AbortRegistry.getInstance().abort(conversationId.toString());
27
+
28
+ await collections.abortedGenerations.updateOne(
29
+ { conversationId },
30
+ { $set: { updatedAt: new Date() }, $setOnInsert: { createdAt: new Date() } },
31
+ { upsert: true }
32
+ );
33
+
34
+ return new Response();
35
+ }
@@ -0,0 +1,3 @@
1
+ export async function GET() {
2
+ return new Response("OK", { status: 200 });
3
+ }
@@ -0,0 +1,5 @@
1
+ import { triggerOauthFlow } from "$lib/server/auth";
2
+
3
+ export async function GET(event) {
4
+ return await triggerOauthFlow(event);
5
+ }
@@ -0,0 +1,103 @@
1
+ import { error, redirect } from "@sveltejs/kit";
2
+ import { getOIDCUserData, validateAndParseCsrfToken } from "$lib/server/auth";
3
+ import { z } from "zod";
4
+ import { base } from "$app/paths";
5
+ import { config } from "$lib/server/config";
6
+ import JSON5 from "json5";
7
+ import { updateUser } from "./updateUser.js";
8
+
9
+ const sanitizeJSONEnv = (val: string, fallback: string) => {
10
+ const raw = (val ?? "").trim();
11
+ const unquoted = raw.startsWith("`") && raw.endsWith("`") ? raw.slice(1, -1) : raw;
12
+ return unquoted || fallback;
13
+ };
14
+
15
+ const allowedUserEmails = z
16
+ .array(z.string().email())
17
+ .optional()
18
+ .default([])
19
+ .parse(JSON5.parse(sanitizeJSONEnv(config.ALLOWED_USER_EMAILS, "[]")));
20
+
21
+ const allowedUserDomains = z
22
+ .array(z.string().regex(/\.\w+$/)) // Contains at least a dot
23
+ .optional()
24
+ .default([])
25
+ .parse(JSON5.parse(sanitizeJSONEnv(config.ALLOWED_USER_DOMAINS, "[]")));
26
+
27
+ export async function GET({ url, locals, cookies, request, getClientAddress }) {
28
+ const { error: errorName, error_description: errorDescription } = z
29
+ .object({
30
+ error: z.string().optional(),
31
+ error_description: z.string().optional(),
32
+ })
33
+ .parse(Object.fromEntries(url.searchParams.entries()));
34
+
35
+ if (errorName) {
36
+ throw error(400, errorName + (errorDescription ? ": " + errorDescription : ""));
37
+ }
38
+
39
+ const { code, state, iss } = z
40
+ .object({
41
+ code: z.string(),
42
+ state: z.string(),
43
+ iss: z.string().optional(),
44
+ })
45
+ .parse(Object.fromEntries(url.searchParams.entries()));
46
+
47
+ const csrfToken = Buffer.from(state, "base64").toString("utf-8");
48
+
49
+ const validatedToken = await validateAndParseCsrfToken(csrfToken, locals.sessionId);
50
+
51
+ if (!validatedToken) {
52
+ throw error(403, "Invalid or expired CSRF token");
53
+ }
54
+
55
+ const codeVerifier = cookies.get("hfChat-codeVerifier");
56
+ if (!codeVerifier) {
57
+ throw error(403, "Code verifier cookie not found");
58
+ }
59
+
60
+ const { userData, token } = await getOIDCUserData(
61
+ { redirectURI: validatedToken.redirectUrl },
62
+ code,
63
+ codeVerifier,
64
+ iss,
65
+ url
66
+ );
67
+
68
+ // Filter by allowed user emails or domains
69
+ if (allowedUserEmails.length > 0 || allowedUserDomains.length > 0) {
70
+ if (!userData.email) {
71
+ throw error(403, "User not allowed: email not returned");
72
+ }
73
+ const emailVerified = userData.email_verified ?? true;
74
+ if (!emailVerified) {
75
+ throw error(403, "User not allowed: email not verified");
76
+ }
77
+
78
+ const emailDomain = userData.email.split("@")[1];
79
+ const isEmailAllowed = allowedUserEmails.includes(userData.email);
80
+ const isDomainAllowed = allowedUserDomains.includes(emailDomain);
81
+
82
+ if (!isEmailAllowed && !isDomainAllowed) {
83
+ throw error(403, "User not allowed");
84
+ }
85
+ }
86
+
87
+ await updateUser({
88
+ userData,
89
+ token,
90
+ locals,
91
+ cookies,
92
+ userAgent: request.headers.get("user-agent") ?? undefined,
93
+ ip: getClientAddress(),
94
+ });
95
+
96
+ // Prefer returning the user to their original in-app path when provided.
97
+ // `validatedToken.next` is sanitized server-side to avoid protocol-relative redirects.
98
+ const next = validatedToken.next;
99
+ if (next) {
100
+ return redirect(302, next);
101
+ }
102
+ return redirect(302, `${base}/`);
103
+ }
@@ -0,0 +1,157 @@
1
+ import { assert, it, describe, afterEach, vi, expect } from "vitest";
2
+ import type { Cookies } from "@sveltejs/kit";
3
+ import { collections } from "$lib/server/database";
4
+ import { updateUser } from "./updateUser";
5
+ import { ObjectId } from "mongodb";
6
+ import { DEFAULT_SETTINGS } from "$lib/types/Settings";
7
+ import { defaultModel } from "$lib/server/models";
8
+ import { findUser } from "$lib/server/auth";
9
+ import type { TokenSet } from "openid-client";
10
+
11
+ const userData = {
12
+ preferred_username: "new-username",
13
+ name: "name",
14
+ picture: "https://example.com/avatar.png",
15
+ sub: "1234567890",
16
+ };
17
+ Object.freeze(userData);
18
+
19
+ const locals = {
20
+ userId: "1234567890",
21
+ sessionId: "1234567890",
22
+ isAdmin: false,
23
+ };
24
+
25
+ const token = {
26
+ access_token: "access_token",
27
+ refresh_token: "refresh_token",
28
+ expires_at: Math.floor(Date.now() / 1000) + 3600, // Expires 1 hour from now
29
+ expires_in: 3600,
30
+ } as TokenSet;
31
+
32
+ // @ts-expect-error SvelteKit cookies dumb mock
33
+ const cookiesMock: Cookies = {
34
+ set: vi.fn(),
35
+ };
36
+
37
+ const insertRandomUser = async () => {
38
+ const res = await collections.users.insertOne({
39
+ _id: new ObjectId(),
40
+ createdAt: new Date(),
41
+ updatedAt: new Date(),
42
+ username: "base-username",
43
+ name: userData.name,
44
+ avatarUrl: userData.picture,
45
+ hfUserId: userData.sub,
46
+ });
47
+
48
+ return res.insertedId;
49
+ };
50
+
51
+ const insertRandomConversations = async (count: number) => {
52
+ const res = await collections.conversations.insertMany(
53
+ new Array(count).fill(0).map(() => ({
54
+ _id: new ObjectId(),
55
+ title: "random title",
56
+ messages: [],
57
+ model: defaultModel.id,
58
+ // embedding model removed in this build
59
+ createdAt: new Date(),
60
+ updatedAt: new Date(),
61
+ sessionId: locals.sessionId,
62
+ }))
63
+ );
64
+
65
+ return res.insertedIds;
66
+ };
67
+
68
+ describe("login", () => {
69
+ it("should update user if existing", async () => {
70
+ await insertRandomUser();
71
+
72
+ await updateUser({ userData, locals, cookies: cookiesMock, token });
73
+
74
+ const existingUser = await collections.users.findOne({ hfUserId: userData.sub });
75
+
76
+ assert.equal(existingUser?.name, userData.name);
77
+
78
+ expect(cookiesMock.set).toBeCalledTimes(1);
79
+ }, 30000);
80
+
81
+ it("should migrate pre-existing conversations for new user", async () => {
82
+ const insertedId = await insertRandomUser();
83
+
84
+ await insertRandomConversations(2);
85
+
86
+ await updateUser({ userData, locals, cookies: cookiesMock, token });
87
+
88
+ const conversationCount = await collections.conversations.countDocuments({
89
+ userId: insertedId,
90
+ sessionId: { $exists: false },
91
+ });
92
+
93
+ assert.equal(conversationCount, 2);
94
+
95
+ await collections.conversations.deleteMany({ userId: insertedId });
96
+ });
97
+
98
+ it("should create default settings for new user", async () => {
99
+ await updateUser({ userData, locals, cookies: cookiesMock, token });
100
+
101
+ // updateUser creates a new sessionId, so we need to use the updated value
102
+ const user = (await findUser(locals.sessionId, undefined, new URL("http://localhost"))).user;
103
+
104
+ assert.exists(user);
105
+
106
+ const settings = await collections.settings.findOne({ userId: user?._id });
107
+
108
+ expect(settings).toMatchObject({
109
+ userId: user?._id,
110
+ updatedAt: expect.any(Date),
111
+ createdAt: expect.any(Date),
112
+ ...DEFAULT_SETTINGS,
113
+ });
114
+
115
+ await collections.settings.deleteOne({ userId: user?._id });
116
+ });
117
+
118
+ it("should migrate pre-existing settings for pre-existing user", async () => {
119
+ const { insertedId } = await collections.settings.insertOne({
120
+ sessionId: locals.sessionId,
121
+ updatedAt: new Date(),
122
+ createdAt: new Date(),
123
+ ...DEFAULT_SETTINGS,
124
+ shareConversationsWithModelAuthors: false,
125
+ });
126
+
127
+ await updateUser({ userData, locals, cookies: cookiesMock, token });
128
+
129
+ const settings = await collections.settings.findOne({
130
+ _id: insertedId,
131
+ sessionId: { $exists: false },
132
+ });
133
+
134
+ assert.exists(settings);
135
+
136
+ const user = await collections.users.findOne({ hfUserId: userData.sub });
137
+
138
+ expect(settings).toMatchObject({
139
+ userId: user?._id,
140
+ updatedAt: expect.any(Date),
141
+ createdAt: expect.any(Date),
142
+ ...DEFAULT_SETTINGS,
143
+ shareConversationsWithModelAuthors: false,
144
+ });
145
+
146
+ await collections.settings.deleteOne({ userId: user?._id });
147
+ });
148
+ });
149
+
150
+ afterEach(async () => {
151
+ await collections.users.deleteMany({ hfUserId: userData.sub });
152
+ await collections.sessions.deleteMany({});
153
+
154
+ locals.userId = "1234567890";
155
+ locals.sessionId = "1234567890";
156
+ vi.clearAllMocks();
157
+ });
@@ -0,0 +1,215 @@
1
+ import {
2
+ getCoupledCookieHash,
3
+ refreshSessionCookie,
4
+ tokenSetToSessionOauth,
5
+ } from "$lib/server/auth";
6
+ import { collections } from "$lib/server/database";
7
+ import { ObjectId } from "mongodb";
8
+ import { DEFAULT_SETTINGS } from "$lib/types/Settings";
9
+ import { z } from "zod";
10
+ import type { UserinfoResponse, TokenSet } from "openid-client";
11
+ import { error, type Cookies } from "@sveltejs/kit";
12
+ import crypto from "crypto";
13
+ import { sha256 } from "$lib/utils/sha256";
14
+ import { addWeeks } from "date-fns";
15
+ import { OIDConfig } from "$lib/server/auth";
16
+ import { config } from "$lib/server/config";
17
+ import { logger } from "$lib/server/logger";
18
+
19
+ export async function updateUser(params: {
20
+ userData: UserinfoResponse;
21
+ token: TokenSet;
22
+ locals: App.Locals;
23
+ cookies: Cookies;
24
+ userAgent?: string;
25
+ ip?: string;
26
+ }) {
27
+ const { userData, token, locals, cookies, userAgent, ip } = params;
28
+
29
+ // Microsoft Entra v1 tokens do not provide preferred_username, instead the username is provided in the upn
30
+ // claim. See https://learn.microsoft.com/en-us/entra/identity-platform/access-token-claims-reference
31
+ if (!userData.preferred_username && userData.upn) {
32
+ userData.preferred_username = userData.upn as string;
33
+ }
34
+
35
+ const {
36
+ preferred_username: username,
37
+ name,
38
+ email,
39
+ picture: avatarUrl,
40
+ sub: hfUserId,
41
+ orgs,
42
+ } = z
43
+ .object({
44
+ preferred_username: z.string().optional(),
45
+ name: z.string(),
46
+ picture: z.string().optional(),
47
+ sub: z.string(),
48
+ email: z.string().email().optional(),
49
+ orgs: z
50
+ .array(
51
+ z.object({
52
+ sub: z.string(),
53
+ name: z.string(),
54
+ picture: z.string(),
55
+ preferred_username: z.string(),
56
+ plan: z.string().optional(),
57
+ })
58
+ )
59
+ .optional(),
60
+ })
61
+ .setKey(OIDConfig.NAME_CLAIM, z.string())
62
+ .refine((data) => data.preferred_username || data.email, {
63
+ message: "Either preferred_username or email must be provided by the provider.",
64
+ })
65
+ .transform((data) => ({
66
+ ...data,
67
+ name: data[OIDConfig.NAME_CLAIM],
68
+ }))
69
+ .parse(userData) as {
70
+ preferred_username?: string;
71
+ email?: string;
72
+ picture?: string;
73
+ sub: string;
74
+ name: string;
75
+ orgs?: Array<{
76
+ sub: string;
77
+ name: string;
78
+ picture: string;
79
+ preferred_username: string;
80
+ plan?: string;
81
+ }>;
82
+ } & Record<string, string>;
83
+
84
+ // Dynamically access user data based on NAME_CLAIM from environment
85
+ // This approach allows us to adapt to different OIDC providers flexibly.
86
+
87
+ logger.info(
88
+ {
89
+ login_username: username,
90
+ login_name: name,
91
+ login_email: email,
92
+ login_orgs: orgs?.map((el) => el.sub),
93
+ },
94
+ "user login"
95
+ );
96
+ // if using huggingface as auth provider, check orgs for earl access and amin rights
97
+ const isAdmin =
98
+ (config.HF_ORG_ADMIN && orgs?.some((org) => org.sub === config.HF_ORG_ADMIN)) || false;
99
+ const isEarlyAccess =
100
+ (config.HF_ORG_EARLY_ACCESS && orgs?.some((org) => org.sub === config.HF_ORG_EARLY_ACCESS)) ||
101
+ false;
102
+
103
+ logger.debug(
104
+ {
105
+ isAdmin,
106
+ isEarlyAccess,
107
+ hfUserId,
108
+ },
109
+ `Updating user ${hfUserId}`
110
+ );
111
+
112
+ // check if user already exists
113
+ const existingUser = await collections.users.findOne({ hfUserId });
114
+ let userId = existingUser?._id;
115
+
116
+ // update session cookie on login
117
+ const previousSessionId = locals.sessionId;
118
+ const secretSessionId = crypto.randomUUID();
119
+ const sessionId = await sha256(secretSessionId);
120
+
121
+ if (await collections.sessions.findOne({ sessionId })) {
122
+ error(500, "Session ID collision");
123
+ }
124
+
125
+ locals.sessionId = sessionId;
126
+
127
+ // Get cookie hash if coupling is enabled
128
+ const coupledCookieHash = await getCoupledCookieHash(cookies);
129
+
130
+ // Prepare OAuth token data for session storage
131
+ const oauthData = tokenSetToSessionOauth(token);
132
+
133
+ if (existingUser) {
134
+ // update existing user if any
135
+ await collections.users.updateOne(
136
+ { _id: existingUser._id },
137
+ { $set: { username, name, avatarUrl, isAdmin, isEarlyAccess } }
138
+ );
139
+
140
+ // remove previous session if it exists and add new one
141
+ await collections.sessions.deleteOne({ sessionId: previousSessionId });
142
+ await collections.sessions.insertOne({
143
+ _id: new ObjectId(),
144
+ sessionId: locals.sessionId,
145
+ userId: existingUser._id,
146
+ createdAt: new Date(),
147
+ updatedAt: new Date(),
148
+ userAgent,
149
+ ip,
150
+ expiresAt: addWeeks(new Date(), 2),
151
+ ...(coupledCookieHash ? { coupledCookieHash } : {}),
152
+ ...(oauthData ? { oauth: oauthData } : {}),
153
+ });
154
+ } else {
155
+ // user doesn't exist yet, create a new one
156
+ const { insertedId } = await collections.users.insertOne({
157
+ _id: new ObjectId(),
158
+ createdAt: new Date(),
159
+ updatedAt: new Date(),
160
+ username,
161
+ name,
162
+ email,
163
+ avatarUrl,
164
+ hfUserId,
165
+ isAdmin,
166
+ isEarlyAccess,
167
+ });
168
+
169
+ userId = insertedId;
170
+
171
+ await collections.sessions.insertOne({
172
+ _id: new ObjectId(),
173
+ sessionId: locals.sessionId,
174
+ userId,
175
+ createdAt: new Date(),
176
+ updatedAt: new Date(),
177
+ userAgent,
178
+ ip,
179
+ expiresAt: addWeeks(new Date(), 2),
180
+ ...(coupledCookieHash ? { coupledCookieHash } : {}),
181
+ ...(oauthData ? { oauth: oauthData } : {}),
182
+ });
183
+
184
+ // move pre-existing settings to new user
185
+ const { matchedCount } = await collections.settings.updateOne(
186
+ { sessionId: previousSessionId },
187
+ {
188
+ $set: { userId, updatedAt: new Date() },
189
+ $unset: { sessionId: "" },
190
+ }
191
+ );
192
+
193
+ if (!matchedCount) {
194
+ // if no settings found for user, create default settings
195
+ await collections.settings.insertOne({
196
+ userId,
197
+ updatedAt: new Date(),
198
+ createdAt: new Date(),
199
+ ...DEFAULT_SETTINGS,
200
+ });
201
+ }
202
+ }
203
+
204
+ // refresh session cookie
205
+ refreshSessionCookie(cookies, secretSessionId);
206
+
207
+ // migrate pre-existing conversations
208
+ await collections.conversations.updateMany(
209
+ { sessionId: previousSessionId },
210
+ {
211
+ $set: { userId },
212
+ $unset: { sessionId: "" },
213
+ }
214
+ );
215
+ }
@@ -0,0 +1,18 @@
1
+ import { dev } from "$app/environment";
2
+ import { base } from "$app/paths";
3
+ import { collections } from "$lib/server/database";
4
+ import { redirect } from "@sveltejs/kit";
5
+ import { config } from "$lib/server/config";
6
+
7
+ export async function POST({ locals, cookies }) {
8
+ await collections.sessions.deleteOne({ sessionId: locals.sessionId });
9
+
10
+ cookies.delete(config.COOKIE_NAME, {
11
+ path: "/",
12
+ // So that it works inside the space's iframe
13
+ sameSite: dev || config.ALLOW_INSECURE_COOKIES === "true" ? "lax" : "none",
14
+ secure: !dev && !(config.ALLOW_INSECURE_COOKIES === "true"),
15
+ httpOnly: true,
16
+ });
17
+ return redirect(302, `${base}/`);
18
+ }
@@ -0,0 +1,18 @@
1
+ import { config } from "$lib/server/config";
2
+ import { MetricsServer } from "$lib/server/metrics";
3
+
4
+ export async function GET() {
5
+ if (config.METRICS_ENABLED !== "true") {
6
+ return new Response("Not Found", { status: 404 });
7
+ }
8
+
9
+ const payload = await MetricsServer.getInstance().render();
10
+
11
+ return new Response(payload, {
12
+ status: 200,
13
+ headers: {
14
+ "Content-Type": "text/plain; version=0.0.4",
15
+ "Cache-Control": "no-store",
16
+ },
17
+ });
18
+ }