ruflo 3.5.1 → 3.5.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (521) hide show
  1. package/dist/rvf.manifest.json +295 -0
  2. package/package.json +16 -2
  3. package/src/chat-ui/Dockerfile +25 -0
  4. package/src/chat-ui/patch-mcp-url-safety.sh +28 -0
  5. package/src/chat-ui/static/chatui/icon-144x144.png +0 -0
  6. package/src/chat-ui/static/chatui/omni-welcome.gif +0 -0
  7. package/src/config/config.example.json +76 -0
  8. package/src/mcp-bridge/Dockerfile +45 -0
  9. package/src/mcp-bridge/index.js +1668 -0
  10. package/src/mcp-bridge/mcp-stdio-kernel.js +159 -0
  11. package/src/mcp-bridge/package.json +17 -0
  12. package/src/mcp-bridge/test-harness.js +470 -0
  13. package/src/nginx/Dockerfile +10 -0
  14. package/src/nginx/nginx.conf +67 -0
  15. package/src/nginx/static/favicon-dark.svg +4 -0
  16. package/src/nginx/static/favicon.svg +4 -0
  17. package/src/nginx/static/icon.svg +5 -0
  18. package/src/nginx/static/logo.svg +9 -0
  19. package/src/nginx/static/manifest.json +22 -0
  20. package/src/nginx/static/welcome.js +184 -0
  21. package/src/ruvocal/.claude/skills/add-model-descriptions/SKILL.md +73 -0
  22. package/src/ruvocal/.devcontainer/Dockerfile +9 -0
  23. package/src/ruvocal/.devcontainer/devcontainer.json +36 -0
  24. package/src/ruvocal/.dockerignore +13 -0
  25. package/src/ruvocal/.env +194 -0
  26. package/src/ruvocal/.env.ci +1 -0
  27. package/src/ruvocal/.eslintignore +13 -0
  28. package/src/ruvocal/.eslintrc.cjs +45 -0
  29. package/src/ruvocal/.github/ISSUE_TEMPLATE/bug-report--chat-ui-.md +43 -0
  30. package/src/ruvocal/.github/ISSUE_TEMPLATE/config-support.md +9 -0
  31. package/src/ruvocal/.github/ISSUE_TEMPLATE/feature-request--chat-ui-.md +17 -0
  32. package/src/ruvocal/.github/ISSUE_TEMPLATE/huggingchat.md +11 -0
  33. package/src/ruvocal/.github/release.yml +16 -0
  34. package/src/ruvocal/.github/workflows/build-docs.yml +18 -0
  35. package/src/ruvocal/.github/workflows/build-image.yml +142 -0
  36. package/src/ruvocal/.github/workflows/build-pr-docs.yml +20 -0
  37. package/src/ruvocal/.github/workflows/deploy-dev.yml +63 -0
  38. package/src/ruvocal/.github/workflows/deploy-prod.yml +78 -0
  39. package/src/ruvocal/.github/workflows/lint-and-test.yml +84 -0
  40. package/src/ruvocal/.github/workflows/slugify.yaml +72 -0
  41. package/src/ruvocal/.github/workflows/trufflehog.yml +17 -0
  42. package/src/ruvocal/.github/workflows/upload-pr-documentation.yml +16 -0
  43. package/src/ruvocal/.husky/lint-stage-config.js +4 -0
  44. package/src/ruvocal/.husky/pre-commit +2 -0
  45. package/src/ruvocal/.prettierignore +14 -0
  46. package/src/ruvocal/.prettierrc +7 -0
  47. package/src/ruvocal/.vscode/launch.json +11 -0
  48. package/src/ruvocal/.vscode/settings.json +14 -0
  49. package/src/ruvocal/CLAUDE.md +126 -0
  50. package/src/ruvocal/Dockerfile +93 -0
  51. package/src/ruvocal/LICENSE +203 -0
  52. package/src/ruvocal/PRIVACY.md +41 -0
  53. package/src/ruvocal/README.md +190 -0
  54. package/src/ruvocal/chart/Chart.yaml +5 -0
  55. package/src/ruvocal/chart/env/dev.yaml +260 -0
  56. package/src/ruvocal/chart/env/prod.yaml +273 -0
  57. package/src/ruvocal/chart/templates/_helpers.tpl +22 -0
  58. package/src/ruvocal/chart/templates/config.yaml +10 -0
  59. package/src/ruvocal/chart/templates/deployment.yaml +81 -0
  60. package/src/ruvocal/chart/templates/hpa.yaml +45 -0
  61. package/src/ruvocal/chart/templates/infisical.yaml +24 -0
  62. package/src/ruvocal/chart/templates/ingress-internal.yaml +32 -0
  63. package/src/ruvocal/chart/templates/ingress.yaml +32 -0
  64. package/src/ruvocal/chart/templates/network-policy.yaml +36 -0
  65. package/src/ruvocal/chart/templates/service-account.yaml +13 -0
  66. package/src/ruvocal/chart/templates/service-monitor.yaml +17 -0
  67. package/src/ruvocal/chart/templates/service.yaml +21 -0
  68. package/src/ruvocal/chart/values.yaml +73 -0
  69. package/src/ruvocal/docker-compose.yml +21 -0
  70. package/src/ruvocal/docs/adr/ADR-029-HUGGINGFACE-CHAT-UI-CLOUD-RUN.md +1236 -0
  71. package/src/ruvocal/docs/adr/ADR-033-RUVECTOR-RUFLO-MCP-INTEGRATION.md +111 -0
  72. package/src/ruvocal/docs/adr/ADR-034-OPTIONAL-MCP-BACKENDS.md +117 -0
  73. package/src/ruvocal/docs/adr/ADR-035-MCP-TOOL-GROUPS.md +186 -0
  74. package/src/ruvocal/docs/adr/ADR-037-AUTOPILOT-CHAT-MODE.md +1500 -0
  75. package/src/ruvocal/docs/adr/ADR-038-RUVOCAL-FORK.md +286 -0
  76. package/src/ruvocal/docs/source/_toctree.yml +30 -0
  77. package/src/ruvocal/docs/source/configuration/common-issues.md +38 -0
  78. package/src/ruvocal/docs/source/configuration/llm-router.md +105 -0
  79. package/src/ruvocal/docs/source/configuration/mcp-tools.md +84 -0
  80. package/src/ruvocal/docs/source/configuration/metrics.md +9 -0
  81. package/src/ruvocal/docs/source/configuration/open-id.md +57 -0
  82. package/src/ruvocal/docs/source/configuration/overview.md +89 -0
  83. package/src/ruvocal/docs/source/configuration/theming.md +20 -0
  84. package/src/ruvocal/docs/source/developing/architecture.md +48 -0
  85. package/src/ruvocal/docs/source/index.md +53 -0
  86. package/src/ruvocal/docs/source/installation/docker.md +43 -0
  87. package/src/ruvocal/docs/source/installation/helm.md +43 -0
  88. package/src/ruvocal/docs/source/installation/local.md +62 -0
  89. package/src/ruvocal/entrypoint.sh +19 -0
  90. package/src/ruvocal/mcp-bridge/.claude-flow/agents/store.json +27 -0
  91. package/src/ruvocal/mcp-bridge/.claude-flow/daemon-state.json +130 -0
  92. package/src/ruvocal/mcp-bridge/.claude-flow/daemon.log +0 -0
  93. package/src/ruvocal/mcp-bridge/.claude-flow/daemon.pid +1 -0
  94. package/src/ruvocal/mcp-bridge/.claude-flow/tasks/store.json +21 -0
  95. package/src/ruvocal/mcp-bridge/.swarm/hnsw.index +0 -0
  96. package/src/ruvocal/mcp-bridge/.swarm/hnsw.metadata.json +1 -0
  97. package/src/ruvocal/mcp-bridge/.swarm/memory.db +0 -0
  98. package/src/ruvocal/mcp-bridge/.swarm/model-router-state.json +14 -0
  99. package/src/ruvocal/mcp-bridge/.swarm/schema.sql +305 -0
  100. package/src/ruvocal/mcp-bridge/Dockerfile +45 -0
  101. package/src/ruvocal/mcp-bridge/cloudbuild.yaml +49 -0
  102. package/src/ruvocal/mcp-bridge/index.js +1864 -0
  103. package/src/ruvocal/mcp-bridge/mcp-stdio-kernel.js +159 -0
  104. package/src/ruvocal/mcp-bridge/package-lock.json +762 -0
  105. package/src/ruvocal/mcp-bridge/package.json +17 -0
  106. package/src/ruvocal/mcp-bridge/test-harness.js +470 -0
  107. package/src/ruvocal/models/add-your-models-here.txt +1 -0
  108. package/src/ruvocal/package-lock.json +11741 -0
  109. package/src/ruvocal/package.json +121 -0
  110. package/src/ruvocal/postcss.config.js +6 -0
  111. package/src/ruvocal/rvf.manifest.json +204 -0
  112. package/src/ruvocal/scripts/config.ts +64 -0
  113. package/src/ruvocal/scripts/generate-welcome.mjs +181 -0
  114. package/src/ruvocal/scripts/populate.ts +288 -0
  115. package/src/ruvocal/scripts/samples.txt +194 -0
  116. package/src/ruvocal/scripts/setups/vitest-setup-client.ts +0 -0
  117. package/src/ruvocal/scripts/setups/vitest-setup-server.ts +44 -0
  118. package/src/ruvocal/scripts/updateLocalEnv.ts +48 -0
  119. package/src/ruvocal/src/ambient.d.ts +7 -0
  120. package/src/ruvocal/src/app.d.ts +29 -0
  121. package/src/ruvocal/src/app.html +53 -0
  122. package/src/ruvocal/src/hooks.server.ts +32 -0
  123. package/src/ruvocal/src/hooks.ts +6 -0
  124. package/src/ruvocal/src/lib/APIClient.ts +148 -0
  125. package/src/ruvocal/src/lib/actions/clickOutside.ts +18 -0
  126. package/src/ruvocal/src/lib/actions/snapScrollToBottom.ts +346 -0
  127. package/src/ruvocal/src/lib/buildPrompt.ts +33 -0
  128. package/src/ruvocal/src/lib/components/AnnouncementBanner.svelte +20 -0
  129. package/src/ruvocal/src/lib/components/BackgroundGenerationPoller.svelte +168 -0
  130. package/src/ruvocal/src/lib/components/CodeBlock.svelte +73 -0
  131. package/src/ruvocal/src/lib/components/CopyToClipBoardBtn.svelte +92 -0
  132. package/src/ruvocal/src/lib/components/DeleteConversationModal.svelte +75 -0
  133. package/src/ruvocal/src/lib/components/EditConversationModal.svelte +100 -0
  134. package/src/ruvocal/src/lib/components/ExpandNavigation.svelte +22 -0
  135. package/src/ruvocal/src/lib/components/HoverTooltip.svelte +44 -0
  136. package/src/ruvocal/src/lib/components/HtmlPreviewModal.svelte +143 -0
  137. package/src/ruvocal/src/lib/components/InfiniteScroll.svelte +50 -0
  138. package/src/ruvocal/src/lib/components/MobileNav.svelte +300 -0
  139. package/src/ruvocal/src/lib/components/Modal.svelte +115 -0
  140. package/src/ruvocal/src/lib/components/ModelCardMetadata.svelte +71 -0
  141. package/src/ruvocal/src/lib/components/NavConversationItem.svelte +151 -0
  142. package/src/ruvocal/src/lib/components/NavMenu.svelte +295 -0
  143. package/src/ruvocal/src/lib/components/Pagination.svelte +97 -0
  144. package/src/ruvocal/src/lib/components/PaginationArrow.svelte +27 -0
  145. package/src/ruvocal/src/lib/components/Portal.svelte +24 -0
  146. package/src/ruvocal/src/lib/components/RetryBtn.svelte +18 -0
  147. package/src/ruvocal/src/lib/components/RuFloUniverse.svelte +185 -0
  148. package/src/ruvocal/src/lib/components/ScrollToBottomBtn.svelte +47 -0
  149. package/src/ruvocal/src/lib/components/ScrollToPreviousBtn.svelte +77 -0
  150. package/src/ruvocal/src/lib/components/ShareConversationModal.svelte +182 -0
  151. package/src/ruvocal/src/lib/components/StopGeneratingBtn.svelte +69 -0
  152. package/src/ruvocal/src/lib/components/SubscribeModal.svelte +87 -0
  153. package/src/ruvocal/src/lib/components/Switch.svelte +36 -0
  154. package/src/ruvocal/src/lib/components/SystemPromptModal.svelte +44 -0
  155. package/src/ruvocal/src/lib/components/Toast.svelte +27 -0
  156. package/src/ruvocal/src/lib/components/Tooltip.svelte +30 -0
  157. package/src/ruvocal/src/lib/components/WelcomeModal.svelte +46 -0
  158. package/src/ruvocal/src/lib/components/chat/Alternatives.svelte +77 -0
  159. package/src/ruvocal/src/lib/components/chat/BlockWrapper.svelte +72 -0
  160. package/src/ruvocal/src/lib/components/chat/ChatInput.svelte +490 -0
  161. package/src/ruvocal/src/lib/components/chat/ChatIntroduction.svelte +123 -0
  162. package/src/ruvocal/src/lib/components/chat/ChatMessage.svelte +548 -0
  163. package/src/ruvocal/src/lib/components/chat/ChatWindow.svelte +939 -0
  164. package/src/ruvocal/src/lib/components/chat/FileDropzone.svelte +92 -0
  165. package/src/ruvocal/src/lib/components/chat/ImageLightbox.svelte +66 -0
  166. package/src/ruvocal/src/lib/components/chat/MarkdownBlock.svelte +23 -0
  167. package/src/ruvocal/src/lib/components/chat/MarkdownRenderer.svelte +69 -0
  168. package/src/ruvocal/src/lib/components/chat/MarkdownRenderer.svelte.test.ts +58 -0
  169. package/src/ruvocal/src/lib/components/chat/MessageAvatar.svelte +103 -0
  170. package/src/ruvocal/src/lib/components/chat/ModelSwitch.svelte +64 -0
  171. package/src/ruvocal/src/lib/components/chat/OpenReasoningResults.svelte +81 -0
  172. package/src/ruvocal/src/lib/components/chat/TaskGroup.svelte +88 -0
  173. package/src/ruvocal/src/lib/components/chat/ToolUpdate.svelte +273 -0
  174. package/src/ruvocal/src/lib/components/chat/UploadedFile.svelte +253 -0
  175. package/src/ruvocal/src/lib/components/chat/UrlFetchModal.svelte +203 -0
  176. package/src/ruvocal/src/lib/components/chat/VoiceRecorder.svelte +214 -0
  177. package/src/ruvocal/src/lib/components/icons/IconBurger.svelte +20 -0
  178. package/src/ruvocal/src/lib/components/icons/IconCheap.svelte +20 -0
  179. package/src/ruvocal/src/lib/components/icons/IconChevron.svelte +24 -0
  180. package/src/ruvocal/src/lib/components/icons/IconDazzled.svelte +40 -0
  181. package/src/ruvocal/src/lib/components/icons/IconFast.svelte +20 -0
  182. package/src/ruvocal/src/lib/components/icons/IconLoading.svelte +22 -0
  183. package/src/ruvocal/src/lib/components/icons/IconMCP.svelte +28 -0
  184. package/src/ruvocal/src/lib/components/icons/IconMoon.svelte +21 -0
  185. package/src/ruvocal/src/lib/components/icons/IconNew.svelte +20 -0
  186. package/src/ruvocal/src/lib/components/icons/IconOmni.svelte +90 -0
  187. package/src/ruvocal/src/lib/components/icons/IconPaperclip.svelte +24 -0
  188. package/src/ruvocal/src/lib/components/icons/IconPro.svelte +37 -0
  189. package/src/ruvocal/src/lib/components/icons/IconShare.svelte +21 -0
  190. package/src/ruvocal/src/lib/components/icons/IconSun.svelte +93 -0
  191. package/src/ruvocal/src/lib/components/icons/Logo.svelte +68 -0
  192. package/src/ruvocal/src/lib/components/icons/LogoHuggingFaceBorderless.svelte +54 -0
  193. package/src/ruvocal/src/lib/components/mcp/AddServerForm.svelte +250 -0
  194. package/src/ruvocal/src/lib/components/mcp/MCPServerManager.svelte +185 -0
  195. package/src/ruvocal/src/lib/components/mcp/ServerCard.svelte +203 -0
  196. package/src/ruvocal/src/lib/components/players/AudioPlayer.svelte +82 -0
  197. package/src/ruvocal/src/lib/components/voice/AudioWaveform.svelte +96 -0
  198. package/src/ruvocal/src/lib/constants/mcpExamples.ts +135 -0
  199. package/src/ruvocal/src/lib/constants/mime.ts +11 -0
  200. package/src/ruvocal/src/lib/constants/pagination.ts +1 -0
  201. package/src/ruvocal/src/lib/constants/publicSepToken.ts +1 -0
  202. package/src/ruvocal/src/lib/constants/routerExamples.ts +209 -0
  203. package/src/ruvocal/src/lib/createShareLink.ts +27 -0
  204. package/src/ruvocal/src/lib/jobs/refresh-conversation-stats.ts +297 -0
  205. package/src/ruvocal/src/lib/migrations/lock.ts +56 -0
  206. package/src/ruvocal/src/lib/migrations/migrations.spec.ts +74 -0
  207. package/src/ruvocal/src/lib/migrations/migrations.ts +109 -0
  208. package/src/ruvocal/src/lib/migrations/routines/01-update-search-assistants.ts +50 -0
  209. package/src/ruvocal/src/lib/migrations/routines/02-update-assistants-models.ts +48 -0
  210. package/src/ruvocal/src/lib/migrations/routines/04-update-message-updates.ts +151 -0
  211. package/src/ruvocal/src/lib/migrations/routines/05-update-message-files.ts +56 -0
  212. package/src/ruvocal/src/lib/migrations/routines/06-trim-message-updates.ts +56 -0
  213. package/src/ruvocal/src/lib/migrations/routines/08-update-featured-to-review.ts +32 -0
  214. package/src/ruvocal/src/lib/migrations/routines/09-delete-empty-conversations.spec.ts +214 -0
  215. package/src/ruvocal/src/lib/migrations/routines/09-delete-empty-conversations.ts +88 -0
  216. package/src/ruvocal/src/lib/migrations/routines/10-update-reports-assistantid.ts +29 -0
  217. package/src/ruvocal/src/lib/migrations/routines/index.ts +15 -0
  218. package/src/ruvocal/src/lib/server/__tests__/conversation-stop-generating.spec.ts +103 -0
  219. package/src/ruvocal/src/lib/server/abortRegistry.ts +57 -0
  220. package/src/ruvocal/src/lib/server/abortedGenerations.ts +43 -0
  221. package/src/ruvocal/src/lib/server/adminToken.ts +62 -0
  222. package/src/ruvocal/src/lib/server/api/__tests__/conversations-id.spec.ts +296 -0
  223. package/src/ruvocal/src/lib/server/api/__tests__/conversations-message.spec.ts +216 -0
  224. package/src/ruvocal/src/lib/server/api/__tests__/conversations.spec.ts +235 -0
  225. package/src/ruvocal/src/lib/server/api/__tests__/misc.spec.ts +72 -0
  226. package/src/ruvocal/src/lib/server/api/__tests__/testHelpers.ts +86 -0
  227. package/src/ruvocal/src/lib/server/api/__tests__/user-reports.spec.ts +78 -0
  228. package/src/ruvocal/src/lib/server/api/__tests__/user.spec.ts +239 -0
  229. package/src/ruvocal/src/lib/server/api/types.ts +37 -0
  230. package/src/ruvocal/src/lib/server/api/utils/requireAuth.ts +22 -0
  231. package/src/ruvocal/src/lib/server/api/utils/resolveConversation.ts +69 -0
  232. package/src/ruvocal/src/lib/server/api/utils/resolveModel.ts +27 -0
  233. package/src/ruvocal/src/lib/server/api/utils/superjsonResponse.ts +15 -0
  234. package/src/ruvocal/src/lib/server/apiToken.ts +11 -0
  235. package/src/ruvocal/src/lib/server/auth.ts +554 -0
  236. package/src/ruvocal/src/lib/server/config.ts +187 -0
  237. package/src/ruvocal/src/lib/server/conversation.ts +83 -0
  238. package/src/ruvocal/src/lib/server/database/__tests__/rvf.spec.ts +709 -0
  239. package/src/ruvocal/src/lib/server/database/postgres.ts +700 -0
  240. package/src/ruvocal/src/lib/server/database/rvf.ts +1078 -0
  241. package/src/ruvocal/src/lib/server/database.ts +145 -0
  242. package/src/ruvocal/src/lib/server/endpoints/document.ts +68 -0
  243. package/src/ruvocal/src/lib/server/endpoints/endpoints.ts +43 -0
  244. package/src/ruvocal/src/lib/server/endpoints/images.ts +211 -0
  245. package/src/ruvocal/src/lib/server/endpoints/openai/endpointOai.ts +266 -0
  246. package/src/ruvocal/src/lib/server/endpoints/openai/openAIChatToTextGenerationStream.ts +212 -0
  247. package/src/ruvocal/src/lib/server/endpoints/openai/openAICompletionToTextGenerationStream.ts +32 -0
  248. package/src/ruvocal/src/lib/server/endpoints/preprocessMessages.ts +61 -0
  249. package/src/ruvocal/src/lib/server/exitHandler.ts +59 -0
  250. package/src/ruvocal/src/lib/server/files/downloadFile.ts +34 -0
  251. package/src/ruvocal/src/lib/server/files/uploadFile.ts +29 -0
  252. package/src/ruvocal/src/lib/server/findRepoRoot.ts +13 -0
  253. package/src/ruvocal/src/lib/server/fonts/Inter-Black.ttf +0 -0
  254. package/src/ruvocal/src/lib/server/fonts/Inter-Bold.ttf +0 -0
  255. package/src/ruvocal/src/lib/server/fonts/Inter-ExtraBold.ttf +0 -0
  256. package/src/ruvocal/src/lib/server/fonts/Inter-ExtraLight.ttf +0 -0
  257. package/src/ruvocal/src/lib/server/fonts/Inter-Light.ttf +0 -0
  258. package/src/ruvocal/src/lib/server/fonts/Inter-Medium.ttf +0 -0
  259. package/src/ruvocal/src/lib/server/fonts/Inter-Regular.ttf +0 -0
  260. package/src/ruvocal/src/lib/server/fonts/Inter-SemiBold.ttf +0 -0
  261. package/src/ruvocal/src/lib/server/fonts/Inter-Thin.ttf +0 -0
  262. package/src/ruvocal/src/lib/server/generateFromDefaultEndpoint.ts +46 -0
  263. package/src/ruvocal/src/lib/server/hooks/error.ts +37 -0
  264. package/src/ruvocal/src/lib/server/hooks/fetch.ts +22 -0
  265. package/src/ruvocal/src/lib/server/hooks/handle.ts +250 -0
  266. package/src/ruvocal/src/lib/server/hooks/init.ts +51 -0
  267. package/src/ruvocal/src/lib/server/isURLLocal.spec.ts +31 -0
  268. package/src/ruvocal/src/lib/server/isURLLocal.ts +74 -0
  269. package/src/ruvocal/src/lib/server/logger.ts +42 -0
  270. package/src/ruvocal/src/lib/server/mcp/clientPool.ts +70 -0
  271. package/src/ruvocal/src/lib/server/mcp/hf.ts +32 -0
  272. package/src/ruvocal/src/lib/server/mcp/httpClient.ts +122 -0
  273. package/src/ruvocal/src/lib/server/mcp/registry.ts +76 -0
  274. package/src/ruvocal/src/lib/server/mcp/tools.ts +196 -0
  275. package/src/ruvocal/src/lib/server/metrics.ts +255 -0
  276. package/src/ruvocal/src/lib/server/models.ts +518 -0
  277. package/src/ruvocal/src/lib/server/requestContext.ts +55 -0
  278. package/src/ruvocal/src/lib/server/router/arch.ts +230 -0
  279. package/src/ruvocal/src/lib/server/router/endpoint.ts +316 -0
  280. package/src/ruvocal/src/lib/server/router/multimodal.ts +28 -0
  281. package/src/ruvocal/src/lib/server/router/policy.ts +49 -0
  282. package/src/ruvocal/src/lib/server/router/toolsRoute.ts +51 -0
  283. package/src/ruvocal/src/lib/server/router/types.ts +21 -0
  284. package/src/ruvocal/src/lib/server/sendSlack.ts +23 -0
  285. package/src/ruvocal/src/lib/server/textGeneration/generate.ts +258 -0
  286. package/src/ruvocal/src/lib/server/textGeneration/index.ts +95 -0
  287. package/src/ruvocal/src/lib/server/textGeneration/mcp/fileRefs.ts +155 -0
  288. package/src/ruvocal/src/lib/server/textGeneration/mcp/routerResolution.ts +108 -0
  289. package/src/ruvocal/src/lib/server/textGeneration/mcp/runMcpFlow.ts +822 -0
  290. package/src/ruvocal/src/lib/server/textGeneration/mcp/toolInvocation.ts +349 -0
  291. package/src/ruvocal/src/lib/server/textGeneration/reasoning.ts +23 -0
  292. package/src/ruvocal/src/lib/server/textGeneration/title.ts +83 -0
  293. package/src/ruvocal/src/lib/server/textGeneration/types.ts +26 -0
  294. package/src/ruvocal/src/lib/server/textGeneration/utils/prepareFiles.ts +88 -0
  295. package/src/ruvocal/src/lib/server/textGeneration/utils/routing.ts +21 -0
  296. package/src/ruvocal/src/lib/server/textGeneration/utils/toolPrompt.ts +49 -0
  297. package/src/ruvocal/src/lib/server/urlSafety.ts +72 -0
  298. package/src/ruvocal/src/lib/server/usageLimits.ts +30 -0
  299. package/src/ruvocal/src/lib/stores/autopilotStore.svelte.ts +175 -0
  300. package/src/ruvocal/src/lib/stores/backgroundGenerations.svelte.ts +32 -0
  301. package/src/ruvocal/src/lib/stores/backgroundGenerations.ts +1 -0
  302. package/src/ruvocal/src/lib/stores/errors.ts +9 -0
  303. package/src/ruvocal/src/lib/stores/isAborted.ts +3 -0
  304. package/src/ruvocal/src/lib/stores/isPro.ts +4 -0
  305. package/src/ruvocal/src/lib/stores/loading.ts +3 -0
  306. package/src/ruvocal/src/lib/stores/mcpServers.ts +345 -0
  307. package/src/ruvocal/src/lib/stores/pendingChatInput.ts +3 -0
  308. package/src/ruvocal/src/lib/stores/pendingMessage.ts +9 -0
  309. package/src/ruvocal/src/lib/stores/settings.ts +182 -0
  310. package/src/ruvocal/src/lib/stores/shareModal.ts +13 -0
  311. package/src/ruvocal/src/lib/stores/titleUpdate.ts +8 -0
  312. package/src/ruvocal/src/lib/switchTheme.ts +124 -0
  313. package/src/ruvocal/src/lib/types/AbortedGeneration.ts +8 -0
  314. package/src/ruvocal/src/lib/types/Assistant.ts +31 -0
  315. package/src/ruvocal/src/lib/types/AssistantStats.ts +11 -0
  316. package/src/ruvocal/src/lib/types/ConfigKey.ts +4 -0
  317. package/src/ruvocal/src/lib/types/ConvSidebar.ts +9 -0
  318. package/src/ruvocal/src/lib/types/Conversation.ts +27 -0
  319. package/src/ruvocal/src/lib/types/ConversationStats.ts +13 -0
  320. package/src/ruvocal/src/lib/types/Message.ts +41 -0
  321. package/src/ruvocal/src/lib/types/MessageEvent.ts +10 -0
  322. package/src/ruvocal/src/lib/types/MessageUpdate.ts +139 -0
  323. package/src/ruvocal/src/lib/types/MigrationResult.ts +7 -0
  324. package/src/ruvocal/src/lib/types/Model.ts +23 -0
  325. package/src/ruvocal/src/lib/types/Report.ts +12 -0
  326. package/src/ruvocal/src/lib/types/Review.ts +6 -0
  327. package/src/ruvocal/src/lib/types/Semaphore.ts +19 -0
  328. package/src/ruvocal/src/lib/types/Session.ts +22 -0
  329. package/src/ruvocal/src/lib/types/Settings.ts +86 -0
  330. package/src/ruvocal/src/lib/types/SharedConversation.ts +9 -0
  331. package/src/ruvocal/src/lib/types/Template.ts +6 -0
  332. package/src/ruvocal/src/lib/types/Timestamps.ts +4 -0
  333. package/src/ruvocal/src/lib/types/TokenCache.ts +6 -0
  334. package/src/ruvocal/src/lib/types/Tool.ts +74 -0
  335. package/src/ruvocal/src/lib/types/UrlDependency.ts +5 -0
  336. package/src/ruvocal/src/lib/types/User.ts +14 -0
  337. package/src/ruvocal/src/lib/utils/PublicConfig.svelte.ts +75 -0
  338. package/src/ruvocal/src/lib/utils/auth.ts +17 -0
  339. package/src/ruvocal/src/lib/utils/chunk.ts +33 -0
  340. package/src/ruvocal/src/lib/utils/cookiesAreEnabled.ts +13 -0
  341. package/src/ruvocal/src/lib/utils/debounce.ts +17 -0
  342. package/src/ruvocal/src/lib/utils/deepestChild.ts +6 -0
  343. package/src/ruvocal/src/lib/utils/favicon.ts +21 -0
  344. package/src/ruvocal/src/lib/utils/fetchJSON.ts +23 -0
  345. package/src/ruvocal/src/lib/utils/file2base64.ts +14 -0
  346. package/src/ruvocal/src/lib/utils/formatUserCount.ts +37 -0
  347. package/src/ruvocal/src/lib/utils/generationState.spec.ts +75 -0
  348. package/src/ruvocal/src/lib/utils/generationState.ts +26 -0
  349. package/src/ruvocal/src/lib/utils/getHref.ts +41 -0
  350. package/src/ruvocal/src/lib/utils/getReturnFromGenerator.ts +7 -0
  351. package/src/ruvocal/src/lib/utils/haptics.ts +64 -0
  352. package/src/ruvocal/src/lib/utils/hashConv.ts +12 -0
  353. package/src/ruvocal/src/lib/utils/hf.ts +17 -0
  354. package/src/ruvocal/src/lib/utils/isDesktop.ts +7 -0
  355. package/src/ruvocal/src/lib/utils/isUrl.ts +8 -0
  356. package/src/ruvocal/src/lib/utils/isVirtualKeyboard.ts +16 -0
  357. package/src/ruvocal/src/lib/utils/loadAttachmentsFromUrls.ts +115 -0
  358. package/src/ruvocal/src/lib/utils/marked.spec.ts +96 -0
  359. package/src/ruvocal/src/lib/utils/marked.ts +531 -0
  360. package/src/ruvocal/src/lib/utils/mcpValidation.ts +147 -0
  361. package/src/ruvocal/src/lib/utils/mergeAsyncGenerators.ts +38 -0
  362. package/src/ruvocal/src/lib/utils/messageUpdates.spec.ts +262 -0
  363. package/src/ruvocal/src/lib/utils/messageUpdates.ts +324 -0
  364. package/src/ruvocal/src/lib/utils/mime.ts +56 -0
  365. package/src/ruvocal/src/lib/utils/models.ts +14 -0
  366. package/src/ruvocal/src/lib/utils/parseBlocks.ts +120 -0
  367. package/src/ruvocal/src/lib/utils/parseIncompleteMarkdown.ts +644 -0
  368. package/src/ruvocal/src/lib/utils/parseStringToList.ts +10 -0
  369. package/src/ruvocal/src/lib/utils/randomUuid.ts +14 -0
  370. package/src/ruvocal/src/lib/utils/searchTokens.ts +33 -0
  371. package/src/ruvocal/src/lib/utils/sha256.ts +7 -0
  372. package/src/ruvocal/src/lib/utils/stringifyError.ts +12 -0
  373. package/src/ruvocal/src/lib/utils/sum.ts +3 -0
  374. package/src/ruvocal/src/lib/utils/template.spec.ts +59 -0
  375. package/src/ruvocal/src/lib/utils/template.ts +53 -0
  376. package/src/ruvocal/src/lib/utils/timeout.ts +9 -0
  377. package/src/ruvocal/src/lib/utils/toolProgress.spec.ts +46 -0
  378. package/src/ruvocal/src/lib/utils/toolProgress.ts +11 -0
  379. package/src/ruvocal/src/lib/utils/tree/addChildren.spec.ts +102 -0
  380. package/src/ruvocal/src/lib/utils/tree/addChildren.ts +48 -0
  381. package/src/ruvocal/src/lib/utils/tree/addSibling.spec.ts +81 -0
  382. package/src/ruvocal/src/lib/utils/tree/addSibling.ts +41 -0
  383. package/src/ruvocal/src/lib/utils/tree/buildSubtree.spec.ts +110 -0
  384. package/src/ruvocal/src/lib/utils/tree/buildSubtree.ts +24 -0
  385. package/src/ruvocal/src/lib/utils/tree/convertLegacyConversation.spec.ts +31 -0
  386. package/src/ruvocal/src/lib/utils/tree/convertLegacyConversation.ts +36 -0
  387. package/src/ruvocal/src/lib/utils/tree/isMessageId.spec.ts +15 -0
  388. package/src/ruvocal/src/lib/utils/tree/isMessageId.ts +5 -0
  389. package/src/ruvocal/src/lib/utils/tree/tree.d.ts +14 -0
  390. package/src/ruvocal/src/lib/utils/tree/treeHelpers.spec.ts +167 -0
  391. package/src/ruvocal/src/lib/utils/updates.ts +39 -0
  392. package/src/ruvocal/src/lib/utils/urlParams.ts +13 -0
  393. package/src/ruvocal/src/lib/workers/autopilotWorker.ts +221 -0
  394. package/src/ruvocal/src/lib/workers/detailFetchWorker.ts +100 -0
  395. package/src/ruvocal/src/lib/workers/markdownWorker.ts +61 -0
  396. package/src/ruvocal/src/routes/+error.svelte +20 -0
  397. package/src/ruvocal/src/routes/+layout.svelte +324 -0
  398. package/src/ruvocal/src/routes/+layout.ts +91 -0
  399. package/src/ruvocal/src/routes/+page.svelte +168 -0
  400. package/src/ruvocal/src/routes/.well-known/oauth-cimd/+server.ts +37 -0
  401. package/src/ruvocal/src/routes/__debug/openai/+server.ts +21 -0
  402. package/src/ruvocal/src/routes/admin/export/+server.ts +159 -0
  403. package/src/ruvocal/src/routes/admin/stats/compute/+server.ts +16 -0
  404. package/src/ruvocal/src/routes/api/conversation/[id]/+server.ts +40 -0
  405. package/src/ruvocal/src/routes/api/conversation/[id]/message/[messageId]/+server.ts +42 -0
  406. package/src/ruvocal/src/routes/api/conversations/+server.ts +48 -0
  407. package/src/ruvocal/src/routes/api/fetch-url/+server.ts +147 -0
  408. package/src/ruvocal/src/routes/api/mcp/health/+server.ts +292 -0
  409. package/src/ruvocal/src/routes/api/mcp/servers/+server.ts +32 -0
  410. package/src/ruvocal/src/routes/api/models/+server.ts +25 -0
  411. package/src/ruvocal/src/routes/api/transcribe/+server.ts +104 -0
  412. package/src/ruvocal/src/routes/api/user/+server.ts +15 -0
  413. package/src/ruvocal/src/routes/api/user/validate-token/+server.ts +20 -0
  414. package/src/ruvocal/src/routes/api/v2/conversations/+server.ts +48 -0
  415. package/src/ruvocal/src/routes/api/v2/conversations/[id]/+server.ts +94 -0
  416. package/src/ruvocal/src/routes/api/v2/conversations/[id]/message/[messageId]/+server.ts +43 -0
  417. package/src/ruvocal/src/routes/api/v2/conversations/import-share/+server.ts +23 -0
  418. package/src/ruvocal/src/routes/api/v2/debug/config/+server.ts +16 -0
  419. package/src/ruvocal/src/routes/api/v2/debug/refresh/+server.ts +30 -0
  420. package/src/ruvocal/src/routes/api/v2/export/+server.ts +196 -0
  421. package/src/ruvocal/src/routes/api/v2/feature-flags/+server.ts +14 -0
  422. package/src/ruvocal/src/routes/api/v2/models/+server.ts +38 -0
  423. package/src/ruvocal/src/routes/api/v2/models/[namespace]/+server.ts +8 -0
  424. package/src/ruvocal/src/routes/api/v2/models/[namespace]/[model]/+server.ts +8 -0
  425. package/src/ruvocal/src/routes/api/v2/models/[namespace]/[model]/subscribe/+server.ts +28 -0
  426. package/src/ruvocal/src/routes/api/v2/models/[namespace]/subscribe/+server.ts +28 -0
  427. package/src/ruvocal/src/routes/api/v2/models/old/+server.ts +7 -0
  428. package/src/ruvocal/src/routes/api/v2/models/refresh/+server.ts +33 -0
  429. package/src/ruvocal/src/routes/api/v2/public-config/+server.ts +7 -0
  430. package/src/ruvocal/src/routes/api/v2/user/+server.ts +17 -0
  431. package/src/ruvocal/src/routes/api/v2/user/billing-orgs/+server.ts +73 -0
  432. package/src/ruvocal/src/routes/api/v2/user/reports/+server.ts +17 -0
  433. package/src/ruvocal/src/routes/api/v2/user/settings/+server.ts +103 -0
  434. package/src/ruvocal/src/routes/conversation/+server.ts +115 -0
  435. package/src/ruvocal/src/routes/conversation/[id]/+page.svelte +582 -0
  436. package/src/ruvocal/src/routes/conversation/[id]/+page.ts +60 -0
  437. package/src/ruvocal/src/routes/conversation/[id]/+server.ts +736 -0
  438. package/src/ruvocal/src/routes/conversation/[id]/message/[messageId]/prompt/+server.ts +66 -0
  439. package/src/ruvocal/src/routes/conversation/[id]/output/[sha256]/+server.ts +58 -0
  440. package/src/ruvocal/src/routes/conversation/[id]/share/+server.ts +69 -0
  441. package/src/ruvocal/src/routes/conversation/[id]/stop-generating/+server.ts +35 -0
  442. package/src/ruvocal/src/routes/healthcheck/+server.ts +3 -0
  443. package/src/ruvocal/src/routes/login/+server.ts +5 -0
  444. package/src/ruvocal/src/routes/login/callback/+server.ts +103 -0
  445. package/src/ruvocal/src/routes/login/callback/updateUser.spec.ts +157 -0
  446. package/src/ruvocal/src/routes/login/callback/updateUser.ts +215 -0
  447. package/src/ruvocal/src/routes/logout/+server.ts +18 -0
  448. package/src/ruvocal/src/routes/metrics/+server.ts +18 -0
  449. package/src/ruvocal/src/routes/models/+page.svelte +233 -0
  450. package/src/ruvocal/src/routes/models/[...model]/+page.svelte +161 -0
  451. package/src/ruvocal/src/routes/models/[...model]/+page.ts +14 -0
  452. package/src/ruvocal/src/routes/models/[...model]/thumbnail.png/+server.ts +64 -0
  453. package/src/ruvocal/src/routes/models/[...model]/thumbnail.png/ModelThumbnail.svelte +28 -0
  454. package/src/ruvocal/src/routes/privacy/+page.svelte +11 -0
  455. package/src/ruvocal/src/routes/r/[id]/+page.ts +34 -0
  456. package/src/ruvocal/src/routes/settings/(nav)/+layout.svelte +282 -0
  457. package/src/ruvocal/src/routes/settings/(nav)/+layout.ts +1 -0
  458. package/src/ruvocal/src/routes/settings/(nav)/+page.svelte +0 -0
  459. package/src/ruvocal/src/routes/settings/(nav)/+server.ts +53 -0
  460. package/src/ruvocal/src/routes/settings/(nav)/[...model]/+page.svelte +464 -0
  461. package/src/ruvocal/src/routes/settings/(nav)/[...model]/+page.ts +14 -0
  462. package/src/ruvocal/src/routes/settings/(nav)/application/+page.svelte +362 -0
  463. package/src/ruvocal/src/routes/settings/+layout.svelte +40 -0
  464. package/src/ruvocal/src/styles/highlight-js.css +195 -0
  465. package/src/ruvocal/src/styles/main.css +144 -0
  466. package/src/ruvocal/static/chatui/apple-touch-icon.png +0 -0
  467. package/src/ruvocal/static/chatui/favicon-dark.svg +3 -0
  468. package/src/ruvocal/static/chatui/favicon-dev.svg +3 -0
  469. package/src/ruvocal/static/chatui/favicon.ico +0 -0
  470. package/src/ruvocal/static/chatui/favicon.svg +3 -0
  471. package/src/ruvocal/static/chatui/icon-128x128.png +0 -0
  472. package/src/ruvocal/static/chatui/icon-144x144.png +0 -0
  473. package/src/ruvocal/static/chatui/icon-192x192.png +0 -0
  474. package/src/ruvocal/static/chatui/icon-256x256.png +0 -0
  475. package/src/ruvocal/static/chatui/icon-36x36.png +0 -0
  476. package/src/ruvocal/static/chatui/icon-48x48.png +0 -0
  477. package/src/ruvocal/static/chatui/icon-512x512.png +0 -0
  478. package/src/ruvocal/static/chatui/icon-72x72.png +0 -0
  479. package/src/ruvocal/static/chatui/icon-96x96.png +0 -0
  480. package/src/ruvocal/static/chatui/icon.svg +3 -0
  481. package/src/ruvocal/static/chatui/logo.svg +7 -0
  482. package/src/ruvocal/static/chatui/manifest.json +54 -0
  483. package/src/ruvocal/static/chatui/omni-welcome.gif +0 -0
  484. package/src/ruvocal/static/chatui/omni-welcome.png +0 -0
  485. package/src/ruvocal/static/chatui/welcome.js +184 -0
  486. package/src/ruvocal/static/chatui/welcome.svg +1 -0
  487. package/src/ruvocal/static/huggingchat/apple-touch-icon.png +0 -0
  488. package/src/ruvocal/static/huggingchat/assistants-thumbnail.png +0 -0
  489. package/src/ruvocal/static/huggingchat/castle-example.jpg +0 -0
  490. package/src/ruvocal/static/huggingchat/favicon-dark.svg +4 -0
  491. package/src/ruvocal/static/huggingchat/favicon-dev.svg +4 -0
  492. package/src/ruvocal/static/huggingchat/favicon.ico +0 -0
  493. package/src/ruvocal/static/huggingchat/favicon.svg +4 -0
  494. package/src/ruvocal/static/huggingchat/fulltext-logo.svg +2 -0
  495. package/src/ruvocal/static/huggingchat/icon-128x128.png +0 -0
  496. package/src/ruvocal/static/huggingchat/icon-144x144.png +0 -0
  497. package/src/ruvocal/static/huggingchat/icon-192x192.png +0 -0
  498. package/src/ruvocal/static/huggingchat/icon-256x256.png +0 -0
  499. package/src/ruvocal/static/huggingchat/icon-36x36.png +0 -0
  500. package/src/ruvocal/static/huggingchat/icon-48x48.png +0 -0
  501. package/src/ruvocal/static/huggingchat/icon-512x512.png +0 -0
  502. package/src/ruvocal/static/huggingchat/icon-72x72.png +0 -0
  503. package/src/ruvocal/static/huggingchat/icon-96x96.png +0 -0
  504. package/src/ruvocal/static/huggingchat/icon.svg +4 -0
  505. package/src/ruvocal/static/huggingchat/logo.svg +4 -0
  506. package/src/ruvocal/static/huggingchat/manifest.json +54 -0
  507. package/src/ruvocal/static/huggingchat/omni-welcome.gif +0 -0
  508. package/src/ruvocal/static/huggingchat/routes.chat.json +226 -0
  509. package/src/ruvocal/static/huggingchat/thumbnail.png +0 -0
  510. package/src/ruvocal/static/huggingchat/tools-thumbnail.png +0 -0
  511. package/src/ruvocal/static/robots.txt +10 -0
  512. package/src/ruvocal/stub/@reflink/reflink/index.js +0 -0
  513. package/src/ruvocal/stub/@reflink/reflink/package.json +5 -0
  514. package/src/ruvocal/svelte.config.js +53 -0
  515. package/src/ruvocal/tailwind.config.cjs +30 -0
  516. package/src/ruvocal/tsconfig.json +19 -0
  517. package/src/ruvocal/vite.config.ts +87 -0
  518. package/src/scripts/deploy.sh +116 -0
  519. package/src/scripts/generate-config.js +245 -0
  520. package/src/scripts/generate-welcome.js +187 -0
  521. package/src/scripts/package-rvf.sh +116 -0
