@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,709 @@
1
+ import { describe, expect, it, beforeEach, afterAll } from "vitest";
2
+ import {
3
+ RvfCollection,
4
+ RvfGridFSBucket,
5
+ ObjectId,
6
+ initRvfStore,
7
+ flushToDisk,
8
+ enableMultiTenant,
9
+ listTenants,
10
+ getTenantStats,
11
+ } from "../rvf";
12
+ import { existsSync, unlinkSync, readFileSync } from "fs";
13
+ import { join } from "path";
14
+ import { tmpdir } from "os";
15
+ import { randomUUID } from "crypto";
16
+
17
+ // ---------------------------------------------------------------------------
18
+ // Test helpers
19
+ // ---------------------------------------------------------------------------
20
+
21
+ interface TestDoc {
22
+ _id?: string;
23
+ name: string;
24
+ age?: number;
25
+ tags?: string[];
26
+ createdAt?: Date;
27
+ updatedAt?: Date;
28
+ nested?: { field: string };
29
+ }
30
+
31
+ const TEST_DB_PATH = join(tmpdir(), `rvf-test-${randomUUID()}.json`);
32
+
33
+ beforeEach(() => {
34
+ // Re-initialize for a fresh store each test
35
+ initRvfStore("");
36
+ });
37
+
38
+ afterAll(() => {
39
+ if (existsSync(TEST_DB_PATH)) unlinkSync(TEST_DB_PATH);
40
+ });
41
+
42
+ // ---------------------------------------------------------------------------
43
+ // CRUD operations
44
+ // ---------------------------------------------------------------------------
45
+
46
+ describe("RvfCollection CRUD", () => {
47
+ it("insertOne and findOne", async () => {
48
+ const coll = new RvfCollection<TestDoc>("test_crud");
49
+ const result = await coll.insertOne({ name: "Alice", age: 30 });
50
+ expect(result.acknowledged).toBe(true);
51
+ expect(result.insertedId).toBeDefined();
52
+
53
+ const found = await coll.findOne({ name: "Alice" });
54
+ expect(found).not.toBeNull();
55
+ expect(found!.name).toBe("Alice");
56
+ expect(found!.age).toBe(30);
57
+ });
58
+
59
+ it("insertMany and find with toArray", async () => {
60
+ const coll = new RvfCollection<TestDoc>("test_insertmany");
61
+ await coll.insertMany([
62
+ { name: "Bob", age: 25 },
63
+ { name: "Carol", age: 35 },
64
+ { name: "Dave", age: 28 },
65
+ ]);
66
+
67
+ const all = await coll.find({}).toArray();
68
+ expect(all).toHaveLength(3);
69
+ });
70
+
71
+ it("updateOne with $set", async () => {
72
+ const coll = new RvfCollection<TestDoc>("test_update");
73
+ await coll.insertOne({ name: "Eve", age: 22 });
74
+ const result = await coll.updateOne({ name: "Eve" }, { $set: { age: 23 } });
75
+ expect(result.matchedCount).toBe(1);
76
+ expect(result.modifiedCount).toBe(1);
77
+
78
+ const updated = await coll.findOne({ name: "Eve" });
79
+ expect(updated!.age).toBe(23);
80
+ });
81
+
82
+ it("updateOne with upsert", async () => {
83
+ const coll = new RvfCollection<TestDoc>("test_upsert");
84
+ const result = await coll.updateOne(
85
+ { name: "Frank" },
86
+ { $set: { age: 40 } },
87
+ { upsert: true }
88
+ );
89
+ expect(result.upsertedCount).toBe(1);
90
+
91
+ const found = await coll.findOne({ name: "Frank" });
92
+ expect(found).not.toBeNull();
93
+ expect(found!.age).toBe(40);
94
+ });
95
+
96
+ it("updateOne with $setOnInsert during upsert", async () => {
97
+ const coll = new RvfCollection<TestDoc>("test_setoninsert");
98
+ await coll.updateOne(
99
+ { name: "Grace" },
100
+ { $set: { age: 50 }, $setOnInsert: { tags: ["new"] } },
101
+ { upsert: true }
102
+ );
103
+
104
+ const found = await coll.findOne({ name: "Grace" });
105
+ expect(found!.tags).toEqual(["new"]);
106
+ });
107
+
108
+ it("updateMany", async () => {
109
+ const coll = new RvfCollection<TestDoc>("test_updatemany");
110
+ await coll.insertMany([
111
+ { name: "A", age: 20 },
112
+ { name: "B", age: 20 },
113
+ { name: "C", age: 30 },
114
+ ]);
115
+
116
+ const result = await coll.updateMany({ age: 20 }, { $set: { age: 21 } });
117
+ expect(result.matchedCount).toBe(2);
118
+ expect(result.modifiedCount).toBe(2);
119
+ });
120
+
121
+ it("deleteOne", async () => {
122
+ const coll = new RvfCollection<TestDoc>("test_delete");
123
+ await coll.insertOne({ name: "ToDelete", age: 99 });
124
+ const result = await coll.deleteOne({ name: "ToDelete" });
125
+ expect(result.deletedCount).toBe(1);
126
+
127
+ const found = await coll.findOne({ name: "ToDelete" });
128
+ expect(found).toBeNull();
129
+ });
130
+
131
+ it("deleteMany", async () => {
132
+ const coll = new RvfCollection<TestDoc>("test_deletemany");
133
+ await coll.insertMany([
134
+ { name: "X", age: 10 },
135
+ { name: "Y", age: 10 },
136
+ { name: "Z", age: 20 },
137
+ ]);
138
+
139
+ const result = await coll.deleteMany({ age: 10 });
140
+ expect(result.deletedCount).toBe(2);
141
+ expect(await coll.countDocuments({})).toBe(1);
142
+ });
143
+
144
+ it("countDocuments", async () => {
145
+ const coll = new RvfCollection<TestDoc>("test_count");
146
+ await coll.insertMany([
147
+ { name: "A", age: 1 },
148
+ { name: "B", age: 2 },
149
+ { name: "C", age: 3 },
150
+ ]);
151
+
152
+ expect(await coll.countDocuments({})).toBe(3);
153
+ expect(await coll.countDocuments({ age: { $gt: 1 } })).toBe(2);
154
+ });
155
+
156
+ it("distinct", async () => {
157
+ const coll = new RvfCollection<TestDoc>("test_distinct");
158
+ await coll.insertMany([
159
+ { name: "A", age: 10 },
160
+ { name: "B", age: 20 },
161
+ { name: "C", age: 10 },
162
+ ]);
163
+
164
+ const ages = await coll.distinct("age");
165
+ expect(ages.sort()).toEqual([10, 20]);
166
+ });
167
+
168
+ it("findOneAndUpdate", async () => {
169
+ const coll = new RvfCollection<TestDoc>("test_findoneupdate");
170
+ await coll.insertOne({ name: "Hank", age: 45 });
171
+
172
+ const result = await coll.findOneAndUpdate(
173
+ { name: "Hank" },
174
+ { $set: { age: 46 } },
175
+ { returnDocument: "after" }
176
+ );
177
+ expect(result.value).not.toBeNull();
178
+ expect(result.value!.age).toBe(46);
179
+ });
180
+
181
+ it("findOneAndDelete", async () => {
182
+ const coll = new RvfCollection<TestDoc>("test_findonedelete");
183
+ await coll.insertOne({ name: "Ivan", age: 60 });
184
+
185
+ const result = await coll.findOneAndDelete({ name: "Ivan" });
186
+ expect(result.value).not.toBeNull();
187
+ expect(result.value!.name).toBe("Ivan");
188
+ expect(await coll.countDocuments({})).toBe(0);
189
+ });
190
+
191
+ it("bulkWrite", async () => {
192
+ const coll = new RvfCollection<TestDoc>("test_bulkwrite");
193
+ await coll.insertMany([
194
+ { name: "A", age: 1 },
195
+ { name: "B", age: 2 },
196
+ ]);
197
+
198
+ await coll.bulkWrite([
199
+ { updateOne: { filter: { name: "A" }, update: { $set: { age: 10 } } } },
200
+ { updateOne: { filter: { name: "B" }, update: { $set: { age: 20 } } } },
201
+ ]);
202
+
203
+ expect((await coll.findOne({ name: "A" }))!.age).toBe(10);
204
+ expect((await coll.findOne({ name: "B" }))!.age).toBe(20);
205
+ });
206
+ });
207
+
208
+ // ---------------------------------------------------------------------------
209
+ // Query operators
210
+ // ---------------------------------------------------------------------------
211
+
212
+ describe("Query operators", () => {
213
+ it("$gt, $gte, $lt, $lte", async () => {
214
+ const coll = new RvfCollection<TestDoc>("test_comparison");
215
+ await coll.insertMany([
216
+ { name: "A", age: 10 },
217
+ { name: "B", age: 20 },
218
+ { name: "C", age: 30 },
219
+ ]);
220
+
221
+ expect(await coll.countDocuments({ age: { $gt: 15 } })).toBe(2);
222
+ expect(await coll.countDocuments({ age: { $gte: 20 } })).toBe(2);
223
+ expect(await coll.countDocuments({ age: { $lt: 25 } })).toBe(2);
224
+ expect(await coll.countDocuments({ age: { $lte: 20 } })).toBe(2);
225
+ });
226
+
227
+ it("$ne", async () => {
228
+ const coll = new RvfCollection<TestDoc>("test_ne");
229
+ await coll.insertMany([
230
+ { name: "A", age: 10 },
231
+ { name: "B", age: 20 },
232
+ ]);
233
+
234
+ expect(await coll.countDocuments({ age: { $ne: 10 } })).toBe(1);
235
+ });
236
+
237
+ it("$in and $nin", async () => {
238
+ const coll = new RvfCollection<TestDoc>("test_in");
239
+ await coll.insertMany([
240
+ { name: "A", age: 10 },
241
+ { name: "B", age: 20 },
242
+ { name: "C", age: 30 },
243
+ ]);
244
+
245
+ expect(await coll.countDocuments({ age: { $in: [10, 30] } })).toBe(2);
246
+ expect(await coll.countDocuments({ age: { $nin: [10, 30] } })).toBe(1);
247
+ });
248
+
249
+ it("$exists", async () => {
250
+ const coll = new RvfCollection<TestDoc>("test_exists");
251
+ await coll.insertMany([
252
+ { name: "A", tags: ["x"] },
253
+ { name: "B" },
254
+ ]);
255
+
256
+ expect(await coll.countDocuments({ tags: { $exists: true } })).toBe(1);
257
+ expect(await coll.countDocuments({ tags: { $exists: false } })).toBe(1);
258
+ });
259
+
260
+ it("$or and $and", async () => {
261
+ const coll = new RvfCollection<TestDoc>("test_logical");
262
+ await coll.insertMany([
263
+ { name: "A", age: 10 },
264
+ { name: "B", age: 20 },
265
+ { name: "C", age: 30 },
266
+ ]);
267
+
268
+ expect(await coll.countDocuments({ $or: [{ age: 10 }, { age: 30 }] })).toBe(2);
269
+ expect(
270
+ await coll.countDocuments({ $and: [{ age: { $gte: 10 } }, { age: { $lte: 20 } }] })
271
+ ).toBe(2);
272
+ });
273
+
274
+ it("$regex", async () => {
275
+ const coll = new RvfCollection<TestDoc>("test_regex");
276
+ await coll.insertMany([
277
+ { name: "Alice" },
278
+ { name: "Bob" },
279
+ { name: "alicia" },
280
+ ]);
281
+
282
+ expect(await coll.countDocuments({ name: { $regex: "ali", $options: "i" } })).toBe(2);
283
+ });
284
+
285
+ it("$not", async () => {
286
+ const coll = new RvfCollection<TestDoc>("test_not");
287
+ await coll.insertMany([
288
+ { name: "A", age: 10 },
289
+ { name: "B", age: 20 },
290
+ ]);
291
+
292
+ expect(await coll.countDocuments({ age: { $not: { $gt: 15 } } })).toBe(1);
293
+ });
294
+ });
295
+
296
+ // ---------------------------------------------------------------------------
297
+ // Update operators
298
+ // ---------------------------------------------------------------------------
299
+
300
+ describe("Update operators", () => {
301
+ it("$inc", async () => {
302
+ const coll = new RvfCollection<TestDoc>("test_inc");
303
+ await coll.insertOne({ name: "Counter", age: 0 });
304
+ await coll.updateOne({ name: "Counter" }, { $inc: { age: 5 } });
305
+ expect((await coll.findOne({ name: "Counter" }))!.age).toBe(5);
306
+ });
307
+
308
+ it("$push", async () => {
309
+ const coll = new RvfCollection<TestDoc>("test_push");
310
+ await coll.insertOne({ name: "Tags", tags: ["a"] });
311
+ await coll.updateOne({ name: "Tags" }, { $push: { tags: "b" } });
312
+ expect((await coll.findOne({ name: "Tags" }))!.tags).toEqual(["a", "b"]);
313
+ });
314
+
315
+ it("$push with $each", async () => {
316
+ const coll = new RvfCollection<TestDoc>("test_push_each");
317
+ await coll.insertOne({ name: "Tags", tags: [] });
318
+ await coll.updateOne({ name: "Tags" }, { $push: { tags: { $each: ["x", "y"] } } });
319
+ expect((await coll.findOne({ name: "Tags" }))!.tags).toEqual(["x", "y"]);
320
+ });
321
+
322
+ it("$pull", async () => {
323
+ const coll = new RvfCollection<TestDoc>("test_pull");
324
+ await coll.insertOne({ name: "Tags", tags: ["a", "b", "c"] });
325
+ await coll.updateOne({ name: "Tags" }, { $pull: { tags: "b" } });
326
+ expect((await coll.findOne({ name: "Tags" }))!.tags).toEqual(["a", "c"]);
327
+ });
328
+
329
+ it("$addToSet", async () => {
330
+ const coll = new RvfCollection<TestDoc>("test_addtoset");
331
+ await coll.insertOne({ name: "Tags", tags: ["a"] });
332
+ await coll.updateOne({ name: "Tags" }, { $addToSet: { tags: "a" } });
333
+ expect((await coll.findOne({ name: "Tags" }))!.tags).toEqual(["a"]);
334
+ await coll.updateOne({ name: "Tags" }, { $addToSet: { tags: "b" } });
335
+ expect((await coll.findOne({ name: "Tags" }))!.tags).toEqual(["a", "b"]);
336
+ });
337
+
338
+ it("$unset", async () => {
339
+ const coll = new RvfCollection<TestDoc>("test_unset");
340
+ await coll.insertOne({ name: "Nested", nested: { field: "val" } });
341
+ await coll.updateOne({ name: "Nested" }, { $unset: { nested: "" } });
342
+ const doc = await coll.findOne({ name: "Nested" });
343
+ expect(doc!.nested).toBeUndefined();
344
+ });
345
+ });
346
+
347
+ // ---------------------------------------------------------------------------
348
+ // Cursor operations
349
+ // ---------------------------------------------------------------------------
350
+
351
+ describe("Cursor", () => {
352
+ it("sort, limit, skip", async () => {
353
+ const coll = new RvfCollection<TestDoc>("test_cursor");
354
+ await coll.insertMany([
355
+ { name: "A", age: 30 },
356
+ { name: "B", age: 10 },
357
+ { name: "C", age: 20 },
358
+ ]);
359
+
360
+ const sorted = await coll.find({}).sort({ age: 1 }).toArray();
361
+ expect(sorted.map((d) => d.age)).toEqual([10, 20, 30]);
362
+
363
+ const limited = await coll.find({}).sort({ age: 1 }).limit(2).toArray();
364
+ expect(limited).toHaveLength(2);
365
+
366
+ const skipped = await coll.find({}).sort({ age: 1 }).skip(1).limit(1).toArray();
367
+ expect(skipped[0].age).toBe(20);
368
+ });
369
+
370
+ it("async iterator", async () => {
371
+ const coll = new RvfCollection<TestDoc>("test_asynciter");
372
+ await coll.insertMany([{ name: "X" }, { name: "Y" }]);
373
+
374
+ const names: string[] = [];
375
+ for await (const doc of coll.find({})) {
376
+ names.push(doc.name);
377
+ }
378
+ expect(names).toHaveLength(2);
379
+ });
380
+
381
+ it("tryNext / hasNext / next", async () => {
382
+ const coll = new RvfCollection<TestDoc>("test_trynext");
383
+ await coll.insertMany([{ name: "A" }, { name: "B" }]);
384
+
385
+ const cursor = coll.find({});
386
+ expect(await cursor.hasNext()).toBe(true);
387
+ const first = await cursor.next();
388
+ expect(first).not.toBeNull();
389
+ const second = await cursor.tryNext();
390
+ expect(second).not.toBeNull();
391
+ const third = await cursor.tryNext();
392
+ expect(third).toBeNull();
393
+ });
394
+
395
+ it("map transforms results", async () => {
396
+ const coll = new RvfCollection<TestDoc>("test_map");
397
+ await coll.insertMany([{ name: "A", age: 10 }, { name: "B", age: 20 }]);
398
+
399
+ const names = await coll.find({}).map((doc) => doc.name).toArray();
400
+ expect(names).toEqual(expect.arrayContaining(["A", "B"]));
401
+ });
402
+ });
403
+
404
+ // ---------------------------------------------------------------------------
405
+ // Aggregation
406
+ // ---------------------------------------------------------------------------
407
+
408
+ describe("Aggregation", () => {
409
+ it("$match + $sort + $limit", async () => {
410
+ const coll = new RvfCollection<TestDoc>("test_agg");
411
+ await coll.insertMany([
412
+ { name: "A", age: 10 },
413
+ { name: "B", age: 20 },
414
+ { name: "C", age: 30 },
415
+ ]);
416
+
417
+ const result = await coll
418
+ .aggregate([{ $match: { age: { $gte: 15 } } }, { $sort: { age: -1 } }, { $limit: 1 }])
419
+ .toArray();
420
+ expect(result).toHaveLength(1);
421
+ expect(result[0].age).toBe(30);
422
+ });
423
+
424
+ it("aggregate().next()", async () => {
425
+ const coll = new RvfCollection<TestDoc>("test_agg_next");
426
+ await coll.insertMany([{ name: "A", age: 10 }, { name: "B", age: 20 }]);
427
+
428
+ const first = await coll.aggregate([{ $sort: { age: 1 } }]).next();
429
+ expect(first).not.toBeNull();
430
+ expect(first!.age).toBe(10);
431
+ });
432
+
433
+ it("$group with $sum", async () => {
434
+ const coll = new RvfCollection<TestDoc>("test_agg_group");
435
+ await coll.insertMany([
436
+ { name: "A", age: 10, tags: ["x"] },
437
+ { name: "B", age: 20, tags: ["x"] },
438
+ { name: "C", age: 30, tags: ["y"] },
439
+ ]);
440
+
441
+ const result = await coll
442
+ .aggregate([
443
+ { $group: { _id: null, totalAge: { $sum: "$age" }, count: { $sum: 1 } } },
444
+ ])
445
+ .toArray();
446
+
447
+ expect(result).toHaveLength(1);
448
+ expect(result[0].totalAge).toBe(60);
449
+ expect(result[0].count).toBe(3);
450
+ });
451
+ });
452
+
453
+ // ---------------------------------------------------------------------------
454
+ // GridFS replacement
455
+ // ---------------------------------------------------------------------------
456
+
457
+ describe("RvfGridFSBucket", () => {
458
+ it("upload and download", async () => {
459
+ const bucket = new RvfGridFSBucket();
460
+ const stream = bucket.openUploadStream("test.txt", { contentType: "text/plain" });
461
+ stream.write(Buffer.from("Hello, RVF!"));
462
+ await stream.end();
463
+
464
+ const chunks = await bucket.openDownloadStream(stream.id).toArray();
465
+ expect(chunks).toHaveLength(1);
466
+ });
467
+
468
+ it("delete file", async () => {
469
+ const bucket = new RvfGridFSBucket();
470
+ const stream = bucket.openUploadStream("delete-me.txt");
471
+ stream.write(Buffer.from("data"));
472
+ await stream.end();
473
+
474
+ await bucket.delete(stream.id);
475
+ await expect(bucket.openDownloadStream(stream.id).toArray()).rejects.toThrow("File not found");
476
+ });
477
+ });
478
+
479
+ // ---------------------------------------------------------------------------
480
+ // Multi-tenant
481
+ // ---------------------------------------------------------------------------
482
+
483
+ describe("Multi-tenant", () => {
484
+ it("tenant-scoped collections are isolated", async () => {
485
+ enableMultiTenant(true);
486
+ const coll = new RvfCollection<TestDoc>("shared_coll");
487
+
488
+ const tenantA = coll.forTenant("tenant-a");
489
+ const tenantB = coll.forTenant("tenant-b");
490
+
491
+ await tenantA.insertOne({ name: "Alice" });
492
+ await tenantB.insertOne({ name: "Bob" });
493
+
494
+ expect(await tenantA.countDocuments({})).toBe(1);
495
+ expect(await tenantB.countDocuments({})).toBe(1);
496
+ expect((await tenantA.findOne({}))!.name).toBe("Alice");
497
+ expect((await tenantB.findOne({}))!.name).toBe("Bob");
498
+
499
+ // Global collection should be empty (tenants don't pollute it)
500
+ expect(await coll.countDocuments({})).toBe(0);
501
+ });
502
+
503
+ it("listTenants and getTenantStats", async () => {
504
+ enableMultiTenant(true);
505
+ const coll = new RvfCollection<TestDoc>("stats_coll");
506
+
507
+ await coll.forTenant("t1").insertMany([{ name: "A" }, { name: "B" }]);
508
+ await coll.forTenant("t2").insertOne({ name: "C" });
509
+
510
+ expect(listTenants()).toContain("t1");
511
+ expect(listTenants()).toContain("t2");
512
+
513
+ const stats = getTenantStats();
514
+ expect(stats["t1"].documents).toBe(2);
515
+ expect(stats["t2"].documents).toBe(1);
516
+ });
517
+ });
518
+
519
+ // ---------------------------------------------------------------------------
520
+ // Persistence
521
+ // ---------------------------------------------------------------------------
522
+
523
+ describe("Persistence", () => {
524
+ it("flush to disk and reload", async () => {
525
+ initRvfStore(TEST_DB_PATH);
526
+ const coll = new RvfCollection<TestDoc>("persist_test");
527
+ await coll.insertMany([
528
+ { name: "Persisted1", age: 1 },
529
+ { name: "Persisted2", age: 2 },
530
+ ]);
531
+
532
+ flushToDisk();
533
+ expect(existsSync(TEST_DB_PATH)).toBe(true);
534
+
535
+ // Verify file structure
536
+ const data = JSON.parse(readFileSync(TEST_DB_PATH, "utf-8"));
537
+ expect(data.rvf_version).toBe("2.0");
538
+ expect(data.format).toBe("rvf-database");
539
+ expect(data.metadata.doc_count).toBeGreaterThan(0);
540
+
541
+ // Reload from disk
542
+ initRvfStore(TEST_DB_PATH);
543
+ const coll2 = new RvfCollection<TestDoc>("persist_test");
544
+ const docs = await coll2.find({}).toArray();
545
+ expect(docs.length).toBe(2);
546
+ expect(docs.find((d) => d.name === "Persisted1")).toBeTruthy();
547
+ });
548
+ });
549
+
550
+ // ---------------------------------------------------------------------------
551
+ // ObjectId
552
+ // ---------------------------------------------------------------------------
553
+
554
+ describe("ObjectId", () => {
555
+ it("equals and toString", () => {
556
+ const id = new ObjectId("abc-123");
557
+ expect(id.toString()).toBe("abc-123");
558
+ expect(id.equals("abc-123")).toBe(true);
559
+ expect(id.equals(new ObjectId("abc-123"))).toBe(true);
560
+ expect(id.equals(new ObjectId("xyz-999"))).toBe(false);
561
+ });
562
+
563
+ it("createFromHexString", () => {
564
+ const id = ObjectId.createFromHexString("hex-val");
565
+ expect(id.toString()).toBe("hex-val");
566
+ });
567
+
568
+ it("toJSON", () => {
569
+ const id = new ObjectId("json-test");
570
+ expect(JSON.stringify({ id })).toBe('{"id":"json-test"}');
571
+ });
572
+ });
573
+
574
+ // ---------------------------------------------------------------------------
575
+ // Performance benchmark
576
+ // ---------------------------------------------------------------------------
577
+
578
+ describe("Performance benchmark", () => {
579
+ it("insert 10,000 documents", async () => {
580
+ const coll = new RvfCollection<TestDoc>("bench_insert");
581
+ const docs = Array.from({ length: 10000 }, (_, i) => ({
582
+ name: `user-${i}`,
583
+ age: Math.floor(Math.random() * 100),
584
+ tags: [`tag-${i % 10}`],
585
+ }));
586
+
587
+ const start = performance.now();
588
+ await coll.insertMany(docs);
589
+ const elapsed = performance.now() - start;
590
+
591
+ console.log(` Insert 10k docs: ${elapsed.toFixed(1)}ms`);
592
+ expect(elapsed).toBeLessThan(5000); // Should be well under 5s
593
+ expect(await coll.countDocuments({})).toBe(10000);
594
+ });
595
+
596
+ it("find with filter on 10k docs", async () => {
597
+ const coll = new RvfCollection<TestDoc>("bench_find");
598
+ await coll.insertMany(
599
+ Array.from({ length: 10000 }, (_, i) => ({
600
+ name: `user-${i}`,
601
+ age: i % 100,
602
+ }))
603
+ );
604
+
605
+ const start = performance.now();
606
+ const results = await coll.find({ age: { $gte: 50, $lt: 60 } }).toArray();
607
+ const elapsed = performance.now() - start;
608
+
609
+ console.log(` Find with range filter (10k): ${elapsed.toFixed(1)}ms (${results.length} results)`);
610
+ expect(elapsed).toBeLessThan(1000);
611
+ expect(results.length).toBe(1000); // 10% of 10k
612
+ });
613
+
614
+ it("updateMany on 10k docs", async () => {
615
+ const coll = new RvfCollection<TestDoc>("bench_update");
616
+ await coll.insertMany(
617
+ Array.from({ length: 10000 }, (_, i) => ({
618
+ name: `user-${i}`,
619
+ age: i % 100,
620
+ }))
621
+ );
622
+
623
+ const start = performance.now();
624
+ const result = await coll.updateMany(
625
+ { age: { $lt: 50 } },
626
+ { $inc: { age: 100 } }
627
+ );
628
+ const elapsed = performance.now() - start;
629
+
630
+ console.log(` UpdateMany (5k matched): ${elapsed.toFixed(1)}ms`);
631
+ expect(elapsed).toBeLessThan(3000);
632
+ expect(result.matchedCount).toBe(5000);
633
+ });
634
+
635
+ it("aggregate pipeline on 10k docs", async () => {
636
+ const coll = new RvfCollection<TestDoc>("bench_agg");
637
+ await coll.insertMany(
638
+ Array.from({ length: 10000 }, (_, i) => ({
639
+ name: `user-${i}`,
640
+ age: i % 100,
641
+ tags: [`group-${i % 5}`],
642
+ }))
643
+ );
644
+
645
+ const start = performance.now();
646
+ const result = await coll
647
+ .aggregate([
648
+ { $match: { age: { $gte: 25 } } },
649
+ { $sort: { age: -1 } },
650
+ { $limit: 100 },
651
+ ])
652
+ .toArray();
653
+ const elapsed = performance.now() - start;
654
+
655
+ console.log(` Aggregate (match+sort+limit): ${elapsed.toFixed(1)}ms`);
656
+ expect(elapsed).toBeLessThan(2000);
657
+ expect(result).toHaveLength(100);
658
+ });
659
+
660
+ it("concurrent read/write operations", async () => {
661
+ const coll = new RvfCollection<TestDoc>("bench_concurrent");
662
+ await coll.insertMany(
663
+ Array.from({ length: 1000 }, (_, i) => ({ name: `user-${i}`, age: i }))
664
+ );
665
+
666
+ const start = performance.now();
667
+
668
+ // Simulate concurrent operations
669
+ await Promise.all([
670
+ coll.find({ age: { $gt: 500 } }).toArray(),
671
+ coll.updateMany({ age: { $lt: 100 } }, { $inc: { age: 1 } }),
672
+ coll.countDocuments({ age: { $gte: 250, $lte: 750 } }),
673
+ coll.find({}).sort({ age: -1 }).limit(10).toArray(),
674
+ coll.distinct("age"),
675
+ ]);
676
+
677
+ const elapsed = performance.now() - start;
678
+ console.log(` 5 concurrent ops (1k docs): ${elapsed.toFixed(1)}ms`);
679
+ expect(elapsed).toBeLessThan(2000);
680
+ });
681
+
682
+ it("multi-tenant isolation performance", async () => {
683
+ enableMultiTenant(true);
684
+ const coll = new RvfCollection<TestDoc>("bench_tenant");
685
+
686
+ // Insert into 10 tenants, 1000 docs each
687
+ const start = performance.now();
688
+ for (let t = 0; t < 10; t++) {
689
+ const tenant = coll.forTenant(`tenant-${t}`);
690
+ await tenant.insertMany(
691
+ Array.from({ length: 1000 }, (_, i) => ({ name: `t${t}-user-${i}`, age: i }))
692
+ );
693
+ }
694
+ const insertElapsed = performance.now() - start;
695
+ console.log(` Multi-tenant insert (10 tenants × 1k): ${insertElapsed.toFixed(1)}ms`);
696
+
697
+ // Query within single tenant should be fast
698
+ const queryStart = performance.now();
699
+ const tenantResults = await coll
700
+ .forTenant("tenant-5")
701
+ .find({ age: { $gt: 500 } })
702
+ .toArray();
703
+ const queryElapsed = performance.now() - queryStart;
704
+ console.log(` Single tenant query (1k docs): ${queryElapsed.toFixed(1)}ms (${tenantResults.length} results)`);
705
+
706
+ expect(tenantResults.length).toBe(499);
707
+ expect(queryElapsed).toBeLessThan(500);
708
+ });
709
+ });