@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,83 @@
1
+ import { config } from "$lib/server/config";
2
+ import { generateFromDefaultEndpoint } from "$lib/server/generateFromDefaultEndpoint";
3
+ import { logger } from "$lib/server/logger";
4
+ import { MessageUpdateType, type MessageUpdate } from "$lib/types/MessageUpdate";
5
+ import type { Conversation } from "$lib/types/Conversation";
6
+ import { getReturnFromGenerator } from "$lib/utils/getReturnFromGenerator";
7
+
8
+ export async function* generateTitleForConversation(
9
+ conv: Conversation,
10
+ locals: App.Locals | undefined
11
+ ): AsyncGenerator<MessageUpdate, undefined, undefined> {
12
+ try {
13
+ const userMessage = conv.messages.find((m) => m.from === "user");
14
+ // HACK: detect if the conversation is new
15
+ if (conv.title !== "New Chat" || !userMessage) return;
16
+
17
+ const prompt = userMessage.content;
18
+ const modelForTitle = config.TASK_MODEL?.trim() ? config.TASK_MODEL : conv.model;
19
+ const title = (await generateTitle(prompt, modelForTitle, locals)) ?? "New Chat";
20
+
21
+ yield {
22
+ type: MessageUpdateType.Title,
23
+ title,
24
+ };
25
+ } catch (cause) {
26
+ logger.error(cause, "Failed while generating title for conversation");
27
+ }
28
+ }
29
+
30
+ async function generateTitle(
31
+ prompt: string,
32
+ modelId: string | undefined,
33
+ locals: App.Locals | undefined
34
+ ) {
35
+ if (config.LLM_SUMMARIZATION !== "true") {
36
+ // When summarization is disabled, use the first five words without adding emojis
37
+ return prompt.split(/\s+/g).slice(0, 5).join(" ");
38
+ }
39
+
40
+ // Tools removed: no tool-based title path
41
+
42
+ return await getReturnFromGenerator(
43
+ generateFromDefaultEndpoint({
44
+ messages: [{ from: "user", content: `User message: "${prompt}"` }],
45
+ preprompt: `You are a chat thread titling assistant.
46
+ Goal: Produce a very short, descriptive title (2–4 words) that names the topic of the user's first message.
47
+
48
+ Rules:
49
+ - Output ONLY the title text. No prefixes, labels, quotes, emojis, hashtags, or trailing punctuation.
50
+ - Use the user's language.
51
+ - Write a noun phrase that names the topic. Do not write instructions.
52
+ - Never output just a pronoun (me/you/I/we/us/myself/yourself). Prefer a neutral subject (e.g., "Assistant", "model", or the concrete topic).
53
+ - Never include meta-words: Summarize, Summary, Title, Prompt, Topic, Subject, About, Question, Request, Chat.
54
+
55
+ Examples:
56
+ User: "Summarize hello" -> Hello
57
+ User: "How do I reverse a string in Python?" -> Python string reversal
58
+ User: "help me plan a NYC weekend" -> NYC weekend plan
59
+ User: "请解释Transformer是如何工作的" -> Transformer 工作原理
60
+ User: "tell me more about you" -> About the assistant
61
+ Return only the title text.`,
62
+ generateSettings: {
63
+ max_tokens: 24,
64
+ temperature: 0,
65
+ },
66
+ modelId,
67
+ locals,
68
+ })
69
+ )
70
+ .then((summary) => {
71
+ const firstFive = prompt.split(/\s+/g).slice(0, 5).join(" ");
72
+ const trimmed = String(summary ?? "").trim();
73
+ // Fallback: if empty, return first five words only (no emoji)
74
+ return trimmed || firstFive;
75
+ })
76
+ .catch((e) => {
77
+ logger.error(e, "Error generating title");
78
+ const firstFive = prompt.split(/\s+/g).slice(0, 5).join(" ");
79
+ return firstFive;
80
+ });
81
+ }
82
+
83
+ // No post-processing: rely solely on prompt instructions above
@@ -0,0 +1,28 @@
1
+ import type { ProcessedModel } from "../models";
2
+ import type { Endpoint } from "../endpoints/endpoints";
3
+ import type { Conversation } from "$lib/types/Conversation";
4
+ import type { Message } from "$lib/types/Message";
5
+ import type { Assistant } from "$lib/types/Assistant";
6
+
7
+ export interface TextGenerationContext {
8
+ model: ProcessedModel;
9
+ endpoint: Endpoint;
10
+ conv: Conversation;
11
+ messages: Message[];
12
+ assistant?: Pick<Assistant, "dynamicPrompt" | "generateSettings">;
13
+ promptedAt: Date;
14
+ ip: string;
15
+ username?: string;
16
+ /** Force-enable multimodal handling for endpoints that support it */
17
+ forceMultimodal?: boolean;
18
+ /** Force-enable tool calling even if model does not advertise support */
19
+ forceTools?: boolean;
20
+ /** Inference provider preference: "auto", "fastest", "cheapest", or a specific provider name */
21
+ provider?: string;
22
+ locals: App.Locals | undefined;
23
+ abortController: AbortController;
24
+ /** Autopilot mode — auto-continue tool calls up to autopilotMaxSteps iterations */
25
+ autopilot?: boolean;
26
+ /** User-configurable cap on autopilot tool-call loops; server falls back to 30 if undefined */
27
+ autopilotMaxSteps?: number;
28
+ }
@@ -0,0 +1,88 @@
1
+ import type { MessageFile } from "$lib/types/Message";
2
+ import type { EndpointMessage } from "$lib/server/endpoints/endpoints";
3
+ import type { OpenAI } from "openai";
4
+ import { TEXT_MIME_ALLOWLIST } from "$lib/constants/mime";
5
+ import type { makeImageProcessor } from "$lib/server/endpoints/images";
6
+
7
+ /**
8
+ * Prepare chat messages for OpenAI-compatible multimodal payloads.
9
+ * - Processes images via the provided imageProcessor (resize/convert) when multimodal is enabled.
10
+ * - Injects text-file content into the user message text.
11
+ * - Leaves messages untouched when no files or multimodal disabled.
12
+ */
13
+ export async function prepareMessagesWithFiles(
14
+ messages: EndpointMessage[],
15
+ imageProcessor: ReturnType<typeof makeImageProcessor>,
16
+ isMultimodal: boolean
17
+ ): Promise<OpenAI.Chat.Completions.ChatCompletionMessageParam[]> {
18
+ return Promise.all(
19
+ messages.map(async (message) => {
20
+ if (message.from === "user" && message.files && message.files.length > 0) {
21
+ const { imageParts, textContent } = await prepareFiles(
22
+ imageProcessor,
23
+ message.files,
24
+ isMultimodal
25
+ );
26
+
27
+ let messageText = message.content;
28
+ if (textContent.length > 0) {
29
+ messageText = textContent + "\n\n" + message.content;
30
+ }
31
+
32
+ if (imageParts.length > 0 && isMultimodal) {
33
+ const parts = [{ type: "text" as const, text: messageText }, ...imageParts];
34
+ return { role: message.from, content: parts };
35
+ }
36
+
37
+ return { role: message.from, content: messageText };
38
+ }
39
+ return { role: message.from, content: message.content };
40
+ })
41
+ );
42
+ }
43
+
44
+ async function prepareFiles(
45
+ imageProcessor: ReturnType<typeof makeImageProcessor>,
46
+ files: MessageFile[],
47
+ isMultimodal: boolean
48
+ ): Promise<{
49
+ imageParts: OpenAI.Chat.Completions.ChatCompletionContentPartImage[];
50
+ textContent: string;
51
+ }> {
52
+ const imageFiles = files.filter((file) => file.mime.startsWith("image/"));
53
+ const textFiles = files.filter((file) => {
54
+ const mime = (file.mime || "").toLowerCase();
55
+ const [fileType, fileSubtype] = mime.split("/");
56
+ return TEXT_MIME_ALLOWLIST.some((allowed) => {
57
+ const [type, subtype] = allowed.toLowerCase().split("/");
58
+ const typeOk = type === "*" || type === fileType;
59
+ const subOk = subtype === "*" || subtype === fileSubtype;
60
+ return typeOk && subOk;
61
+ });
62
+ });
63
+
64
+ let imageParts: OpenAI.Chat.Completions.ChatCompletionContentPartImage[] = [];
65
+ if (isMultimodal && imageFiles.length > 0) {
66
+ const processedFiles = await Promise.all(imageFiles.map(imageProcessor));
67
+ imageParts = processedFiles.map((file) => ({
68
+ type: "image_url" as const,
69
+ image_url: {
70
+ url: `data:${file.mime};base64,${file.image.toString("base64")}`,
71
+ detail: "auto",
72
+ },
73
+ }));
74
+ }
75
+
76
+ let textContent = "";
77
+ if (textFiles.length > 0) {
78
+ const textParts = await Promise.all(
79
+ textFiles.map(async (file) => {
80
+ const content = Buffer.from(file.value, "base64").toString("utf-8");
81
+ return `<document name="${file.name}" type="${file.mime}">\n${content}\n</document>`;
82
+ })
83
+ );
84
+ textContent = textParts.join("\n\n");
85
+ }
86
+
87
+ return { imageParts, textContent };
88
+ }
@@ -0,0 +1,21 @@
1
+ import type { EndpointMessage } from "../../endpoints/endpoints";
2
+
3
+ const ROUTER_REASONING_REGEX = /<think>[\s\S]*?(?:<\/think>|$)/g;
4
+
5
+ export function stripReasoningBlocks(text: string): string {
6
+ const stripped = text.replace(ROUTER_REASONING_REGEX, "");
7
+ return stripped === text ? text : stripped.trim();
8
+ }
9
+
10
+ export function stripReasoningFromMessageForRouting(message: EndpointMessage): EndpointMessage {
11
+ const clone = { ...message } as EndpointMessage & { reasoning?: string };
12
+ if ("reasoning" in clone) {
13
+ delete clone.reasoning;
14
+ }
15
+ const content =
16
+ typeof message.content === "string" ? stripReasoningBlocks(message.content) : message.content;
17
+ return {
18
+ ...clone,
19
+ content,
20
+ };
21
+ }
@@ -0,0 +1,49 @@
1
+ import type { OpenAiTool } from "$lib/server/mcp/tools";
2
+
3
+ export function buildToolPreprompt(tools: OpenAiTool[], autopilot?: boolean): string {
4
+ if (!Array.isArray(tools) || tools.length === 0) return "";
5
+ const names = tools
6
+ .map((t) => (t?.function?.name ? String(t.function.name) : ""))
7
+ .filter((s) => s.length > 0);
8
+ if (names.length === 0) return "";
9
+ const now = new Date();
10
+ const currentDate = now.toLocaleDateString("en-US", {
11
+ year: "numeric",
12
+ month: "long",
13
+ day: "numeric",
14
+ });
15
+ const isoDate = `${now.getFullYear()}-${String(now.getMonth() + 1).padStart(2, "0")}-${String(now.getDate()).padStart(2, "0")}`;
16
+ const lines = [
17
+ `You have access to these tools: ${names.join(", ")}.`,
18
+ `Today's date: ${currentDate} (${isoDate}).`,
19
+ ];
20
+
21
+ if (autopilot) {
22
+ lines.push(
23
+ `AUTOPILOT MODE ENABLED — PARALLEL SWARM EXECUTION. Follow these rules STRICTLY:`,
24
+ `1. NEVER ask for confirmation. Make reasonable assumptions. Proceed immediately.`,
25
+ `2. ALWAYS call MULTIPLE tools in parallel. If a task needs search + memory + routing, call ALL THREE at once in a single response — not sequentially.`,
26
+ `3. For complex tasks: spawn a SWARM of parallel tool calls. Example: call search, web_research, memory_search, and hooks_route ALL in one response.`,
27
+ `4. After getting results, immediately call the NEXT BATCH of tools in parallel. Keep chaining until done.`,
28
+ `5. If a tool fails, try alternatives immediately — do not report failure and stop.`,
29
+ `6. Only provide a final text response when ALL work is done and you have nothing left to execute.`,
30
+ `7. NEVER explain what you plan to do. Just DO IT by calling tools.`,
31
+ `8. Maximize parallel execution: if you can call 3+ tools at once, DO IT. Sequential is only for dependencies.`,
32
+ );
33
+ } else {
34
+ lines.push(
35
+ `IMPORTANT: Do NOT call a tool unless the user's request requires capabilities you lack (e.g., real-time data, image generation, code execution) or external information you do not have. For tasks like writing code, creative writing, math, or building apps, respond directly without tools. When in doubt, do not use a tool.`,
36
+ );
37
+ }
38
+
39
+ lines.push(
40
+ `PARALLEL TOOL CALLS: When multiple tool calls are needed and they are independent of each other (i.e., one does not need the result of another), call them all at once in a single response instead of one at a time. Only chain tool calls sequentially when a later call depends on an earlier call's output.`,
41
+ `SEARCH: Use 3-6 precise keywords. For historical events, include the year the event occurred. For recent or current topics, use today's year (${now.getFullYear()}). When a tool accepts date-range parameters (e.g., startPublishedDate, endPublishedDate), always use today's date (${isoDate}) as the end date unless the user specifies otherwise. For multi-part questions, search each part separately.`,
42
+ `ANSWER: State only facts explicitly in the results. If info is missing or results conflict, say so. Never fabricate URLs or facts.`,
43
+ `INTERACTIVE APPS: When asked to build an interactive application, game, or visualization without a specific language/framework preference, create a single self-contained HTML file with embedded CSS and JavaScript.`,
44
+ `If a tool generates an image, you can inline it directly: ![alt text](image_url).`,
45
+ `If a tool needs an image, set its image field ("input_image", "image", or "image_url") to a reference like "image_1", "image_2", etc. (ordered by when the user uploaded them).`,
46
+ `Default to image references; only use a full http(s) URL when the tool description explicitly asks for one, or reuse a URL a previous tool returned.`,
47
+ );
48
+ return lines.join(" ");
49
+ }
@@ -0,0 +1,77 @@
1
+ import { Address4, Address6 } from "ip-address";
2
+ import { isIP } from "node:net";
3
+
4
+ const UNSAFE_IPV4_SUBNETS = [
5
+ "0.0.0.0/8",
6
+ "100.64.0.0/10",
7
+ "127.0.0.0/8",
8
+ "169.254.0.0/16",
9
+ "172.16.0.0/12",
10
+ "192.168.0.0/16",
11
+ ].map((s) => new Address4(s));
12
+
13
+ function isUnsafeIp(address: string): boolean {
14
+ const family = isIP(address);
15
+
16
+ if (family === 4) {
17
+ const addr = new Address4(address);
18
+ return UNSAFE_IPV4_SUBNETS.some((subnet) => addr.isInSubnet(subnet));
19
+ }
20
+
21
+ if (family === 6) {
22
+ const addr = new Address6(address);
23
+ // Check IPv4-mapped IPv6 addresses (e.g. ::ffff:127.0.0.1)
24
+ if (addr.is4()) {
25
+ const v4 = addr.to4();
26
+ return UNSAFE_IPV4_SUBNETS.some((subnet) => v4.isInSubnet(subnet));
27
+ }
28
+ return addr.isLoopback() || addr.isLinkLocal();
29
+ }
30
+
31
+ return true; // Unknown format → block
32
+ }
33
+
34
+ /**
35
+ * Synchronous URL validation: checks protocol and hostname string.
36
+ */
37
+ export function isValidUrl(urlString: string): boolean {
38
+ try {
39
+ const url = new URL(urlString.trim());
40
+ const hostname = url.hostname.toLowerCase();
41
+ // Allow HTTP for localhost/loopback/Docker-internal (dev & local MCP bridge)
42
+ if (
43
+ hostname === "localhost" ||
44
+ hostname === "127.0.0.1" ||
45
+ hostname === "::1" ||
46
+ hostname === "host.docker.internal"
47
+ ) {
48
+ return url.protocol === "http:" || url.protocol === "https:";
49
+ }
50
+ // Allow HTTP for Docker-internal service names (no dots = private network)
51
+ if (!hostname.includes(".") && url.protocol === "http:") {
52
+ return true;
53
+ }
54
+ if (url.protocol !== "https:") {
55
+ return false;
56
+ }
57
+ // If the hostname is a raw IP literal, validate it
58
+ const cleanHostname = hostname.replace(/^\[|]$/g, "");
59
+ if (isIP(cleanHostname)) {
60
+ return !isUnsafeIp(cleanHostname);
61
+ }
62
+ return true;
63
+ } catch {
64
+ return false;
65
+ }
66
+ }
67
+
68
+ /**
69
+ * Assert that a resolved IP address is safe (not internal/private).
70
+ * Throws if the IP is internal. Used in undici's custom DNS lookup
71
+ * to validate IPs at connection time (prevents TOCTOU DNS rebinding).
72
+ */
73
+ export function assertSafeIp(address: string, hostname: string): void {
74
+ if (isUnsafeIp(address)) {
75
+ throw new Error(`Resolved IP for ${hostname} is internal (${address})`);
76
+ }
77
+ }
@@ -0,0 +1,30 @@
1
+ import { z } from "zod";
2
+ import { config } from "$lib/server/config";
3
+ import JSON5 from "json5";
4
+
5
+ const sanitizeJSONEnv = (val: string, fallback: string) => {
6
+ const raw = (val ?? "").trim();
7
+ const unquoted = raw.startsWith("`") && raw.endsWith("`") ? raw.slice(1, -1) : raw;
8
+ return unquoted || fallback;
9
+ };
10
+
11
+ // RATE_LIMIT is the legacy way to define messages per minute limit
12
+ export const usageLimitsSchema = z
13
+ .object({
14
+ conversations: z.coerce.number().optional(), // how many conversations
15
+ messages: z.coerce.number().optional(), // how many messages in a conversation
16
+ messageLength: z.coerce.number().optional(), // how long can a message be before we cut it off
17
+ messagesPerMinute: z
18
+ .preprocess((val) => {
19
+ if (val === undefined) {
20
+ return config.RATE_LIMIT;
21
+ }
22
+ return val;
23
+ }, z.coerce.number().optional())
24
+ .optional(), // how many messages per minute
25
+ })
26
+ .optional();
27
+
28
+ export const usageLimits = usageLimitsSchema.parse(
29
+ JSON5.parse(sanitizeJSONEnv(config.USAGE_LIMITS, "{}"))
30
+ );
@@ -0,0 +1,175 @@
1
+ /**
2
+ * Autopilot Store — Svelte 5 runes-based store for managing autopilot Web Worker state.
3
+ *
4
+ * Provides reactive state for autopilot groups, tasks, and text content.
5
+ * Communicates with AutopilotWorker and DetailFetchWorker via postMessage.
6
+ *
7
+ * ADR-037 Part 2+3: Parallel Task UI + Web Workers
8
+ */
9
+
10
+ import type { GroupState, AutopilotUIUpdate } from "$lib/workers/autopilotWorker";
11
+ import type { DetailWorkerOutgoing } from "$lib/workers/detailFetchWorker";
12
+
13
+ export interface AutopilotState {
14
+ active: boolean;
15
+ maxSteps: number;
16
+ groups: GroupState[];
17
+ textContent: string;
18
+ error: string | null;
19
+ totalSteps: number;
20
+ totalTasks: number;
21
+ duration: number;
22
+ paused: boolean;
23
+ pauseReason: string | null;
24
+ }
25
+
26
+ const defaultState: AutopilotState = {
27
+ active: false,
28
+ maxSteps: 20,
29
+ groups: [],
30
+ textContent: "",
31
+ error: null,
32
+ totalSteps: 0,
33
+ totalTasks: 0,
34
+ duration: 0,
35
+ paused: false,
36
+ pauseReason: null,
37
+ };
38
+
39
+ let state = $state<AutopilotState>({ ...defaultState });
40
+
41
+ let autopilotWorker: Worker | null = null;
42
+ let detailWorker: Worker | null = null;
43
+ const detailCallbacks = new Map<string, (content: string | null, error?: string) => void>();
44
+
45
+ async function ensureWorkers() {
46
+ if (typeof window === "undefined") return;
47
+
48
+ if (!autopilotWorker) {
49
+ const mod = await import("$lib/workers/autopilotWorker?worker");
50
+ autopilotWorker = new mod.default();
51
+ autopilotWorker.onmessage = handleWorkerMessage;
52
+ }
53
+
54
+ if (!detailWorker) {
55
+ const mod = await import("$lib/workers/detailFetchWorker?worker");
56
+ detailWorker = new mod.default();
57
+ detailWorker.onmessage = handleDetailMessage;
58
+ }
59
+ }
60
+
61
+ function handleWorkerMessage(e: MessageEvent) {
62
+ const msg = e.data;
63
+
64
+ switch (msg.type) {
65
+ case "batch_update":
66
+ state.groups = msg.groups;
67
+ for (const update of msg.updates as AutopilotUIUpdate[]) {
68
+ applyUpdate(update);
69
+ }
70
+ break;
71
+
72
+ case "text":
73
+ state.textContent += msg.content;
74
+ break;
75
+
76
+ case "done":
77
+ state.active = false;
78
+ state.groups = msg.groups;
79
+ break;
80
+
81
+ case "error":
82
+ state.active = false;
83
+ state.error = msg.error;
84
+ break;
85
+
86
+ case "stopped":
87
+ state.active = false;
88
+ state.groups = msg.groups;
89
+ break;
90
+ }
91
+ }
92
+
93
+ function applyUpdate(update: AutopilotUIUpdate) {
94
+ switch (update.type) {
95
+ case "start":
96
+ state.maxSteps = update.maxSteps;
97
+ break;
98
+ case "end":
99
+ state.totalSteps = update.totalSteps;
100
+ state.totalTasks = update.totalTasks;
101
+ state.duration = update.duration;
102
+ break;
103
+ case "text":
104
+ state.textContent += update.content;
105
+ break;
106
+ case "paused":
107
+ state.paused = true;
108
+ state.pauseReason = update.reason;
109
+ break;
110
+ case "error_event":
111
+ state.error = update.error;
112
+ break;
113
+ }
114
+ }
115
+
116
+ function handleDetailMessage(e: MessageEvent<DetailWorkerOutgoing>) {
117
+ const msg = e.data;
118
+ if (msg.type === "detail") {
119
+ const cb = detailCallbacks.get(msg.detailToken);
120
+ if (cb) {
121
+ cb(msg.content);
122
+ detailCallbacks.delete(msg.detailToken);
123
+ }
124
+ } else if (msg.type === "detail_error") {
125
+ const cb = detailCallbacks.get(msg.detailToken);
126
+ if (cb) {
127
+ cb(null, msg.error);
128
+ detailCallbacks.delete(msg.detailToken);
129
+ }
130
+ }
131
+ }
132
+
133
+ export function useAutopilot() {
134
+ return {
135
+ get state() {
136
+ return state;
137
+ },
138
+
139
+ async start(url: string, headers: Record<string, string>, body: unknown) {
140
+ await ensureWorkers();
141
+ Object.assign(state, { ...defaultState, active: true });
142
+ autopilotWorker?.postMessage({ type: "start", url, headers, body });
143
+ },
144
+
145
+ stop() {
146
+ autopilotWorker?.postMessage({ type: "stop" });
147
+ },
148
+
149
+ async fetchDetail(detailToken: string, bridgeUrl: string): Promise<string> {
150
+ await ensureWorkers();
151
+ return new Promise((resolve, reject) => {
152
+ detailCallbacks.set(detailToken, (content, error) => {
153
+ if (error) reject(new Error(error));
154
+ else resolve(content!);
155
+ });
156
+ detailWorker?.postMessage({ type: "fetch", detailToken, bridgeUrl });
157
+ });
158
+ },
159
+
160
+ prefetchDetail(detailToken: string, bridgeUrl: string) {
161
+ detailWorker?.postMessage({ type: "prefetch", detailToken, bridgeUrl });
162
+ },
163
+
164
+ evictDetail(detailToken: string) {
165
+ detailWorker?.postMessage({ type: "evict", detailToken });
166
+ },
167
+
168
+ destroy() {
169
+ autopilotWorker?.terminate();
170
+ detailWorker?.terminate();
171
+ autopilotWorker = null;
172
+ detailWorker = null;
173
+ },
174
+ };
175
+ }
@@ -0,0 +1,32 @@
1
+ export type BackgroundGeneration = {
2
+ id: string;
3
+ startedAt: number;
4
+ };
5
+
6
+ export const backgroundGenerationEntries = $state<BackgroundGeneration[]>([]);
7
+
8
+ export function addBackgroundGeneration(entry: BackgroundGeneration) {
9
+ const index = backgroundGenerationEntries.findIndex(({ id }) => id === entry.id);
10
+
11
+ if (index === -1) {
12
+ backgroundGenerationEntries.push(entry);
13
+ return;
14
+ }
15
+
16
+ backgroundGenerationEntries[index] = entry;
17
+ }
18
+
19
+ export function removeBackgroundGeneration(id: string) {
20
+ const index = backgroundGenerationEntries.findIndex((entry) => entry.id === id);
21
+ if (index === -1) return;
22
+
23
+ backgroundGenerationEntries.splice(index, 1);
24
+ }
25
+
26
+ export function clearBackgroundGenerations() {
27
+ backgroundGenerationEntries.length = 0;
28
+ }
29
+
30
+ export function hasBackgroundGeneration(id: string) {
31
+ return backgroundGenerationEntries.some((entry) => entry.id === id);
32
+ }
@@ -0,0 +1 @@
1
+ export * from "./backgroundGenerations.svelte";
@@ -0,0 +1,9 @@
1
+ import { writable } from "svelte/store";
2
+
3
+ export const ERROR_MESSAGES = {
4
+ default: "Oops, something went wrong.",
5
+ authOnly: "You have to be logged in.",
6
+ rateLimited: "You are sending too many messages. Try again later.",
7
+ };
8
+
9
+ export const error = writable<string | undefined>(undefined);
@@ -0,0 +1,3 @@
1
+ import { writable } from "svelte/store";
2
+
3
+ export const isAborted = writable<boolean>(false);
@@ -0,0 +1,4 @@
1
+ import { writable } from "svelte/store";
2
+
3
+ // null = unknown/loading, true = PRO, false = not PRO
4
+ export const isPro = writable<boolean | null>(null);
@@ -0,0 +1,3 @@
1
+ import { writable } from "svelte/store";
2
+
3
+ export const loading = writable(false);