@@ -0,0 +1,185 @@
1
+ <script lang="ts">
2
+ import { onMount } from "svelte";
3
+
4
+ interface Props {
5
+ width?: number;
6
+ height?: number;
7
+ }
8
+
9
+ let { width = 420, height = 192 }: Props = $props();
10
+ let canvas: HTMLCanvasElement;
11
+
12
+ onMount(() => {
13
+ const ctx = canvas.getContext("2d");
14
+ if (!ctx) return;
15
+
16
+ const dpr = Math.min(window.devicePixelRatio, 2);
17
+ canvas.width = width * dpr;
18
+ canvas.height = height * dpr;
19
+ ctx.scale(dpr, dpr);
20
+
21
+ // Nodes — tightly clustered around center where RuFlo text sits
22
+ const nodes = [
23
+ { x: 0.50, y: 0.50, r: 4, color: "#3b82f6" },
24
+ { x: 0.35, y: 0.42, r: 2.5, color: "#06b6d4" },
25
+ { x: 0.65, y: 0.42, r: 2.5, color: "#818cf8" },
26
+ { x: 0.38, y: 0.58, r: 2.5, color: "#2dd4bf" },
27
+ { x: 0.62, y: 0.58, r: 2.5, color: "#a78bfa" },
28
+ { x: 0.26, y: 0.50, r: 2, color: "#38bdf8" },
29
+ { x: 0.74, y: 0.50, r: 2, color: "#c084fc" },
30
+ { x: 0.44, y: 0.35, r: 2, color: "#22d3ee" },
31
+ { x: 0.56, y: 0.65, r: 2, color: "#6366f1" },
32
+ { x: 0.30, y: 0.62, r: 1.8, color: "#0ea5e9" },
33
+ { x: 0.70, y: 0.62, r: 1.8, color: "#8b5cf6" },
34
+ { x: 0.42, y: 0.70, r: 1.8, color: "#14b8a6" },
35
+ { x: 0.58, y: 0.35, r: 1.8, color: "#60a5fa" },
36
+ { x: 0.32, y: 0.34, r: 1.8, color: "#34d399" },
37
+ { x: 0.68, y: 0.34, r: 1.8, color: "#a855f7" },
38
+ ];
39
+
40
+ const edges: [number, number][] = [
41
+ [0, 1], [0, 2], [0, 3], [0, 4],
42
+ [1, 5], [1, 7], [1, 13],
43
+ [2, 6], [2, 12], [2, 14],
44
+ [3, 5], [3, 8], [3, 9],
45
+ [4, 6], [4, 10], [4, 14],
46
+ [5, 9], [5, 13],
47
+ [6, 10], [6, 14],
48
+ [7, 12], [7, 13],
49
+ [8, 9], [8, 11],
50
+ [10, 14], [11, 9],
51
+ ];
52
+
53
+ // Stars
54
+ const stars = Array.from({ length: 120 }, () => ({
55
+ x: Math.random() * width,
56
+ y: Math.random() * height,
57
+ r: Math.random() * 1.2 + 0.3,
58
+ phase: Math.random() * Math.PI * 2,
59
+ }));
60
+
61
+ // Particles traveling along edges
62
+ const particles = edges.map((_, i) => ({
63
+ edge: i,
64
+ t: Math.random(),
65
+ speed: 0.002 + Math.random() * 0.003,
66
+ }));
67
+
68
+ let animId: number;
69
+ let time = 0;
70
+
71
+ function draw() {
72
+ animId = requestAnimationFrame(draw);
73
+ time += 0.008;
74
+
75
+ ctx.fillStyle = "#06060f";
76
+ ctx.fillRect(0, 0, width, height);
77
+
78
+ // Stars
79
+ for (const s of stars) {
80
+ const alpha = 0.3 + Math.sin(time * 1.5 + s.phase) * 0.2;
81
+ ctx.beginPath();
82
+ ctx.arc(s.x, s.y, s.r, 0, Math.PI * 2);
83
+ ctx.fillStyle = `rgba(255,255,255,${alpha})`;
84
+ ctx.fill();
85
+ }
86
+
87
+ // Compute animated node positions
88
+ const px = nodes.map((n, i) => n.x * width + Math.sin(time + i) * 3);
89
+ const py = nodes.map((n, i) => n.y * height + Math.cos(time * 0.8 + i * 1.3) * 2);
90
+
91
+ // Edges
92
+ for (let i = 0; i < edges.length; i++) {
93
+ const [a, b] = edges[i];
94
+ const alpha = 0.12 + Math.sin(time * 1.5 + i) * 0.06;
95
+ ctx.beginPath();
96
+ ctx.moveTo(px[a], py[a]);
97
+ ctx.lineTo(px[b], py[b]);
98
+ ctx.strokeStyle = `rgba(59,130,246,${alpha})`;
99
+ ctx.lineWidth = 1;
100
+ ctx.stroke();
101
+ }
102
+
103
+ // Particles along edges
104
+ for (const p of particles) {
105
+ p.t = (p.t + p.speed) % 1;
106
+ const [a, b] = edges[p.edge];
107
+ const x = px[a] + (px[b] - px[a]) * p.t;
108
+ const y = py[a] + (py[b] - py[a]) * p.t;
109
+ ctx.beginPath();
110
+ ctx.arc(x, y, 1.2, 0, Math.PI * 2);
111
+ ctx.fillStyle = "rgba(96,165,250,0.7)";
112
+ ctx.fill();
113
+ }
114
+
115
+ // Orbital rings (ellipses) — centered with nodes and text
116
+ ctx.save();
117
+ ctx.translate(width * 0.5, height * 0.5);
118
+ ctx.rotate(time * 0.2);
119
+ ctx.beginPath();
120
+ ctx.ellipse(0, 0, 45, 20, 0, 0, Math.PI * 2);
121
+ ctx.strokeStyle = "rgba(59,130,246,0.1)";
122
+ ctx.lineWidth = 1.5;
123
+ ctx.stroke();
124
+ ctx.restore();
125
+
126
+ ctx.save();
127
+ ctx.translate(width * 0.5, height * 0.5);
128
+ ctx.rotate(-time * 0.15 + 0.5);
129
+ ctx.beginPath();
130
+ ctx.ellipse(0, 0, 55, 25, 0.4, 0, Math.PI * 2);
131
+ ctx.strokeStyle = "rgba(129,140,248,0.07)";
132
+ ctx.lineWidth = 1.5;
133
+ ctx.stroke();
134
+ ctx.restore();
135
+
136
+ // Nodes with glow
137
+ for (let i = 0; i < nodes.length; i++) {
138
+ const n = nodes[i];
139
+ const pulse = 1 + Math.sin(time * 2 + i * 0.7) * 0.15;
140
+ const r = n.r * pulse;
141
+
142
+ // Glow
143
+ const grad = ctx.createRadialGradient(px[i], py[i], 0, px[i], py[i], r * 5);
144
+ grad.addColorStop(0, n.color + "30");
145
+ grad.addColorStop(1, n.color + "00");
146
+ ctx.beginPath();
147
+ ctx.arc(px[i], py[i], r * 5, 0, Math.PI * 2);
148
+ ctx.fillStyle = grad;
149
+ ctx.fill();
150
+
151
+ // Core
152
+ ctx.beginPath();
153
+ ctx.arc(px[i], py[i], r, 0, Math.PI * 2);
154
+ ctx.fillStyle = n.color;
155
+ ctx.fill();
156
+ }
157
+ }
158
+
159
+ draw();
160
+
161
+ return () => {
162
+ cancelAnimationFrame(animId);
163
+ };
164
+ });
165
+ </script>
166
+
167
+ <div
168
+ class="relative h-full w-full overflow-hidden"
169
+ style="background: #06060f;"
170
+ >
171
+ <canvas
172
+ bind:this={canvas}
173
+ style="width: {width}px; height: {height}px;"
174
+ ></canvas>
175
+ <!-- Overlay text -->
176
+ <div class="pointer-events-none absolute inset-0 flex flex-col items-center justify-center">
177
+ <h2
178
+ class="text-3xl font-light tracking-[0.3em] text-indigo-100"
179
+ style="text-shadow: 0 0 20px rgba(99, 102, 241, 0.5), 0 0 40px rgba(99, 102, 241, 0.2);"
180
+ >
181
+ RuFlo
182
+ </h2>
183
+ <p class="mt-1 text-[10px] tracking-[0.25em] text-slate-400/70">INTELLIGENT WORKFLOWS</p>
184
+ </div>
185
+ </div>
@@ -0,0 +1,47 @@
1
+ <script lang="ts">
2
+ import { fade } from "svelte/transition";
3
+ import IconChevron from "./icons/IconChevron.svelte";
4
+
5
+ interface Props {
6
+ scrollNode: HTMLElement;
7
+ class?: string;
8
+ }
9
+
10
+ let { scrollNode, class: className = "" }: Props = $props();
11
+
12
+ let visible = $state(false);
13
+ let observer: ResizeObserver | null = $state(null);
14
+
15
+ function updateVisibility() {
16
+ if (!scrollNode) return;
17
+ visible =
18
+ Math.ceil(scrollNode.scrollTop) + 200 < scrollNode.scrollHeight - scrollNode.clientHeight;
19
+ }
20
+
21
+ function destroy() {
22
+ observer?.disconnect();
23
+ scrollNode?.removeEventListener("scroll", updateVisibility);
24
+ }
25
+ const cleanup = $effect.root(() => {
26
+ $effect(() => {
27
+ if (scrollNode) {
28
+ if (window.ResizeObserver) {
29
+ observer = new ResizeObserver(() => updateVisibility());
30
+ observer.observe(scrollNode);
31
+ cleanup();
32
+ }
33
+ scrollNode?.addEventListener("scroll", updateVisibility);
34
+ }
35
+ });
36
+ return () => destroy();
37
+ });
38
+ </script>
39
+
40
+ {#if visible}
41
+ <button
42
+ transition:fade={{ duration: 150 }}
43
+ onclick={() => scrollNode.scrollTo({ top: scrollNode.scrollHeight, behavior: "smooth" })}
44
+ class="btn absolute flex h-[41px] w-[41px] rounded-full border bg-white shadow-md transition-all hover:bg-gray-100 dark:border-gray-600 dark:bg-gray-700 dark:shadow-gray-950 dark:hover:bg-gray-600 {className}"
45
+ ><IconChevron classNames="mt-[2px]" /></button
46
+ >
47
+ {/if}
@@ -0,0 +1,77 @@
1
+ <script lang="ts">
2
+ import { fade } from "svelte/transition";
3
+ import { onDestroy, untrack } from "svelte";
4
+ import IconChevron from "./icons/IconChevron.svelte";
5
+
6
+ let visible = $state(false);
7
+ interface Props {
8
+ scrollNode: HTMLElement;
9
+ class?: string;
10
+ }
11
+
12
+ let { scrollNode, class: className = "" }: Props = $props();
13
+ let observer: ResizeObserver | null = $state(null);
14
+
15
+ function updateVisibility() {
16
+ if (!scrollNode) return;
17
+ visible =
18
+ Math.ceil(scrollNode.scrollTop) + 200 < scrollNode.scrollHeight - scrollNode.clientHeight &&
19
+ scrollNode.scrollTop > 200;
20
+ }
21
+
22
+ function scrollToPrevious() {
23
+ if (!scrollNode) return;
24
+ const messages = scrollNode.querySelectorAll("[data-message-id]");
25
+ const scrollTop = scrollNode.scrollTop;
26
+ let previousMessage: Element | null = null;
27
+
28
+ for (let i = messages.length - 1; i >= 0; i--) {
29
+ const messageTop =
30
+ messages[i].getBoundingClientRect().top +
31
+ scrollTop -
32
+ scrollNode.getBoundingClientRect().top;
33
+ if (messageTop < scrollTop - 1) {
34
+ previousMessage = messages[i];
35
+ break;
36
+ }
37
+ }
38
+
39
+ if (previousMessage) {
40
+ previousMessage.scrollIntoView({ behavior: "smooth", block: "start" });
41
+ }
42
+ }
43
+
44
+ function destroy() {
45
+ observer?.disconnect();
46
+ scrollNode?.removeEventListener("scroll", updateVisibility);
47
+ }
48
+
49
+ onDestroy(destroy);
50
+
51
+ $effect(() => {
52
+ scrollNode &&
53
+ untrack(() => {
54
+ if (scrollNode) {
55
+ destroy();
56
+
57
+ if (window.ResizeObserver) {
58
+ observer = new ResizeObserver(() => {
59
+ updateVisibility();
60
+ });
61
+ observer.observe(scrollNode);
62
+ }
63
+ scrollNode.addEventListener("scroll", updateVisibility);
64
+ }
65
+ });
66
+ });
67
+ </script>
68
+
69
+ {#if visible}
70
+ <button
71
+ transition:fade={{ duration: 150 }}
72
+ onclick={scrollToPrevious}
73
+ class="btn absolute flex h-[41px] w-[41px] rounded-full border bg-white shadow-md transition-all hover:bg-gray-100 dark:border-gray-600 dark:bg-gray-700 dark:shadow-gray-950 dark:hover:bg-gray-600 {className}"
74
+ >
75
+ <IconChevron classNames="rotate-180 mt-[2px]" />
76
+ </button>
77
+ {/if}
@@ -0,0 +1,182 @@
1
+ <script lang="ts">
2
+ import Modal from "$lib/components/Modal.svelte";
3
+ import { base } from "$app/paths";
4
+ import { page } from "$app/state";
5
+ import CarbonLink from "~icons/carbon/link";
6
+ import CarbonCheckmark from "~icons/carbon/checkmark";
7
+ import EosIconsLoading from "~icons/eos-icons/loading";
8
+ import CopyToClipBoardBtn from "$lib/components/CopyToClipBoardBtn.svelte";
9
+ import { onMount } from "svelte";
10
+ import { createShareLink } from "$lib/createShareLink";
11
+
12
+ interface Props {
13
+ open?: boolean;
14
+ onclose?: () => void;
15
+ oncopied?: () => void;
16
+ }
17
+
18
+ let { open = false, onclose, oncopied }: Props = $props();
19
+
20
+ let creating = $state(false);
21
+ let createdUrl: string | null = $state(null);
22
+ let errorMsg: string | null = $state(null);
23
+ let justCopied = $state(false);
24
+
25
+ async function handleCreate() {
26
+ try {
27
+ creating = true;
28
+ errorMsg = null;
29
+ createdUrl = await createShareLink(page.params.id ?? "");
30
+ } catch (e) {
31
+ errorMsg = (e as Error).message || "Could not create link";
32
+ } finally {
33
+ creating = false;
34
+ }
35
+ }
36
+
37
+ function close() {
38
+ open = false;
39
+ onclose?.();
40
+ }
41
+
42
+ // If the current page is already a shared chat (7-char id), pre-fill the link
43
+ onMount(async () => {
44
+ if (page.params.id && page.params.id.length === 7) {
45
+ try {
46
+ createdUrl = await createShareLink(page.params.id);
47
+ } catch (e) {
48
+ // ignore
49
+ }
50
+ }
51
+ });
52
+
53
+ function withLeafId(url: string | null): string | null {
54
+ if (!url) return url;
55
+ try {
56
+ const leafId = localStorage.getItem("leafId");
57
+ if (!leafId) return url;
58
+ const u = new URL(url);
59
+ u.searchParams.set("leafId", leafId);
60
+ return u.toString();
61
+ } catch (e) {
62
+ return url;
63
+ }
64
+ }
65
+ </script>
66
+
67
+ {#if open}
68
+ <Modal onclose={close} width="w-[90dvh] md:w-[500px]">
69
+ <div class="flex w-full flex-col gap-3 p-5 sm:gap-5 sm:p-6">
70
+ <!-- Header + copy -->
71
+ {#if createdUrl}
72
+ <div class="flex items-start justify-between">
73
+ <div class="text-xl font-semibold text-gray-800 dark:text-gray-200">
74
+ Public link created
75
+ </div>
76
+ <button type="button" class="group" onclick={close} aria-label="Close">
77
+ <svg
78
+ xmlns="http://www.w3.org/2000/svg"
79
+ viewBox="0 0 32 32"
80
+ class="size-5 text-gray-700 group-hover:text-gray-500 dark:text-gray-300 dark:group-hover:text-gray-400"
81
+ >
82
+ <path
83
+ d="M24 9.41 22.59 8 16 14.59 9.41 8 8 9.41 14.59 16 8 22.59 9.41 24 16 17.41 22.59 24 24 22.59 17.41 16 24 9.41z"
84
+ fill="currentColor"
85
+ />
86
+ </svg>
87
+ </button>
88
+ </div>
89
+ <div class="text-sm text-gray-600 dark:text-gray-400">
90
+ A public link to your chat has been created.
91
+ </div>
92
+ {:else}
93
+ <div class="flex items-start justify-between">
94
+ <div class="text-xl font-semibold text-gray-800 dark:text-gray-200">
95
+ Share public link to chat
96
+ </div>
97
+ <button type="button" class="group" onclick={close} aria-label="Close">
98
+ <svg
99
+ xmlns="http://www.w3.org/2000/svg"
100
+ viewBox="0 0 32 32"
101
+ class="size-5 text-gray-700 group-hover:text-gray-500 dark:text-gray-300 dark:group-hover:text-gray-400"
102
+ >
103
+ <path
104
+ d="M24 9.41 22.59 8 16 14.59 9.41 8 8 9.41 14.59 16 8 22.59 9.41 24 16 17.41 22.59 24 24 22.59 17.41 16 24 9.41z"
105
+ fill="currentColor"
106
+ />
107
+ </svg>
108
+ </button>
109
+ </div>
110
+ <div class="text-sm text-gray-600 dark:text-gray-400">
111
+ Any messages you add after sharing stay private.
112
+ </div>
113
+ {/if}
114
+
115
+ {#if errorMsg}
116
+ <div
117
+ class="rounded-lg border border-red-200 bg-red-50 px-3 py-2 text-sm text-red-700 dark:border-red-700 dark:bg-red-500/10 dark:text-red-300"
118
+ >
119
+ {errorMsg}
120
+ </div>
121
+ {/if}
122
+
123
+ <!-- URL row -->
124
+ <div
125
+ class="flex h-12 items-center gap-2 whitespace-nowrap rounded-2xl border border-gray-200 bg-gray-50 p-2.5 dark:border-gray-700 dark:bg-gray-800"
126
+ >
127
+ <input
128
+ class="w-full truncate bg-transparent text-[15px] text-gray-700 outline-none placeholder:text-gray-400 dark:text-gray-200 dark:placeholder:text-gray-500 max-sm:text-sm"
129
+ readonly
130
+ value={createdUrl ??
131
+ `${page.data.publicConfig.PUBLIC_SHARE_PREFIX || `${page.data.publicConfig.PUBLIC_ORIGIN || page.url.origin}${base}`}/r/...`}
132
+ />
133
+
134
+ {#if createdUrl}
135
+ <CopyToClipBoardBtn
136
+ classNames="inline-flex items-center rounded-xl -mr-1 border border-gray-300 bg-white px-3 py-1.5 text-sm font-medium text-gray-900 shadow enabled:hover:bg-gray-50 dark:border-gray-700 dark:bg-gray-700 dark:text-gray-100 dark:enabled:hover:bg-gray-600"
137
+ showTooltip={false}
138
+ value={withLeafId(createdUrl) ?? createdUrl}
139
+ onClick={() => {
140
+ justCopied = true;
141
+ oncopied?.();
142
+ setTimeout(() => (justCopied = false), 1200);
143
+ }}
144
+ >
145
+ {#snippet children()}
146
+ <span class="inline-flex items-center gap-1.5">
147
+ {#if justCopied}
148
+ <CarbonCheckmark class="text-[.95em] text-green-600 dark:text-green-400" />
149
+ Copied
150
+ {:else}
151
+ <!-- Use the copy icon provided by CopyToClipBoardBtn default otherwise -->
152
+ <svg width="1em" height="1em" viewBox="0 0 32 32" class="text-[.95em]"
153
+ ><path
154
+ fill="currentColor"
155
+ d="M28 10v18H10V10zm-2 2H12v14h14zm-4-8v2H6v14H4V4z"
156
+ /></svg
157
+ >
158
+ Copy link
159
+ {/if}
160
+ </span>
161
+ {/snippet}
162
+ </CopyToClipBoardBtn>
163
+ {:else}
164
+ <button
165
+ class="-mr-1 inline-flex items-center gap-2 rounded-xl border border-gray-300 bg-white px-3 py-1.5 text-sm font-medium text-gray-900 shadow hover:bg-gray-50 disabled:opacity-50 dark:border-gray-700 dark:bg-gray-700 dark:text-gray-100 dark:hover:bg-gray-600"
166
+ type="button"
167
+ disabled={creating}
168
+ onclick={handleCreate}
169
+ >
170
+ {#if creating}
171
+ <EosIconsLoading class="text-[1.05em]" />
172
+ Creating…
173
+ {:else}
174
+ <CarbonLink class="text-[1.05em]" />
175
+ Create link
176
+ {/if}
177
+ </button>
178
+ {/if}
179
+ </div>
180
+ </div>
181
+ </Modal>
182
+ {/if}
@@ -0,0 +1,69 @@
1
+ <script lang="ts">
2
+ import CarbonStopFilledAlt from "~icons/carbon/stop-filled-alt";
3
+
4
+ interface Props {
5
+ classNames?: string;
6
+ onClick?: () => void;
7
+ showBorder?: boolean;
8
+ }
9
+
10
+ let { classNames = "", onClick, showBorder = false }: Props = $props();
11
+ </script>
12
+
13
+ <button
14
+ type="button"
15
+ onclick={onClick}
16
+ class={`btn stop-generating-btn ${showBorder ? "stop-generating-btn--spinning" : ""} ${classNames}`}
17
+ aria-label="Stop generating"
18
+ >
19
+ <span class="sr-only">Stop generating</span>
20
+ <CarbonStopFilledAlt class="size-3.5 text-gray-500" />
21
+ </button>
22
+
23
+ <style lang="postcss">
24
+ .stop-generating-btn {
25
+ position: relative;
26
+ display: inline-flex;
27
+ align-items: center;
28
+ justify-content: center;
29
+ border-radius: 9999px;
30
+ --stop-generating-ring-color: rgba(31, 41, 55, 0.35);
31
+ }
32
+
33
+ .stop-generating-btn :global(svg) {
34
+ display: block;
35
+ }
36
+
37
+ .stop-generating-btn::after {
38
+ content: "";
39
+ position: absolute;
40
+ inset: -2px;
41
+ border-radius: inherit;
42
+ pointer-events: none;
43
+ background: transparent;
44
+ }
45
+
46
+ .stop-generating-btn--spinning::after {
47
+ background: conic-gradient(
48
+ from 0deg,
49
+ transparent 0deg 240deg,
50
+ var(--stop-generating-ring-color) 240deg 360deg
51
+ );
52
+ mask: radial-gradient(farthest-side, transparent calc(100% - 2px), #000 calc(100% - 1px));
53
+ animation: stop-generating-rotate 1.2s linear infinite;
54
+ }
55
+
56
+ :global(.dark) .stop-generating-btn {
57
+ --stop-generating-ring-color: rgba(255, 255, 255, 0.2);
58
+ }
59
+
60
+ @keyframes stop-generating-rotate {
61
+ from {
62
+ transform: rotate(0deg);
63
+ }
64
+
65
+ to {
66
+ transform: rotate(360deg);
67
+ }
68
+ }
69
+ </style>
@@ -0,0 +1,87 @@
1
+ <script lang="ts">
2
+ import Modal from "$lib/components/Modal.svelte";
3
+ import { isPro } from "$lib/stores/isPro";
4
+ import IconPro from "$lib/components/icons/IconPro.svelte";
5
+ import IconDazzled from "$lib/components/icons/IconDazzled.svelte";
6
+
7
+ interface Props {
8
+ close: () => void;
9
+ }
10
+
11
+ let { close }: Props = $props();
12
+ </script>
13
+
14
+ <Modal closeOnBackdrop={false} onclose={close} width="!max-w-[420px] !m-4">
15
+ <div
16
+ class="flex w-full flex-col gap-8 bg-white bg-gradient-to-b to-transparent px-6 pb-7 dark:bg-black dark:from-white/10 dark:to-white/5"
17
+ >
18
+ <div
19
+ class="-mx-6 grid h-48 select-none place-items-center bg-gradient-to-t from-black/5 dark:from-white/10"
20
+ >
21
+ <div class="flex flex-col items-center justify-center gap-2.5 px-8 text-center">
22
+ <div
23
+ class="flex size-14 items-center justify-center rounded-full text-3xl {$isPro
24
+ ? 'bg-gradient-to-br from-yellow-500/15 via-orange-500/15 to-red-500/15'
25
+ : 'bg-gradient-to-br from-pink-500/15 from-15% via-green-500/15 to-yellow-500/15'}"
26
+ >
27
+ {#if $isPro}
28
+ <IconDazzled />
29
+ {:else}
30
+ <IconPro classNames="!mr-0" />
31
+ {/if}
32
+ </div>
33
+ <h2 class="text-2xl font-semibold text-gray-900 dark:text-gray-100">
34
+ {$isPro ? "Out of Credits" : "Upgrade Required"}
35
+ </h2>
36
+ </div>
37
+ </div>
38
+
39
+ <div class="text-gray-700 dark:text-gray-200">
40
+ {#if $isPro}
41
+ <p class="text-[15px] leading-relaxed">
42
+ You've used all your available credits. Purchase additional credits to continue using
43
+ HuggingChat.
44
+ </p>
45
+ <p class="mt-3 text-[15px] italic leading-relaxed opacity-75">
46
+ Your credits can be used in other HF services and external apps via Inference Providers.
47
+ </p>
48
+ {:else}
49
+ <p class="text-[15px] leading-relaxed">
50
+ You've reached your message limit. Upgrade to Hugging Face PRO to continue using
51
+ HuggingChat.
52
+ </p>
53
+ <p class="mt-3 text-[15px] italic leading-relaxed opacity-75">
54
+ It's also possible to use your PRO credits in your favorite AI tools.
55
+ </p>
56
+ {/if}
57
+ </div>
58
+
59
+ <div class="flex flex-col gap-2.5">
60
+ {#if $isPro}
61
+ <a
62
+ href="https://huggingface.co/settings/billing?add-credits=true"
63
+ target="_blank"
64
+ rel="noopener noreferrer"
65
+ class="w-full rounded-xl bg-black px-5 py-2.5 text-center text-base font-medium text-white hover:bg-gray-800 dark:bg-white dark:text-black dark:hover:bg-gray-200"
66
+ >
67
+ Purchase Credits
68
+ </a>
69
+ {:else}
70
+ <a
71
+ href="https://huggingface.co/subscribe/pro?from=HuggingChat"
72
+ target="_blank"
73
+ rel="noopener noreferrer"
74
+ class="w-full rounded-xl bg-black px-5 py-2.5 text-center text-base font-medium text-white hover:bg-gray-800 dark:bg-white dark:text-black dark:hover:bg-gray-200"
75
+ >
76
+ Upgrade to Pro
77
+ </a>
78
+ {/if}
79
+ <button
80
+ class="w-full rounded-xl bg-gray-200 px-5 py-2.5 text-base font-medium text-gray-700 hover:bg-gray-300/80 dark:bg-white/5 dark:text-gray-200 dark:hover:bg-white/10"
81
+ onclick={close}
82
+ >
83
+ Maybe later
84
+ </button>
85
+ </div>
86
+ </div>
87
+ </Modal>
@@ -0,0 +1,36 @@
1
+ <script lang="ts">
2
+ import { tap } from "$lib/utils/haptics";
3
+
4
+ interface Props {
5
+ checked: boolean;
6
+ name: string;
7
+ }
8
+
9
+ let { checked = $bindable(), name }: Props = $props();
10
+
11
+ function toggle() {
12
+ checked = !checked;
13
+ tap();
14
+ }
15
+
16
+ function onKeydown(e: KeyboardEvent) {
17
+ if (e.key === " " || e.key === "Enter") {
18
+ e.preventDefault();
19
+ toggle();
20
+ }
21
+ }
22
+ </script>
23
+
24
+ <input bind:checked type="checkbox" {name} class="peer pointer-events-none absolute opacity-0" />
25
+ <div
26
+ aria-checked={checked}
27
+ aria-roledescription="switch"
28
+ aria-label="switch"
29
+ role="switch"
30
+ tabindex="0"
31
+ onclick={toggle}
32
+ onkeydown={onKeydown}
33
+ class="relative inline-flex h-5 w-9 shrink-0 cursor-pointer items-center rounded-full bg-gray-300 p-1 shadow-inner ring-gray-400 peer-checked:bg-blue-600 hover:bg-gray-400 peer-checked:hover:bg-blue-600 focus-visible:ring focus-visible:ring-offset-1 dark:bg-gray-600 dark:ring-gray-700 dark:hover:bg-gray-500 dark:peer-checked:hover:bg-blue-600 peer-checked:[&>div]:translate-x-3.5"
34
+ >
35
+ <div class="h-3.5 w-3.5 rounded-full bg-white shadow-sm transition-transform"></div>
36
+ </div>