ruflo 3.5.1 → 3.5.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/rvf.manifest.json +295 -0
- package/package.json +16 -2
- package/src/chat-ui/Dockerfile +25 -0
- package/src/chat-ui/patch-mcp-url-safety.sh +28 -0
- package/src/chat-ui/static/chatui/icon-144x144.png +0 -0
- package/src/chat-ui/static/chatui/omni-welcome.gif +0 -0
- package/src/config/config.example.json +76 -0
- package/src/mcp-bridge/Dockerfile +45 -0
- package/src/mcp-bridge/index.js +1668 -0
- package/src/mcp-bridge/mcp-stdio-kernel.js +159 -0
- package/src/mcp-bridge/package.json +17 -0
- package/src/mcp-bridge/test-harness.js +470 -0
- package/src/nginx/Dockerfile +10 -0
- package/src/nginx/nginx.conf +67 -0
- package/src/nginx/static/favicon-dark.svg +4 -0
- package/src/nginx/static/favicon.svg +4 -0
- package/src/nginx/static/icon.svg +5 -0
- package/src/nginx/static/logo.svg +9 -0
- package/src/nginx/static/manifest.json +22 -0
- package/src/nginx/static/welcome.js +184 -0
- package/src/ruvocal/.claude/skills/add-model-descriptions/SKILL.md +73 -0
- package/src/ruvocal/.devcontainer/Dockerfile +9 -0
- package/src/ruvocal/.devcontainer/devcontainer.json +36 -0
- package/src/ruvocal/.dockerignore +13 -0
- package/src/ruvocal/.env +194 -0
- package/src/ruvocal/.env.ci +1 -0
- package/src/ruvocal/.eslintignore +13 -0
- package/src/ruvocal/.eslintrc.cjs +45 -0
- package/src/ruvocal/.github/ISSUE_TEMPLATE/bug-report--chat-ui-.md +43 -0
- package/src/ruvocal/.github/ISSUE_TEMPLATE/config-support.md +9 -0
- package/src/ruvocal/.github/ISSUE_TEMPLATE/feature-request--chat-ui-.md +17 -0
- package/src/ruvocal/.github/ISSUE_TEMPLATE/huggingchat.md +11 -0
- package/src/ruvocal/.github/release.yml +16 -0
- package/src/ruvocal/.github/workflows/build-docs.yml +18 -0
- package/src/ruvocal/.github/workflows/build-image.yml +142 -0
- package/src/ruvocal/.github/workflows/build-pr-docs.yml +20 -0
- package/src/ruvocal/.github/workflows/deploy-dev.yml +63 -0
- package/src/ruvocal/.github/workflows/deploy-prod.yml +78 -0
- package/src/ruvocal/.github/workflows/lint-and-test.yml +84 -0
- package/src/ruvocal/.github/workflows/slugify.yaml +72 -0
- package/src/ruvocal/.github/workflows/trufflehog.yml +17 -0
- package/src/ruvocal/.github/workflows/upload-pr-documentation.yml +16 -0
- package/src/ruvocal/.husky/lint-stage-config.js +4 -0
- package/src/ruvocal/.husky/pre-commit +2 -0
- package/src/ruvocal/.prettierignore +14 -0
- package/src/ruvocal/.prettierrc +7 -0
- package/src/ruvocal/.vscode/launch.json +11 -0
- package/src/ruvocal/.vscode/settings.json +14 -0
- package/src/ruvocal/CLAUDE.md +126 -0
- package/src/ruvocal/Dockerfile +93 -0
- package/src/ruvocal/LICENSE +203 -0
- package/src/ruvocal/PRIVACY.md +41 -0
- package/src/ruvocal/README.md +190 -0
- package/src/ruvocal/chart/Chart.yaml +5 -0
- package/src/ruvocal/chart/env/dev.yaml +260 -0
- package/src/ruvocal/chart/env/prod.yaml +273 -0
- package/src/ruvocal/chart/templates/_helpers.tpl +22 -0
- package/src/ruvocal/chart/templates/config.yaml +10 -0
- package/src/ruvocal/chart/templates/deployment.yaml +81 -0
- package/src/ruvocal/chart/templates/hpa.yaml +45 -0
- package/src/ruvocal/chart/templates/infisical.yaml +24 -0
- package/src/ruvocal/chart/templates/ingress-internal.yaml +32 -0
- package/src/ruvocal/chart/templates/ingress.yaml +32 -0
- package/src/ruvocal/chart/templates/network-policy.yaml +36 -0
- package/src/ruvocal/chart/templates/service-account.yaml +13 -0
- package/src/ruvocal/chart/templates/service-monitor.yaml +17 -0
- package/src/ruvocal/chart/templates/service.yaml +21 -0
- package/src/ruvocal/chart/values.yaml +73 -0
- package/src/ruvocal/docker-compose.yml +21 -0
- package/src/ruvocal/docs/adr/ADR-029-HUGGINGFACE-CHAT-UI-CLOUD-RUN.md +1236 -0
- package/src/ruvocal/docs/adr/ADR-033-RUVECTOR-RUFLO-MCP-INTEGRATION.md +111 -0
- package/src/ruvocal/docs/adr/ADR-034-OPTIONAL-MCP-BACKENDS.md +117 -0
- package/src/ruvocal/docs/adr/ADR-035-MCP-TOOL-GROUPS.md +186 -0
- package/src/ruvocal/docs/adr/ADR-037-AUTOPILOT-CHAT-MODE.md +1500 -0
- package/src/ruvocal/docs/adr/ADR-038-RUVOCAL-FORK.md +286 -0
- package/src/ruvocal/docs/source/_toctree.yml +30 -0
- package/src/ruvocal/docs/source/configuration/common-issues.md +38 -0
- package/src/ruvocal/docs/source/configuration/llm-router.md +105 -0
- package/src/ruvocal/docs/source/configuration/mcp-tools.md +84 -0
- package/src/ruvocal/docs/source/configuration/metrics.md +9 -0
- package/src/ruvocal/docs/source/configuration/open-id.md +57 -0
- package/src/ruvocal/docs/source/configuration/overview.md +89 -0
- package/src/ruvocal/docs/source/configuration/theming.md +20 -0
- package/src/ruvocal/docs/source/developing/architecture.md +48 -0
- package/src/ruvocal/docs/source/index.md +53 -0
- package/src/ruvocal/docs/source/installation/docker.md +43 -0
- package/src/ruvocal/docs/source/installation/helm.md +43 -0
- package/src/ruvocal/docs/source/installation/local.md +62 -0
- package/src/ruvocal/entrypoint.sh +19 -0
- package/src/ruvocal/mcp-bridge/.claude-flow/agents/store.json +27 -0
- package/src/ruvocal/mcp-bridge/.claude-flow/daemon-state.json +130 -0
- package/src/ruvocal/mcp-bridge/.claude-flow/daemon.log +0 -0
- package/src/ruvocal/mcp-bridge/.claude-flow/daemon.pid +1 -0
- package/src/ruvocal/mcp-bridge/.claude-flow/tasks/store.json +21 -0
- package/src/ruvocal/mcp-bridge/.swarm/hnsw.index +0 -0
- package/src/ruvocal/mcp-bridge/.swarm/hnsw.metadata.json +1 -0
- package/src/ruvocal/mcp-bridge/.swarm/memory.db +0 -0
- package/src/ruvocal/mcp-bridge/.swarm/model-router-state.json +14 -0
- package/src/ruvocal/mcp-bridge/.swarm/schema.sql +305 -0
- package/src/ruvocal/mcp-bridge/Dockerfile +45 -0
- package/src/ruvocal/mcp-bridge/cloudbuild.yaml +49 -0
- package/src/ruvocal/mcp-bridge/index.js +1864 -0
- package/src/ruvocal/mcp-bridge/mcp-stdio-kernel.js +159 -0
- package/src/ruvocal/mcp-bridge/package-lock.json +762 -0
- package/src/ruvocal/mcp-bridge/package.json +17 -0
- package/src/ruvocal/mcp-bridge/test-harness.js +470 -0
- package/src/ruvocal/models/add-your-models-here.txt +1 -0
- package/src/ruvocal/package-lock.json +11741 -0
- package/src/ruvocal/package.json +121 -0
- package/src/ruvocal/postcss.config.js +6 -0
- package/src/ruvocal/rvf.manifest.json +204 -0
- package/src/ruvocal/scripts/config.ts +64 -0
- package/src/ruvocal/scripts/generate-welcome.mjs +181 -0
- package/src/ruvocal/scripts/populate.ts +288 -0
- package/src/ruvocal/scripts/samples.txt +194 -0
- package/src/ruvocal/scripts/setups/vitest-setup-client.ts +0 -0
- package/src/ruvocal/scripts/setups/vitest-setup-server.ts +44 -0
- package/src/ruvocal/scripts/updateLocalEnv.ts +48 -0
- package/src/ruvocal/src/ambient.d.ts +7 -0
- package/src/ruvocal/src/app.d.ts +29 -0
- package/src/ruvocal/src/app.html +53 -0
- package/src/ruvocal/src/hooks.server.ts +32 -0
- package/src/ruvocal/src/hooks.ts +6 -0
- package/src/ruvocal/src/lib/APIClient.ts +148 -0
- package/src/ruvocal/src/lib/actions/clickOutside.ts +18 -0
- package/src/ruvocal/src/lib/actions/snapScrollToBottom.ts +346 -0
- package/src/ruvocal/src/lib/buildPrompt.ts +33 -0
- package/src/ruvocal/src/lib/components/AnnouncementBanner.svelte +20 -0
- package/src/ruvocal/src/lib/components/BackgroundGenerationPoller.svelte +168 -0
- package/src/ruvocal/src/lib/components/CodeBlock.svelte +73 -0
- package/src/ruvocal/src/lib/components/CopyToClipBoardBtn.svelte +92 -0
- package/src/ruvocal/src/lib/components/DeleteConversationModal.svelte +75 -0
- package/src/ruvocal/src/lib/components/EditConversationModal.svelte +100 -0
- package/src/ruvocal/src/lib/components/ExpandNavigation.svelte +22 -0
- package/src/ruvocal/src/lib/components/HoverTooltip.svelte +44 -0
- package/src/ruvocal/src/lib/components/HtmlPreviewModal.svelte +143 -0
- package/src/ruvocal/src/lib/components/InfiniteScroll.svelte +50 -0
- package/src/ruvocal/src/lib/components/MobileNav.svelte +300 -0
- package/src/ruvocal/src/lib/components/Modal.svelte +115 -0
- package/src/ruvocal/src/lib/components/ModelCardMetadata.svelte +71 -0
- package/src/ruvocal/src/lib/components/NavConversationItem.svelte +151 -0
- package/src/ruvocal/src/lib/components/NavMenu.svelte +295 -0
- package/src/ruvocal/src/lib/components/Pagination.svelte +97 -0
- package/src/ruvocal/src/lib/components/PaginationArrow.svelte +27 -0
- package/src/ruvocal/src/lib/components/Portal.svelte +24 -0
- package/src/ruvocal/src/lib/components/RetryBtn.svelte +18 -0
- package/src/ruvocal/src/lib/components/RuFloUniverse.svelte +185 -0
- package/src/ruvocal/src/lib/components/ScrollToBottomBtn.svelte +47 -0
- package/src/ruvocal/src/lib/components/ScrollToPreviousBtn.svelte +77 -0
- package/src/ruvocal/src/lib/components/ShareConversationModal.svelte +182 -0
- package/src/ruvocal/src/lib/components/StopGeneratingBtn.svelte +69 -0
- package/src/ruvocal/src/lib/components/SubscribeModal.svelte +87 -0
- package/src/ruvocal/src/lib/components/Switch.svelte +36 -0
- package/src/ruvocal/src/lib/components/SystemPromptModal.svelte +44 -0
- package/src/ruvocal/src/lib/components/Toast.svelte +27 -0
- package/src/ruvocal/src/lib/components/Tooltip.svelte +30 -0
- package/src/ruvocal/src/lib/components/WelcomeModal.svelte +46 -0
- package/src/ruvocal/src/lib/components/chat/Alternatives.svelte +77 -0
- package/src/ruvocal/src/lib/components/chat/BlockWrapper.svelte +72 -0
- package/src/ruvocal/src/lib/components/chat/ChatInput.svelte +490 -0
- package/src/ruvocal/src/lib/components/chat/ChatIntroduction.svelte +123 -0
- package/src/ruvocal/src/lib/components/chat/ChatMessage.svelte +548 -0
- package/src/ruvocal/src/lib/components/chat/ChatWindow.svelte +939 -0
- package/src/ruvocal/src/lib/components/chat/FileDropzone.svelte +92 -0
- package/src/ruvocal/src/lib/components/chat/ImageLightbox.svelte +66 -0
- package/src/ruvocal/src/lib/components/chat/MarkdownBlock.svelte +23 -0
- package/src/ruvocal/src/lib/components/chat/MarkdownRenderer.svelte +69 -0
- package/src/ruvocal/src/lib/components/chat/MarkdownRenderer.svelte.test.ts +58 -0
- package/src/ruvocal/src/lib/components/chat/MessageAvatar.svelte +103 -0
- package/src/ruvocal/src/lib/components/chat/ModelSwitch.svelte +64 -0
- package/src/ruvocal/src/lib/components/chat/OpenReasoningResults.svelte +81 -0
- package/src/ruvocal/src/lib/components/chat/TaskGroup.svelte +88 -0
- package/src/ruvocal/src/lib/components/chat/ToolUpdate.svelte +273 -0
- package/src/ruvocal/src/lib/components/chat/UploadedFile.svelte +253 -0
- package/src/ruvocal/src/lib/components/chat/UrlFetchModal.svelte +203 -0
- package/src/ruvocal/src/lib/components/chat/VoiceRecorder.svelte +214 -0
- package/src/ruvocal/src/lib/components/icons/IconBurger.svelte +20 -0
- package/src/ruvocal/src/lib/components/icons/IconCheap.svelte +20 -0
- package/src/ruvocal/src/lib/components/icons/IconChevron.svelte +24 -0
- package/src/ruvocal/src/lib/components/icons/IconDazzled.svelte +40 -0
- package/src/ruvocal/src/lib/components/icons/IconFast.svelte +20 -0
- package/src/ruvocal/src/lib/components/icons/IconLoading.svelte +22 -0
- package/src/ruvocal/src/lib/components/icons/IconMCP.svelte +28 -0
- package/src/ruvocal/src/lib/components/icons/IconMoon.svelte +21 -0
- package/src/ruvocal/src/lib/components/icons/IconNew.svelte +20 -0
- package/src/ruvocal/src/lib/components/icons/IconOmni.svelte +90 -0
- package/src/ruvocal/src/lib/components/icons/IconPaperclip.svelte +24 -0
- package/src/ruvocal/src/lib/components/icons/IconPro.svelte +37 -0
- package/src/ruvocal/src/lib/components/icons/IconShare.svelte +21 -0
- package/src/ruvocal/src/lib/components/icons/IconSun.svelte +93 -0
- package/src/ruvocal/src/lib/components/icons/Logo.svelte +68 -0
- package/src/ruvocal/src/lib/components/icons/LogoHuggingFaceBorderless.svelte +54 -0
- package/src/ruvocal/src/lib/components/mcp/AddServerForm.svelte +250 -0
- package/src/ruvocal/src/lib/components/mcp/MCPServerManager.svelte +185 -0
- package/src/ruvocal/src/lib/components/mcp/ServerCard.svelte +203 -0
- package/src/ruvocal/src/lib/components/players/AudioPlayer.svelte +82 -0
- package/src/ruvocal/src/lib/components/voice/AudioWaveform.svelte +96 -0
- package/src/ruvocal/src/lib/constants/mcpExamples.ts +135 -0
- package/src/ruvocal/src/lib/constants/mime.ts +11 -0
- package/src/ruvocal/src/lib/constants/pagination.ts +1 -0
- package/src/ruvocal/src/lib/constants/publicSepToken.ts +1 -0
- package/src/ruvocal/src/lib/constants/routerExamples.ts +209 -0
- package/src/ruvocal/src/lib/createShareLink.ts +27 -0
- package/src/ruvocal/src/lib/jobs/refresh-conversation-stats.ts +297 -0
- package/src/ruvocal/src/lib/migrations/lock.ts +56 -0
- package/src/ruvocal/src/lib/migrations/migrations.spec.ts +74 -0
- package/src/ruvocal/src/lib/migrations/migrations.ts +109 -0
- package/src/ruvocal/src/lib/migrations/routines/01-update-search-assistants.ts +50 -0
- package/src/ruvocal/src/lib/migrations/routines/02-update-assistants-models.ts +48 -0
- package/src/ruvocal/src/lib/migrations/routines/04-update-message-updates.ts +151 -0
- package/src/ruvocal/src/lib/migrations/routines/05-update-message-files.ts +56 -0
- package/src/ruvocal/src/lib/migrations/routines/06-trim-message-updates.ts +56 -0
- package/src/ruvocal/src/lib/migrations/routines/08-update-featured-to-review.ts +32 -0
- package/src/ruvocal/src/lib/migrations/routines/09-delete-empty-conversations.spec.ts +214 -0
- package/src/ruvocal/src/lib/migrations/routines/09-delete-empty-conversations.ts +88 -0
- package/src/ruvocal/src/lib/migrations/routines/10-update-reports-assistantid.ts +29 -0
- package/src/ruvocal/src/lib/migrations/routines/index.ts +15 -0
- package/src/ruvocal/src/lib/server/__tests__/conversation-stop-generating.spec.ts +103 -0
- package/src/ruvocal/src/lib/server/abortRegistry.ts +57 -0
- package/src/ruvocal/src/lib/server/abortedGenerations.ts +43 -0
- package/src/ruvocal/src/lib/server/adminToken.ts +62 -0
- package/src/ruvocal/src/lib/server/api/__tests__/conversations-id.spec.ts +296 -0
- package/src/ruvocal/src/lib/server/api/__tests__/conversations-message.spec.ts +216 -0
- package/src/ruvocal/src/lib/server/api/__tests__/conversations.spec.ts +235 -0
- package/src/ruvocal/src/lib/server/api/__tests__/misc.spec.ts +72 -0
- package/src/ruvocal/src/lib/server/api/__tests__/testHelpers.ts +86 -0
- package/src/ruvocal/src/lib/server/api/__tests__/user-reports.spec.ts +78 -0
- package/src/ruvocal/src/lib/server/api/__tests__/user.spec.ts +239 -0
- package/src/ruvocal/src/lib/server/api/types.ts +37 -0
- package/src/ruvocal/src/lib/server/api/utils/requireAuth.ts +22 -0
- package/src/ruvocal/src/lib/server/api/utils/resolveConversation.ts +69 -0
- package/src/ruvocal/src/lib/server/api/utils/resolveModel.ts +27 -0
- package/src/ruvocal/src/lib/server/api/utils/superjsonResponse.ts +15 -0
- package/src/ruvocal/src/lib/server/apiToken.ts +11 -0
- package/src/ruvocal/src/lib/server/auth.ts +554 -0
- package/src/ruvocal/src/lib/server/config.ts +187 -0
- package/src/ruvocal/src/lib/server/conversation.ts +83 -0
- package/src/ruvocal/src/lib/server/database/__tests__/rvf.spec.ts +709 -0
- package/src/ruvocal/src/lib/server/database/postgres.ts +700 -0
- package/src/ruvocal/src/lib/server/database/rvf.ts +1078 -0
- package/src/ruvocal/src/lib/server/database.ts +145 -0
- package/src/ruvocal/src/lib/server/endpoints/document.ts +68 -0
- package/src/ruvocal/src/lib/server/endpoints/endpoints.ts +43 -0
- package/src/ruvocal/src/lib/server/endpoints/images.ts +211 -0
- package/src/ruvocal/src/lib/server/endpoints/openai/endpointOai.ts +266 -0
- package/src/ruvocal/src/lib/server/endpoints/openai/openAIChatToTextGenerationStream.ts +212 -0
- package/src/ruvocal/src/lib/server/endpoints/openai/openAICompletionToTextGenerationStream.ts +32 -0
- package/src/ruvocal/src/lib/server/endpoints/preprocessMessages.ts +61 -0
- package/src/ruvocal/src/lib/server/exitHandler.ts +59 -0
- package/src/ruvocal/src/lib/server/files/downloadFile.ts +34 -0
- package/src/ruvocal/src/lib/server/files/uploadFile.ts +29 -0
- package/src/ruvocal/src/lib/server/findRepoRoot.ts +13 -0
- package/src/ruvocal/src/lib/server/fonts/Inter-Black.ttf +0 -0
- package/src/ruvocal/src/lib/server/fonts/Inter-Bold.ttf +0 -0
- package/src/ruvocal/src/lib/server/fonts/Inter-ExtraBold.ttf +0 -0
- package/src/ruvocal/src/lib/server/fonts/Inter-ExtraLight.ttf +0 -0
- package/src/ruvocal/src/lib/server/fonts/Inter-Light.ttf +0 -0
- package/src/ruvocal/src/lib/server/fonts/Inter-Medium.ttf +0 -0
- package/src/ruvocal/src/lib/server/fonts/Inter-Regular.ttf +0 -0
- package/src/ruvocal/src/lib/server/fonts/Inter-SemiBold.ttf +0 -0
- package/src/ruvocal/src/lib/server/fonts/Inter-Thin.ttf +0 -0
- package/src/ruvocal/src/lib/server/generateFromDefaultEndpoint.ts +46 -0
- package/src/ruvocal/src/lib/server/hooks/error.ts +37 -0
- package/src/ruvocal/src/lib/server/hooks/fetch.ts +22 -0
- package/src/ruvocal/src/lib/server/hooks/handle.ts +250 -0
- package/src/ruvocal/src/lib/server/hooks/init.ts +51 -0
- package/src/ruvocal/src/lib/server/isURLLocal.spec.ts +31 -0
- package/src/ruvocal/src/lib/server/isURLLocal.ts +74 -0
- package/src/ruvocal/src/lib/server/logger.ts +42 -0
- package/src/ruvocal/src/lib/server/mcp/clientPool.ts +70 -0
- package/src/ruvocal/src/lib/server/mcp/hf.ts +32 -0
- package/src/ruvocal/src/lib/server/mcp/httpClient.ts +122 -0
- package/src/ruvocal/src/lib/server/mcp/registry.ts +76 -0
- package/src/ruvocal/src/lib/server/mcp/tools.ts +196 -0
- package/src/ruvocal/src/lib/server/metrics.ts +255 -0
- package/src/ruvocal/src/lib/server/models.ts +518 -0
- package/src/ruvocal/src/lib/server/requestContext.ts +55 -0
- package/src/ruvocal/src/lib/server/router/arch.ts +230 -0
- package/src/ruvocal/src/lib/server/router/endpoint.ts +316 -0
- package/src/ruvocal/src/lib/server/router/multimodal.ts +28 -0
- package/src/ruvocal/src/lib/server/router/policy.ts +49 -0
- package/src/ruvocal/src/lib/server/router/toolsRoute.ts +51 -0
- package/src/ruvocal/src/lib/server/router/types.ts +21 -0
- package/src/ruvocal/src/lib/server/sendSlack.ts +23 -0
- package/src/ruvocal/src/lib/server/textGeneration/generate.ts +258 -0
- package/src/ruvocal/src/lib/server/textGeneration/index.ts +95 -0
- package/src/ruvocal/src/lib/server/textGeneration/mcp/fileRefs.ts +155 -0
- package/src/ruvocal/src/lib/server/textGeneration/mcp/routerResolution.ts +108 -0
- package/src/ruvocal/src/lib/server/textGeneration/mcp/runMcpFlow.ts +822 -0
- package/src/ruvocal/src/lib/server/textGeneration/mcp/toolInvocation.ts +349 -0
- package/src/ruvocal/src/lib/server/textGeneration/reasoning.ts +23 -0
- package/src/ruvocal/src/lib/server/textGeneration/title.ts +83 -0
- package/src/ruvocal/src/lib/server/textGeneration/types.ts +26 -0
- package/src/ruvocal/src/lib/server/textGeneration/utils/prepareFiles.ts +88 -0
- package/src/ruvocal/src/lib/server/textGeneration/utils/routing.ts +21 -0
- package/src/ruvocal/src/lib/server/textGeneration/utils/toolPrompt.ts +49 -0
- package/src/ruvocal/src/lib/server/urlSafety.ts +72 -0
- package/src/ruvocal/src/lib/server/usageLimits.ts +30 -0
- package/src/ruvocal/src/lib/stores/autopilotStore.svelte.ts +175 -0
- package/src/ruvocal/src/lib/stores/backgroundGenerations.svelte.ts +32 -0
- package/src/ruvocal/src/lib/stores/backgroundGenerations.ts +1 -0
- package/src/ruvocal/src/lib/stores/errors.ts +9 -0
- package/src/ruvocal/src/lib/stores/isAborted.ts +3 -0
- package/src/ruvocal/src/lib/stores/isPro.ts +4 -0
- package/src/ruvocal/src/lib/stores/loading.ts +3 -0
- package/src/ruvocal/src/lib/stores/mcpServers.ts +345 -0
- package/src/ruvocal/src/lib/stores/pendingChatInput.ts +3 -0
- package/src/ruvocal/src/lib/stores/pendingMessage.ts +9 -0
- package/src/ruvocal/src/lib/stores/settings.ts +182 -0
- package/src/ruvocal/src/lib/stores/shareModal.ts +13 -0
- package/src/ruvocal/src/lib/stores/titleUpdate.ts +8 -0
- package/src/ruvocal/src/lib/switchTheme.ts +124 -0
- package/src/ruvocal/src/lib/types/AbortedGeneration.ts +8 -0
- package/src/ruvocal/src/lib/types/Assistant.ts +31 -0
- package/src/ruvocal/src/lib/types/AssistantStats.ts +11 -0
- package/src/ruvocal/src/lib/types/ConfigKey.ts +4 -0
- package/src/ruvocal/src/lib/types/ConvSidebar.ts +9 -0
- package/src/ruvocal/src/lib/types/Conversation.ts +27 -0
- package/src/ruvocal/src/lib/types/ConversationStats.ts +13 -0
- package/src/ruvocal/src/lib/types/Message.ts +41 -0
- package/src/ruvocal/src/lib/types/MessageEvent.ts +10 -0
- package/src/ruvocal/src/lib/types/MessageUpdate.ts +139 -0
- package/src/ruvocal/src/lib/types/MigrationResult.ts +7 -0
- package/src/ruvocal/src/lib/types/Model.ts +23 -0
- package/src/ruvocal/src/lib/types/Report.ts +12 -0
- package/src/ruvocal/src/lib/types/Review.ts +6 -0
- package/src/ruvocal/src/lib/types/Semaphore.ts +19 -0
- package/src/ruvocal/src/lib/types/Session.ts +22 -0
- package/src/ruvocal/src/lib/types/Settings.ts +86 -0
- package/src/ruvocal/src/lib/types/SharedConversation.ts +9 -0
- package/src/ruvocal/src/lib/types/Template.ts +6 -0
- package/src/ruvocal/src/lib/types/Timestamps.ts +4 -0
- package/src/ruvocal/src/lib/types/TokenCache.ts +6 -0
- package/src/ruvocal/src/lib/types/Tool.ts +74 -0
- package/src/ruvocal/src/lib/types/UrlDependency.ts +5 -0
- package/src/ruvocal/src/lib/types/User.ts +14 -0
- package/src/ruvocal/src/lib/utils/PublicConfig.svelte.ts +75 -0
- package/src/ruvocal/src/lib/utils/auth.ts +17 -0
- package/src/ruvocal/src/lib/utils/chunk.ts +33 -0
- package/src/ruvocal/src/lib/utils/cookiesAreEnabled.ts +13 -0
- package/src/ruvocal/src/lib/utils/debounce.ts +17 -0
- package/src/ruvocal/src/lib/utils/deepestChild.ts +6 -0
- package/src/ruvocal/src/lib/utils/favicon.ts +21 -0
- package/src/ruvocal/src/lib/utils/fetchJSON.ts +23 -0
- package/src/ruvocal/src/lib/utils/file2base64.ts +14 -0
- package/src/ruvocal/src/lib/utils/formatUserCount.ts +37 -0
- package/src/ruvocal/src/lib/utils/generationState.spec.ts +75 -0
- package/src/ruvocal/src/lib/utils/generationState.ts +26 -0
- package/src/ruvocal/src/lib/utils/getHref.ts +41 -0
- package/src/ruvocal/src/lib/utils/getReturnFromGenerator.ts +7 -0
- package/src/ruvocal/src/lib/utils/haptics.ts +64 -0
- package/src/ruvocal/src/lib/utils/hashConv.ts +12 -0
- package/src/ruvocal/src/lib/utils/hf.ts +17 -0
- package/src/ruvocal/src/lib/utils/isDesktop.ts +7 -0
- package/src/ruvocal/src/lib/utils/isUrl.ts +8 -0
- package/src/ruvocal/src/lib/utils/isVirtualKeyboard.ts +16 -0
- package/src/ruvocal/src/lib/utils/loadAttachmentsFromUrls.ts +115 -0
- package/src/ruvocal/src/lib/utils/marked.spec.ts +96 -0
- package/src/ruvocal/src/lib/utils/marked.ts +531 -0
- package/src/ruvocal/src/lib/utils/mcpValidation.ts +147 -0
- package/src/ruvocal/src/lib/utils/mergeAsyncGenerators.ts +38 -0
- package/src/ruvocal/src/lib/utils/messageUpdates.spec.ts +262 -0
- package/src/ruvocal/src/lib/utils/messageUpdates.ts +324 -0
- package/src/ruvocal/src/lib/utils/mime.ts +56 -0
- package/src/ruvocal/src/lib/utils/models.ts +14 -0
- package/src/ruvocal/src/lib/utils/parseBlocks.ts +120 -0
- package/src/ruvocal/src/lib/utils/parseIncompleteMarkdown.ts +644 -0
- package/src/ruvocal/src/lib/utils/parseStringToList.ts +10 -0
- package/src/ruvocal/src/lib/utils/randomUuid.ts +14 -0
- package/src/ruvocal/src/lib/utils/searchTokens.ts +33 -0
- package/src/ruvocal/src/lib/utils/sha256.ts +7 -0
- package/src/ruvocal/src/lib/utils/stringifyError.ts +12 -0
- package/src/ruvocal/src/lib/utils/sum.ts +3 -0
- package/src/ruvocal/src/lib/utils/template.spec.ts +59 -0
- package/src/ruvocal/src/lib/utils/template.ts +53 -0
- package/src/ruvocal/src/lib/utils/timeout.ts +9 -0
- package/src/ruvocal/src/lib/utils/toolProgress.spec.ts +46 -0
- package/src/ruvocal/src/lib/utils/toolProgress.ts +11 -0
- package/src/ruvocal/src/lib/utils/tree/addChildren.spec.ts +102 -0
- package/src/ruvocal/src/lib/utils/tree/addChildren.ts +48 -0
- package/src/ruvocal/src/lib/utils/tree/addSibling.spec.ts +81 -0
- package/src/ruvocal/src/lib/utils/tree/addSibling.ts +41 -0
- package/src/ruvocal/src/lib/utils/tree/buildSubtree.spec.ts +110 -0
- package/src/ruvocal/src/lib/utils/tree/buildSubtree.ts +24 -0
- package/src/ruvocal/src/lib/utils/tree/convertLegacyConversation.spec.ts +31 -0
- package/src/ruvocal/src/lib/utils/tree/convertLegacyConversation.ts +36 -0
- package/src/ruvocal/src/lib/utils/tree/isMessageId.spec.ts +15 -0
- package/src/ruvocal/src/lib/utils/tree/isMessageId.ts +5 -0
- package/src/ruvocal/src/lib/utils/tree/tree.d.ts +14 -0
- package/src/ruvocal/src/lib/utils/tree/treeHelpers.spec.ts +167 -0
- package/src/ruvocal/src/lib/utils/updates.ts +39 -0
- package/src/ruvocal/src/lib/utils/urlParams.ts +13 -0
- package/src/ruvocal/src/lib/workers/autopilotWorker.ts +221 -0
- package/src/ruvocal/src/lib/workers/detailFetchWorker.ts +100 -0
- package/src/ruvocal/src/lib/workers/markdownWorker.ts +61 -0
- package/src/ruvocal/src/routes/+error.svelte +20 -0
- package/src/ruvocal/src/routes/+layout.svelte +324 -0
- package/src/ruvocal/src/routes/+layout.ts +91 -0
- package/src/ruvocal/src/routes/+page.svelte +168 -0
- package/src/ruvocal/src/routes/.well-known/oauth-cimd/+server.ts +37 -0
- package/src/ruvocal/src/routes/__debug/openai/+server.ts +21 -0
- package/src/ruvocal/src/routes/admin/export/+server.ts +159 -0
- package/src/ruvocal/src/routes/admin/stats/compute/+server.ts +16 -0
- package/src/ruvocal/src/routes/api/conversation/[id]/+server.ts +40 -0
- package/src/ruvocal/src/routes/api/conversation/[id]/message/[messageId]/+server.ts +42 -0
- package/src/ruvocal/src/routes/api/conversations/+server.ts +48 -0
- package/src/ruvocal/src/routes/api/fetch-url/+server.ts +147 -0
- package/src/ruvocal/src/routes/api/mcp/health/+server.ts +292 -0
- package/src/ruvocal/src/routes/api/mcp/servers/+server.ts +32 -0
- package/src/ruvocal/src/routes/api/models/+server.ts +25 -0
- package/src/ruvocal/src/routes/api/transcribe/+server.ts +104 -0
- package/src/ruvocal/src/routes/api/user/+server.ts +15 -0
- package/src/ruvocal/src/routes/api/user/validate-token/+server.ts +20 -0
- package/src/ruvocal/src/routes/api/v2/conversations/+server.ts +48 -0
- package/src/ruvocal/src/routes/api/v2/conversations/[id]/+server.ts +94 -0
- package/src/ruvocal/src/routes/api/v2/conversations/[id]/message/[messageId]/+server.ts +43 -0
- package/src/ruvocal/src/routes/api/v2/conversations/import-share/+server.ts +23 -0
- package/src/ruvocal/src/routes/api/v2/debug/config/+server.ts +16 -0
- package/src/ruvocal/src/routes/api/v2/debug/refresh/+server.ts +30 -0
- package/src/ruvocal/src/routes/api/v2/export/+server.ts +196 -0
- package/src/ruvocal/src/routes/api/v2/feature-flags/+server.ts +14 -0
- package/src/ruvocal/src/routes/api/v2/models/+server.ts +38 -0
- package/src/ruvocal/src/routes/api/v2/models/[namespace]/+server.ts +8 -0
- package/src/ruvocal/src/routes/api/v2/models/[namespace]/[model]/+server.ts +8 -0
- package/src/ruvocal/src/routes/api/v2/models/[namespace]/[model]/subscribe/+server.ts +28 -0
- package/src/ruvocal/src/routes/api/v2/models/[namespace]/subscribe/+server.ts +28 -0
- package/src/ruvocal/src/routes/api/v2/models/old/+server.ts +7 -0
- package/src/ruvocal/src/routes/api/v2/models/refresh/+server.ts +33 -0
- package/src/ruvocal/src/routes/api/v2/public-config/+server.ts +7 -0
- package/src/ruvocal/src/routes/api/v2/user/+server.ts +17 -0
- package/src/ruvocal/src/routes/api/v2/user/billing-orgs/+server.ts +73 -0
- package/src/ruvocal/src/routes/api/v2/user/reports/+server.ts +17 -0
- package/src/ruvocal/src/routes/api/v2/user/settings/+server.ts +103 -0
- package/src/ruvocal/src/routes/conversation/+server.ts +115 -0
- package/src/ruvocal/src/routes/conversation/[id]/+page.svelte +582 -0
- package/src/ruvocal/src/routes/conversation/[id]/+page.ts +60 -0
- package/src/ruvocal/src/routes/conversation/[id]/+server.ts +736 -0
- package/src/ruvocal/src/routes/conversation/[id]/message/[messageId]/prompt/+server.ts +66 -0
- package/src/ruvocal/src/routes/conversation/[id]/output/[sha256]/+server.ts +58 -0
- package/src/ruvocal/src/routes/conversation/[id]/share/+server.ts +69 -0
- package/src/ruvocal/src/routes/conversation/[id]/stop-generating/+server.ts +35 -0
- package/src/ruvocal/src/routes/healthcheck/+server.ts +3 -0
- package/src/ruvocal/src/routes/login/+server.ts +5 -0
- package/src/ruvocal/src/routes/login/callback/+server.ts +103 -0
- package/src/ruvocal/src/routes/login/callback/updateUser.spec.ts +157 -0
- package/src/ruvocal/src/routes/login/callback/updateUser.ts +215 -0
- package/src/ruvocal/src/routes/logout/+server.ts +18 -0
- package/src/ruvocal/src/routes/metrics/+server.ts +18 -0
- package/src/ruvocal/src/routes/models/+page.svelte +233 -0
- package/src/ruvocal/src/routes/models/[...model]/+page.svelte +161 -0
- package/src/ruvocal/src/routes/models/[...model]/+page.ts +14 -0
- package/src/ruvocal/src/routes/models/[...model]/thumbnail.png/+server.ts +64 -0
- package/src/ruvocal/src/routes/models/[...model]/thumbnail.png/ModelThumbnail.svelte +28 -0
- package/src/ruvocal/src/routes/privacy/+page.svelte +11 -0
- package/src/ruvocal/src/routes/r/[id]/+page.ts +34 -0
- package/src/ruvocal/src/routes/settings/(nav)/+layout.svelte +282 -0
- package/src/ruvocal/src/routes/settings/(nav)/+layout.ts +1 -0
- package/src/ruvocal/src/routes/settings/(nav)/+page.svelte +0 -0
- package/src/ruvocal/src/routes/settings/(nav)/+server.ts +53 -0
- package/src/ruvocal/src/routes/settings/(nav)/[...model]/+page.svelte +464 -0
- package/src/ruvocal/src/routes/settings/(nav)/[...model]/+page.ts +14 -0
- package/src/ruvocal/src/routes/settings/(nav)/application/+page.svelte +362 -0
- package/src/ruvocal/src/routes/settings/+layout.svelte +40 -0
- package/src/ruvocal/src/styles/highlight-js.css +195 -0
- package/src/ruvocal/src/styles/main.css +144 -0
- package/src/ruvocal/static/chatui/apple-touch-icon.png +0 -0
- package/src/ruvocal/static/chatui/favicon-dark.svg +3 -0
- package/src/ruvocal/static/chatui/favicon-dev.svg +3 -0
- package/src/ruvocal/static/chatui/favicon.ico +0 -0
- package/src/ruvocal/static/chatui/favicon.svg +3 -0
- package/src/ruvocal/static/chatui/icon-128x128.png +0 -0
- package/src/ruvocal/static/chatui/icon-144x144.png +0 -0
- package/src/ruvocal/static/chatui/icon-192x192.png +0 -0
- package/src/ruvocal/static/chatui/icon-256x256.png +0 -0
- package/src/ruvocal/static/chatui/icon-36x36.png +0 -0
- package/src/ruvocal/static/chatui/icon-48x48.png +0 -0
- package/src/ruvocal/static/chatui/icon-512x512.png +0 -0
- package/src/ruvocal/static/chatui/icon-72x72.png +0 -0
- package/src/ruvocal/static/chatui/icon-96x96.png +0 -0
- package/src/ruvocal/static/chatui/icon.svg +3 -0
- package/src/ruvocal/static/chatui/logo.svg +7 -0
- package/src/ruvocal/static/chatui/manifest.json +54 -0
- package/src/ruvocal/static/chatui/omni-welcome.gif +0 -0
- package/src/ruvocal/static/chatui/omni-welcome.png +0 -0
- package/src/ruvocal/static/chatui/welcome.js +184 -0
- package/src/ruvocal/static/chatui/welcome.svg +1 -0
- package/src/ruvocal/static/huggingchat/apple-touch-icon.png +0 -0
- package/src/ruvocal/static/huggingchat/assistants-thumbnail.png +0 -0
- package/src/ruvocal/static/huggingchat/castle-example.jpg +0 -0
- package/src/ruvocal/static/huggingchat/favicon-dark.svg +4 -0
- package/src/ruvocal/static/huggingchat/favicon-dev.svg +4 -0
- package/src/ruvocal/static/huggingchat/favicon.ico +0 -0
- package/src/ruvocal/static/huggingchat/favicon.svg +4 -0
- package/src/ruvocal/static/huggingchat/fulltext-logo.svg +2 -0
- package/src/ruvocal/static/huggingchat/icon-128x128.png +0 -0
- package/src/ruvocal/static/huggingchat/icon-144x144.png +0 -0
- package/src/ruvocal/static/huggingchat/icon-192x192.png +0 -0
- package/src/ruvocal/static/huggingchat/icon-256x256.png +0 -0
- package/src/ruvocal/static/huggingchat/icon-36x36.png +0 -0
- package/src/ruvocal/static/huggingchat/icon-48x48.png +0 -0
- package/src/ruvocal/static/huggingchat/icon-512x512.png +0 -0
- package/src/ruvocal/static/huggingchat/icon-72x72.png +0 -0
- package/src/ruvocal/static/huggingchat/icon-96x96.png +0 -0
- package/src/ruvocal/static/huggingchat/icon.svg +4 -0
- package/src/ruvocal/static/huggingchat/logo.svg +4 -0
- package/src/ruvocal/static/huggingchat/manifest.json +54 -0
- package/src/ruvocal/static/huggingchat/omni-welcome.gif +0 -0
- package/src/ruvocal/static/huggingchat/routes.chat.json +226 -0
- package/src/ruvocal/static/huggingchat/thumbnail.png +0 -0
- package/src/ruvocal/static/huggingchat/tools-thumbnail.png +0 -0
- package/src/ruvocal/static/robots.txt +10 -0
- package/src/ruvocal/stub/@reflink/reflink/index.js +0 -0
- package/src/ruvocal/stub/@reflink/reflink/package.json +5 -0
- package/src/ruvocal/svelte.config.js +53 -0
- package/src/ruvocal/tailwind.config.cjs +30 -0
- package/src/ruvocal/tsconfig.json +19 -0
- package/src/ruvocal/vite.config.ts +87 -0
- package/src/scripts/deploy.sh +116 -0
- package/src/scripts/generate-config.js +245 -0
- package/src/scripts/generate-welcome.js +187 -0
- package/src/scripts/package-rvf.sh +116 -0
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
export async function sha256(input: string): Promise<string> {
|
|
2
|
+
const utf8 = new TextEncoder().encode(input);
|
|
3
|
+
const hashBuffer = await crypto.subtle.digest("SHA-256", utf8);
|
|
4
|
+
const hashArray = Array.from(new Uint8Array(hashBuffer));
|
|
5
|
+
const hashHex = hashArray.map((bytes) => bytes.toString(16).padStart(2, "0")).join("");
|
|
6
|
+
return hashHex;
|
|
7
|
+
}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
/** Takes an unknown error and attempts to convert it to a string */
|
|
2
|
+
export function stringifyError(error: unknown): string {
|
|
3
|
+
if (error instanceof Error) return error.message;
|
|
4
|
+
if (typeof error === "string") return error;
|
|
5
|
+
if (typeof error === "object" && error !== null) {
|
|
6
|
+
// try a few common properties
|
|
7
|
+
if ("message" in error && typeof error.message === "string") return error.message;
|
|
8
|
+
if ("body" in error && typeof error.body === "string") return error.body;
|
|
9
|
+
if ("name" in error && typeof error.name === "string") return error.name;
|
|
10
|
+
}
|
|
11
|
+
return "Unknown error";
|
|
12
|
+
}
|
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
import { describe, test, expect } from "vitest";
|
|
2
|
+
import { compileTemplate } from "./template";
|
|
3
|
+
|
|
4
|
+
// Test data for simple templates
|
|
5
|
+
const modelData = {
|
|
6
|
+
preprompt: "Hello",
|
|
7
|
+
};
|
|
8
|
+
|
|
9
|
+
const simpleTemplate = "Test: {{preprompt}} and {{foo}}";
|
|
10
|
+
|
|
11
|
+
// Additional realistic test data for Llama 70B templates
|
|
12
|
+
const messages = [
|
|
13
|
+
{ from: "user", content: "Hello there" },
|
|
14
|
+
{ from: "assistant", content: "Hi, how can I help?" },
|
|
15
|
+
];
|
|
16
|
+
|
|
17
|
+
// Handlebars Llama 70B Template
|
|
18
|
+
const llama70bTemplateHB = `<s>{{#if preprompt}}Source: system\n\n{{preprompt}}<step>{{/if}}{{#each messages}}{{#ifUser}}Source: user\n\n{{content}}<step>{{/ifUser}}{{#ifAssistant}}Source: assistant\n\n{{content}}<step>{{/ifAssistant}}{{/each}}Source: assistant\nDestination: user\n\n`;
|
|
19
|
+
|
|
20
|
+
// Expected output for Handlebars Llama 70B Template
|
|
21
|
+
const expectedHB =
|
|
22
|
+
"<s>Source: system\n\nSystem Message<step>Source: user\n\nHello there<step>Source: assistant\n\nHi, how can I help?<step>Source: assistant\nDestination: user\n\n";
|
|
23
|
+
|
|
24
|
+
// Jinja Llama 70B Template
|
|
25
|
+
const llama70bTemplateJinja = `<s>{% if preprompt %}Source: system\n\n{{ preprompt }}<step>{% endif %}{% for message in messages %}{% if message.from == 'user' %}Source: user\n\n{{ message.content }}<step>{% elif message.from == 'assistant' %}Source: assistant\n\n{{ message.content }}<step>{% endif %}{% endfor %}Source: assistant\nDestination: user\n\n`;
|
|
26
|
+
|
|
27
|
+
// Expected output for Jinja Llama 70B Template
|
|
28
|
+
const expectedJinja =
|
|
29
|
+
"<s>Source: system\n\nSystem Message<step>Source: user\n\nHello there<step>Source: assistant\n\nHi, how can I help?<step>Source: assistant\nDestination: user\n\n";
|
|
30
|
+
|
|
31
|
+
describe("Template Engine Rendering", () => {
|
|
32
|
+
test("should render using Handlebars fallback when no templateEngine is specified", () => {
|
|
33
|
+
const render = compileTemplate(simpleTemplate, modelData);
|
|
34
|
+
const result = render({ foo: "World" });
|
|
35
|
+
expect(result).toBe("Test: Hello and World");
|
|
36
|
+
});
|
|
37
|
+
|
|
38
|
+
test('should render using Jinja when templateEngine is set to "jinja"', () => {
|
|
39
|
+
const render = compileTemplate(simpleTemplate, modelData);
|
|
40
|
+
const result = render({ foo: "World" });
|
|
41
|
+
expect(result).toBe("Test: Hello and World");
|
|
42
|
+
});
|
|
43
|
+
|
|
44
|
+
// Realistic Llama 70B template tests
|
|
45
|
+
test("should render realistic Llama 70B template using Handlebars", () => {
|
|
46
|
+
const render = compileTemplate(llama70bTemplateHB, { preprompt: "System Message" });
|
|
47
|
+
const result = render({ messages });
|
|
48
|
+
expect(result).toBe(expectedHB);
|
|
49
|
+
});
|
|
50
|
+
|
|
51
|
+
test("should render realistic Llama 70B template using Jinja", () => {
|
|
52
|
+
const render = compileTemplate(llama70bTemplateJinja, {
|
|
53
|
+
preprompt: "System Message",
|
|
54
|
+
});
|
|
55
|
+
const result = render({ messages });
|
|
56
|
+
// Trim both outputs to account for whitespace differences in Jinja engine
|
|
57
|
+
expect(result.trim()).toBe(expectedJinja.trim());
|
|
58
|
+
});
|
|
59
|
+
});
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
import type { Message } from "$lib/types/Message";
|
|
2
|
+
import Handlebars from "handlebars";
|
|
3
|
+
import { Template } from "@huggingface/jinja";
|
|
4
|
+
import { logger } from "$lib/server/logger";
|
|
5
|
+
|
|
6
|
+
// Register Handlebars helpers
|
|
7
|
+
Handlebars.registerHelper("ifUser", function (this: Pick<Message, "from" | "content">, options) {
|
|
8
|
+
if (this.from == "user") return options.fn(this);
|
|
9
|
+
});
|
|
10
|
+
|
|
11
|
+
Handlebars.registerHelper(
|
|
12
|
+
"ifAssistant",
|
|
13
|
+
function (this: Pick<Message, "from" | "content">, options) {
|
|
14
|
+
if (this.from == "assistant") return options.fn(this);
|
|
15
|
+
}
|
|
16
|
+
);
|
|
17
|
+
|
|
18
|
+
// Updated compileTemplate to try Jinja and fallback to Handlebars if Jinja fails
|
|
19
|
+
export function compileTemplate<T>(
|
|
20
|
+
input: string,
|
|
21
|
+
model: { preprompt: string; templateEngine?: string }
|
|
22
|
+
) {
|
|
23
|
+
let jinjaTemplate: Template | undefined;
|
|
24
|
+
try {
|
|
25
|
+
// Try to compile with Jinja
|
|
26
|
+
jinjaTemplate = new Template(input);
|
|
27
|
+
} catch (e) {
|
|
28
|
+
// logger.error(e, "Could not compile with Jinja");
|
|
29
|
+
// Could not compile with Jinja
|
|
30
|
+
jinjaTemplate = undefined;
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
const hbTemplate = Handlebars.compile<T>(input, {
|
|
34
|
+
knownHelpers: { ifUser: true, ifAssistant: true },
|
|
35
|
+
knownHelpersOnly: true,
|
|
36
|
+
noEscape: true,
|
|
37
|
+
strict: true,
|
|
38
|
+
preventIndent: true,
|
|
39
|
+
});
|
|
40
|
+
|
|
41
|
+
return function render(inputs: T) {
|
|
42
|
+
if (jinjaTemplate) {
|
|
43
|
+
try {
|
|
44
|
+
return jinjaTemplate.render({ ...model, ...inputs });
|
|
45
|
+
} catch (e) {
|
|
46
|
+
logger.error(e, "Could not render with Jinja");
|
|
47
|
+
// Fallback to Handlebars if Jinja rendering fails
|
|
48
|
+
return hbTemplate({ ...model, ...inputs });
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
return hbTemplate({ ...model, ...inputs });
|
|
52
|
+
};
|
|
53
|
+
}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
export const timeout = <T>(prom: Promise<T>, time: number): Promise<T> => {
|
|
2
|
+
let timer: NodeJS.Timeout;
|
|
3
|
+
return Promise.race([
|
|
4
|
+
prom,
|
|
5
|
+
new Promise<T>((_, reject) => {
|
|
6
|
+
timer = setTimeout(() => reject(new Error(`Timeout after ${time / 1000} seconds`)), time);
|
|
7
|
+
}),
|
|
8
|
+
]).finally(() => clearTimeout(timer));
|
|
9
|
+
};
|
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
import { describe, expect, test } from "vitest";
|
|
2
|
+
|
|
3
|
+
import { MessageToolUpdateType, MessageUpdateType } from "$lib/types/MessageUpdate";
|
|
4
|
+
import { formatToolProgressLabel } from "./toolProgress";
|
|
5
|
+
|
|
6
|
+
describe("formatToolProgressLabel", () => {
|
|
7
|
+
test("returns empty string when progress is missing", () => {
|
|
8
|
+
expect(formatToolProgressLabel(undefined)).toBe("");
|
|
9
|
+
});
|
|
10
|
+
|
|
11
|
+
test("formats progress with message", () => {
|
|
12
|
+
expect(
|
|
13
|
+
formatToolProgressLabel({
|
|
14
|
+
type: MessageUpdateType.Tool,
|
|
15
|
+
subtype: MessageToolUpdateType.Progress,
|
|
16
|
+
uuid: "tool-1",
|
|
17
|
+
progress: 3,
|
|
18
|
+
total: 10,
|
|
19
|
+
message: "Indexing",
|
|
20
|
+
})
|
|
21
|
+
).toBe("Indexing (3/10)");
|
|
22
|
+
});
|
|
23
|
+
|
|
24
|
+
test("formats progress without message", () => {
|
|
25
|
+
expect(
|
|
26
|
+
formatToolProgressLabel({
|
|
27
|
+
type: MessageUpdateType.Tool,
|
|
28
|
+
subtype: MessageToolUpdateType.Progress,
|
|
29
|
+
uuid: "tool-2",
|
|
30
|
+
progress: 7,
|
|
31
|
+
})
|
|
32
|
+
).toBe("Progress: 7");
|
|
33
|
+
});
|
|
34
|
+
|
|
35
|
+
test("formats progress with message and no total", () => {
|
|
36
|
+
expect(
|
|
37
|
+
formatToolProgressLabel({
|
|
38
|
+
type: MessageUpdateType.Tool,
|
|
39
|
+
subtype: MessageToolUpdateType.Progress,
|
|
40
|
+
uuid: "tool-3",
|
|
41
|
+
progress: 12,
|
|
42
|
+
message: "ZeroGPU Initializing xxx",
|
|
43
|
+
})
|
|
44
|
+
).toBe("ZeroGPU Initializing xxx (12)");
|
|
45
|
+
});
|
|
46
|
+
});
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import type { MessageToolProgressUpdate } from "$lib/types/MessageUpdate";
|
|
2
|
+
|
|
3
|
+
export function formatToolProgressLabel(progress?: MessageToolProgressUpdate): string {
|
|
4
|
+
if (!progress) return "";
|
|
5
|
+
const total = typeof progress.total === "number" ? `/${progress.total}` : "";
|
|
6
|
+
const value = `${progress.progress}${total}`;
|
|
7
|
+
if (progress.message && progress.message.trim().length > 0) {
|
|
8
|
+
return `${progress.message} (${value})`;
|
|
9
|
+
}
|
|
10
|
+
return `Progress: ${value}`;
|
|
11
|
+
}
|
|
@@ -0,0 +1,102 @@
|
|
|
1
|
+
import { collections } from "$lib/server/database";
|
|
2
|
+
import { ObjectId } from "mongodb";
|
|
3
|
+
import { describe, expect, it } from "vitest";
|
|
4
|
+
|
|
5
|
+
import { insertLegacyConversation, insertSideBranchesConversation } from "./treeHelpers.spec";
|
|
6
|
+
import { addChildren } from "./addChildren";
|
|
7
|
+
import type { Message } from "$lib/types/Message";
|
|
8
|
+
|
|
9
|
+
const newMessage: Omit<Message, "id"> = {
|
|
10
|
+
content: "new message",
|
|
11
|
+
from: "user",
|
|
12
|
+
};
|
|
13
|
+
|
|
14
|
+
Object.freeze(newMessage);
|
|
15
|
+
|
|
16
|
+
describe("addChildren", async () => {
|
|
17
|
+
it("should let you append on legacy conversations", async () => {
|
|
18
|
+
const convId = await insertLegacyConversation();
|
|
19
|
+
const conv = await collections.conversations.findOne({ _id: new ObjectId(convId) });
|
|
20
|
+
if (!conv) throw new Error("Conversation not found");
|
|
21
|
+
|
|
22
|
+
const convLength = conv.messages.length;
|
|
23
|
+
|
|
24
|
+
addChildren(conv, newMessage, conv.messages[conv.messages.length - 1].id);
|
|
25
|
+
expect(conv.messages.length).toEqual(convLength + 1);
|
|
26
|
+
});
|
|
27
|
+
it("should not let you create branches on legacy conversations", async () => {
|
|
28
|
+
const convId = await insertLegacyConversation();
|
|
29
|
+
const conv = await collections.conversations.findOne({ _id: new ObjectId(convId) });
|
|
30
|
+
if (!conv) throw new Error("Conversation not found");
|
|
31
|
+
|
|
32
|
+
expect(() => addChildren(conv, newMessage, conv.messages[0].id)).toThrow();
|
|
33
|
+
});
|
|
34
|
+
it("should not let you create a message that already exists", async () => {
|
|
35
|
+
const convId = await insertLegacyConversation();
|
|
36
|
+
const conv = await collections.conversations.findOne({ _id: new ObjectId(convId) });
|
|
37
|
+
if (!conv) throw new Error("Conversation not found");
|
|
38
|
+
|
|
39
|
+
const messageThatAlreadyExists: Message = {
|
|
40
|
+
id: conv.messages[0].id,
|
|
41
|
+
content: "new message",
|
|
42
|
+
from: "user",
|
|
43
|
+
};
|
|
44
|
+
|
|
45
|
+
expect(() => addChildren(conv, messageThatAlreadyExists, conv.messages[0].id)).toThrow();
|
|
46
|
+
});
|
|
47
|
+
it("should let you create branches on conversations with subtrees", async () => {
|
|
48
|
+
const convId = await insertSideBranchesConversation();
|
|
49
|
+
const conv = await collections.conversations.findOne({ _id: new ObjectId(convId) });
|
|
50
|
+
if (!conv) throw new Error("Conversation not found");
|
|
51
|
+
|
|
52
|
+
const nChildren = conv.messages[0].children?.length;
|
|
53
|
+
if (!nChildren) throw new Error("No children found");
|
|
54
|
+
addChildren(conv, newMessage, conv.messages[0].id);
|
|
55
|
+
expect(conv.messages[0].children?.length).toEqual(nChildren + 1);
|
|
56
|
+
});
|
|
57
|
+
|
|
58
|
+
it("should let you create a new leaf", async () => {
|
|
59
|
+
const convId = await insertSideBranchesConversation();
|
|
60
|
+
const conv = await collections.conversations.findOne({ _id: new ObjectId(convId) });
|
|
61
|
+
if (!conv) throw new Error("Conversation not found");
|
|
62
|
+
|
|
63
|
+
const parentId = conv.messages[conv.messages.length - 1].id;
|
|
64
|
+
const nChildren = conv.messages[conv.messages.length - 1].children?.length;
|
|
65
|
+
|
|
66
|
+
if (nChildren === undefined) throw new Error("No children found");
|
|
67
|
+
expect(nChildren).toEqual(0);
|
|
68
|
+
|
|
69
|
+
addChildren(conv, newMessage, parentId);
|
|
70
|
+
expect(conv.messages[conv.messages.length - 2].children?.length).toEqual(nChildren + 1);
|
|
71
|
+
});
|
|
72
|
+
|
|
73
|
+
it("should let you append to an empty conversation without specifying a parentId", async () => {
|
|
74
|
+
const conv = {
|
|
75
|
+
_id: new ObjectId(),
|
|
76
|
+
rootMessageId: undefined,
|
|
77
|
+
messages: [] as Message[],
|
|
78
|
+
};
|
|
79
|
+
|
|
80
|
+
addChildren(conv, newMessage);
|
|
81
|
+
expect(conv.messages.length).toEqual(1);
|
|
82
|
+
expect(conv.rootMessageId).toEqual(conv.messages[0].id);
|
|
83
|
+
});
|
|
84
|
+
|
|
85
|
+
it("should throw if you don't specify a parentId in a conversation with messages", async () => {
|
|
86
|
+
const convId = await insertLegacyConversation();
|
|
87
|
+
const conv = await collections.conversations.findOne({ _id: new ObjectId(convId) });
|
|
88
|
+
if (!conv) throw new Error("Conversation not found");
|
|
89
|
+
|
|
90
|
+
expect(() => addChildren(conv, newMessage)).toThrow();
|
|
91
|
+
});
|
|
92
|
+
|
|
93
|
+
it("should return the id of the new message", async () => {
|
|
94
|
+
const convId = await insertLegacyConversation();
|
|
95
|
+
const conv = await collections.conversations.findOne({ _id: new ObjectId(convId) });
|
|
96
|
+
if (!conv) throw new Error("Conversation not found");
|
|
97
|
+
|
|
98
|
+
expect(addChildren(conv, newMessage, conv.messages[conv.messages.length - 1].id)).toEqual(
|
|
99
|
+
conv.messages[conv.messages.length - 1].id
|
|
100
|
+
);
|
|
101
|
+
});
|
|
102
|
+
});
|
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
import { v4 } from "uuid";
|
|
2
|
+
import type { Tree, TreeId, NewNode, TreeNode } from "./tree";
|
|
3
|
+
|
|
4
|
+
export function addChildren<T>(conv: Tree<T>, message: NewNode<T>, parentId?: TreeId): TreeId {
|
|
5
|
+
// if this is the first message we just push it
|
|
6
|
+
if (conv.messages.length === 0) {
|
|
7
|
+
const messageId = v4();
|
|
8
|
+
conv.rootMessageId = messageId;
|
|
9
|
+
conv.messages.push({
|
|
10
|
+
...message,
|
|
11
|
+
ancestors: [],
|
|
12
|
+
id: messageId,
|
|
13
|
+
} as TreeNode<T>);
|
|
14
|
+
return messageId;
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
if (!parentId) {
|
|
18
|
+
throw new Error("You need to specify a parentId if this is not the first message");
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
const messageId = v4();
|
|
22
|
+
if (!conv.rootMessageId) {
|
|
23
|
+
// if there is no parentId we just push the message
|
|
24
|
+
if (!!parentId && parentId !== conv.messages[conv.messages.length - 1].id) {
|
|
25
|
+
throw new Error("This is a legacy conversation, you can only append to the last message");
|
|
26
|
+
}
|
|
27
|
+
conv.messages.push({ ...message, id: messageId } as TreeNode<T>);
|
|
28
|
+
return messageId;
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
const ancestors = [...(conv.messages.find((m) => m.id === parentId)?.ancestors ?? []), parentId];
|
|
32
|
+
conv.messages.push({
|
|
33
|
+
...message,
|
|
34
|
+
ancestors,
|
|
35
|
+
id: messageId,
|
|
36
|
+
children: [],
|
|
37
|
+
} as TreeNode<T>);
|
|
38
|
+
|
|
39
|
+
const parent = conv.messages.find((m) => m.id === parentId);
|
|
40
|
+
|
|
41
|
+
if (parent) {
|
|
42
|
+
if (parent.children) {
|
|
43
|
+
parent.children.push(messageId);
|
|
44
|
+
} else parent.children = [messageId];
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
return messageId;
|
|
48
|
+
}
|
|
@@ -0,0 +1,81 @@
|
|
|
1
|
+
import { collections } from "$lib/server/database";
|
|
2
|
+
import { ObjectId } from "mongodb";
|
|
3
|
+
import { describe, expect, it } from "vitest";
|
|
4
|
+
|
|
5
|
+
import { insertLegacyConversation, insertSideBranchesConversation } from "./treeHelpers.spec";
|
|
6
|
+
import type { Message } from "$lib/types/Message";
|
|
7
|
+
import { addSibling } from "./addSibling";
|
|
8
|
+
import type { Conversation } from "$lib/types/Conversation";
|
|
9
|
+
|
|
10
|
+
const newMessage = {
|
|
11
|
+
content: "new message",
|
|
12
|
+
from: "user" as const,
|
|
13
|
+
};
|
|
14
|
+
|
|
15
|
+
Object.freeze(newMessage);
|
|
16
|
+
|
|
17
|
+
describe("addSibling", async () => {
|
|
18
|
+
it("should fail on empty conversations", () => {
|
|
19
|
+
const conv = {
|
|
20
|
+
_id: new ObjectId(),
|
|
21
|
+
rootMessageId: undefined,
|
|
22
|
+
messages: [] as Message[],
|
|
23
|
+
} satisfies Pick<Conversation, "_id" | "rootMessageId" | "messages">;
|
|
24
|
+
|
|
25
|
+
expect(() => addSibling(conv, newMessage, "not-a-real-id-test")).toThrow(
|
|
26
|
+
"Cannot add a sibling to an empty conversation"
|
|
27
|
+
);
|
|
28
|
+
});
|
|
29
|
+
|
|
30
|
+
it("should fail on legacy conversations", async () => {
|
|
31
|
+
const convId = await insertLegacyConversation();
|
|
32
|
+
const conv = await collections.conversations.findOne({ _id: new ObjectId(convId) });
|
|
33
|
+
if (!conv) throw new Error("Conversation not found");
|
|
34
|
+
|
|
35
|
+
expect(() => addSibling(conv, newMessage, conv.messages[0].id)).toThrow(
|
|
36
|
+
"Cannot add a sibling to a legacy conversation"
|
|
37
|
+
);
|
|
38
|
+
});
|
|
39
|
+
|
|
40
|
+
it("should fail if the sibling message doesn't exist", async () => {
|
|
41
|
+
const convId = await insertSideBranchesConversation();
|
|
42
|
+
const conv = await collections.conversations.findOne({ _id: new ObjectId(convId) });
|
|
43
|
+
if (!conv) throw new Error("Conversation not found");
|
|
44
|
+
|
|
45
|
+
expect(() => addSibling(conv, newMessage, "not-a-real-id-test")).toThrow(
|
|
46
|
+
"The sibling message doesn't exist"
|
|
47
|
+
);
|
|
48
|
+
});
|
|
49
|
+
|
|
50
|
+
// TODO: This behaviour should be fixed, we do not need to fail on the root message.
|
|
51
|
+
it("should fail if the sibling message is the root message", async () => {
|
|
52
|
+
const convId = await insertSideBranchesConversation();
|
|
53
|
+
const conv = await collections.conversations.findOne({ _id: new ObjectId(convId) });
|
|
54
|
+
if (!conv) throw new Error("Conversation not found");
|
|
55
|
+
if (!conv.rootMessageId) throw new Error("Root message not found");
|
|
56
|
+
|
|
57
|
+
expect(() => addSibling(conv, newMessage, conv.rootMessageId as Message["id"])).toThrow(
|
|
58
|
+
"The sibling message is the root message, therefore we can't add a sibling"
|
|
59
|
+
);
|
|
60
|
+
});
|
|
61
|
+
|
|
62
|
+
it("should add a sibling to a message", async () => {
|
|
63
|
+
const convId = await insertSideBranchesConversation();
|
|
64
|
+
const conv = await collections.conversations.findOne({ _id: new ObjectId(convId) });
|
|
65
|
+
if (!conv) throw new Error("Conversation not found");
|
|
66
|
+
|
|
67
|
+
// add sibling and check children count for parnets
|
|
68
|
+
|
|
69
|
+
const nChildren = conv.messages[1].children?.length;
|
|
70
|
+
const siblingId = addSibling(conv, newMessage, conv.messages[2].id);
|
|
71
|
+
const nChildrenNew = conv.messages[1].children?.length;
|
|
72
|
+
|
|
73
|
+
if (!nChildren) throw new Error("No children found");
|
|
74
|
+
|
|
75
|
+
expect(nChildrenNew).toBe(nChildren + 1);
|
|
76
|
+
|
|
77
|
+
// make sure siblings have the same ancestors
|
|
78
|
+
const sibling = conv.messages.find((m) => m.id === siblingId);
|
|
79
|
+
expect(sibling?.ancestors).toEqual(conv.messages[2].ancestors);
|
|
80
|
+
});
|
|
81
|
+
});
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
import { v4 } from "uuid";
|
|
2
|
+
import type { Tree, TreeId, NewNode, TreeNode } from "./tree";
|
|
3
|
+
|
|
4
|
+
export function addSibling<T>(conv: Tree<T>, message: NewNode<T>, siblingId: TreeId): TreeId {
|
|
5
|
+
if (conv.messages.length === 0) {
|
|
6
|
+
throw new Error("Cannot add a sibling to an empty conversation");
|
|
7
|
+
}
|
|
8
|
+
if (!conv.rootMessageId) {
|
|
9
|
+
throw new Error("Cannot add a sibling to a legacy conversation");
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
const sibling = conv.messages.find((m) => m.id === siblingId);
|
|
13
|
+
|
|
14
|
+
if (!sibling) {
|
|
15
|
+
throw new Error("The sibling message doesn't exist");
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
if (!sibling.ancestors || sibling.ancestors?.length === 0) {
|
|
19
|
+
throw new Error("The sibling message is the root message, therefore we can't add a sibling");
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
const messageId = v4();
|
|
23
|
+
|
|
24
|
+
conv.messages.push({
|
|
25
|
+
...message,
|
|
26
|
+
id: messageId,
|
|
27
|
+
ancestors: sibling.ancestors,
|
|
28
|
+
children: [],
|
|
29
|
+
} as TreeNode<T>);
|
|
30
|
+
|
|
31
|
+
const nearestAncestorId = sibling.ancestors[sibling.ancestors.length - 1];
|
|
32
|
+
const nearestAncestor = conv.messages.find((m) => m.id === nearestAncestorId);
|
|
33
|
+
|
|
34
|
+
if (nearestAncestor) {
|
|
35
|
+
if (nearestAncestor.children) {
|
|
36
|
+
nearestAncestor.children.push(messageId);
|
|
37
|
+
} else nearestAncestor.children = [messageId];
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
return messageId;
|
|
41
|
+
}
|
|
@@ -0,0 +1,110 @@
|
|
|
1
|
+
import { collections } from "$lib/server/database";
|
|
2
|
+
import { ObjectId } from "mongodb";
|
|
3
|
+
import { describe, expect, it } from "vitest";
|
|
4
|
+
|
|
5
|
+
import {
|
|
6
|
+
insertLegacyConversation,
|
|
7
|
+
insertLinearBranchConversation,
|
|
8
|
+
insertSideBranchesConversation,
|
|
9
|
+
} from "./treeHelpers.spec";
|
|
10
|
+
import { buildSubtree } from "./buildSubtree";
|
|
11
|
+
|
|
12
|
+
describe("buildSubtree", () => {
|
|
13
|
+
it("a subtree in a legacy conversation should be just a slice", async () => {
|
|
14
|
+
const convId = await insertLegacyConversation();
|
|
15
|
+
const conv = await collections.conversations.findOne({ _id: new ObjectId(convId) });
|
|
16
|
+
if (!conv) throw new Error("Conversation not found");
|
|
17
|
+
|
|
18
|
+
// check middle
|
|
19
|
+
const id = conv.messages[2].id;
|
|
20
|
+
const subtree = buildSubtree(conv, id);
|
|
21
|
+
expect(subtree).toEqual(conv.messages.slice(0, 3));
|
|
22
|
+
|
|
23
|
+
// check zero
|
|
24
|
+
const id2 = conv.messages[0].id;
|
|
25
|
+
const subtree2 = buildSubtree(conv, id2);
|
|
26
|
+
expect(subtree2).toEqual(conv.messages.slice(0, 1));
|
|
27
|
+
|
|
28
|
+
//check full length
|
|
29
|
+
const id3 = conv.messages[conv.messages.length - 1].id;
|
|
30
|
+
const subtree3 = buildSubtree(conv, id3);
|
|
31
|
+
expect(subtree3).toEqual(conv.messages);
|
|
32
|
+
});
|
|
33
|
+
|
|
34
|
+
it("a subtree in a linear branch conversation should be the ancestors and the message", async () => {
|
|
35
|
+
const convId = await insertLinearBranchConversation();
|
|
36
|
+
const conv = await collections.conversations.findOne({ _id: new ObjectId(convId) });
|
|
37
|
+
if (!conv) throw new Error("Conversation not found");
|
|
38
|
+
|
|
39
|
+
// check middle
|
|
40
|
+
const id = conv.messages[1].id;
|
|
41
|
+
const subtree = buildSubtree(conv, id);
|
|
42
|
+
expect(subtree).toEqual([conv.messages[0], conv.messages[1]]);
|
|
43
|
+
|
|
44
|
+
// check zero
|
|
45
|
+
const id2 = conv.messages[0].id;
|
|
46
|
+
const subtree2 = buildSubtree(conv, id2);
|
|
47
|
+
expect(subtree2).toEqual([conv.messages[0]]);
|
|
48
|
+
|
|
49
|
+
//check full length
|
|
50
|
+
const id3 = conv.messages[conv.messages.length - 1].id;
|
|
51
|
+
const subtree3 = buildSubtree(conv, id3);
|
|
52
|
+
expect(subtree3).toEqual(conv.messages);
|
|
53
|
+
});
|
|
54
|
+
|
|
55
|
+
it("should throw an error if the message is not found", async () => {
|
|
56
|
+
const convId = await insertLinearBranchConversation();
|
|
57
|
+
const conv = await collections.conversations.findOne({ _id: new ObjectId(convId) });
|
|
58
|
+
if (!conv) throw new Error("Conversation not found");
|
|
59
|
+
|
|
60
|
+
const id = "not-a-real-id-test";
|
|
61
|
+
|
|
62
|
+
expect(() => buildSubtree(conv, id)).toThrow("Message not found");
|
|
63
|
+
});
|
|
64
|
+
|
|
65
|
+
it("should throw an error if the ancestor is not found", async () => {
|
|
66
|
+
const convId = await insertLinearBranchConversation();
|
|
67
|
+
const conv = await collections.conversations.findOne({ _id: new ObjectId(convId) });
|
|
68
|
+
if (!conv) throw new Error("Conversation not found");
|
|
69
|
+
|
|
70
|
+
const id = "1-1-1-1-2";
|
|
71
|
+
|
|
72
|
+
conv.messages[1].ancestors = ["not-a-real-id-test"];
|
|
73
|
+
|
|
74
|
+
expect(() => buildSubtree(conv, id)).toThrow("Ancestor not found");
|
|
75
|
+
});
|
|
76
|
+
|
|
77
|
+
it("should work on empty conversations", () => {
|
|
78
|
+
const conv = {
|
|
79
|
+
_id: new ObjectId(),
|
|
80
|
+
rootMessageId: undefined,
|
|
81
|
+
messages: [],
|
|
82
|
+
};
|
|
83
|
+
|
|
84
|
+
const subtree = buildSubtree(conv, "not-a-real-id-test");
|
|
85
|
+
expect(subtree).toEqual([]);
|
|
86
|
+
});
|
|
87
|
+
|
|
88
|
+
it("should work for conversation with subtrees", async () => {
|
|
89
|
+
const convId = await insertSideBranchesConversation();
|
|
90
|
+
const conv = await collections.conversations.findOne({ _id: new ObjectId(convId) });
|
|
91
|
+
if (!conv) throw new Error("Conversation not found");
|
|
92
|
+
|
|
93
|
+
const subtree = buildSubtree(conv, "1-1-1-1-2");
|
|
94
|
+
expect(subtree).toEqual([conv.messages[0], conv.messages[1]]);
|
|
95
|
+
|
|
96
|
+
const subtree2 = buildSubtree(conv, "1-1-1-1-4");
|
|
97
|
+
expect(subtree2).toEqual([
|
|
98
|
+
conv.messages[0],
|
|
99
|
+
conv.messages[1],
|
|
100
|
+
conv.messages[2],
|
|
101
|
+
conv.messages[3],
|
|
102
|
+
]);
|
|
103
|
+
|
|
104
|
+
const subtree3 = buildSubtree(conv, "1-1-1-1-6");
|
|
105
|
+
expect(subtree3).toEqual([conv.messages[0], conv.messages[4], conv.messages[5]]);
|
|
106
|
+
|
|
107
|
+
const subtree4 = buildSubtree(conv, "1-1-1-1-7");
|
|
108
|
+
expect(subtree4).toEqual([conv.messages[0], conv.messages[4], conv.messages[6]]);
|
|
109
|
+
});
|
|
110
|
+
});
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import type { Tree, TreeId, TreeNode } from "./tree";
|
|
2
|
+
|
|
3
|
+
export function buildSubtree<T>(conv: Tree<T>, id: TreeId): TreeNode<T>[] {
|
|
4
|
+
if (!conv.rootMessageId) {
|
|
5
|
+
if (conv.messages.length === 0) return [];
|
|
6
|
+
// legacy conversation slice up to id
|
|
7
|
+
const index = conv.messages.findIndex((m) => m.id === id);
|
|
8
|
+
if (index === -1) throw new Error("Message not found");
|
|
9
|
+
return conv.messages.slice(0, index + 1);
|
|
10
|
+
} else {
|
|
11
|
+
// find the message with the right id then create the ancestor tree
|
|
12
|
+
const message = conv.messages.find((m) => m.id === id);
|
|
13
|
+
if (!message) throw new Error("Message not found");
|
|
14
|
+
|
|
15
|
+
return [
|
|
16
|
+
...(message.ancestors?.map((ancestorId) => {
|
|
17
|
+
const ancestor = conv.messages.find((m) => m.id === ancestorId);
|
|
18
|
+
if (!ancestor) throw new Error("Ancestor not found");
|
|
19
|
+
return ancestor;
|
|
20
|
+
}) ?? []),
|
|
21
|
+
message,
|
|
22
|
+
];
|
|
23
|
+
}
|
|
24
|
+
}
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
import { collections } from "$lib/server/database";
|
|
2
|
+
import { ObjectId } from "mongodb";
|
|
3
|
+
import { describe, expect, it } from "vitest";
|
|
4
|
+
|
|
5
|
+
import { convertLegacyConversation } from "./convertLegacyConversation";
|
|
6
|
+
import { insertLegacyConversation } from "./treeHelpers.spec";
|
|
7
|
+
|
|
8
|
+
describe("convertLegacyConversation", () => {
|
|
9
|
+
it("should convert a legacy conversation", async () => {
|
|
10
|
+
const convId = await insertLegacyConversation();
|
|
11
|
+
const conv = await collections.conversations.findOne({ _id: new ObjectId(convId) });
|
|
12
|
+
if (!conv) throw new Error("Conversation not found");
|
|
13
|
+
|
|
14
|
+
const newConv = convertLegacyConversation(conv);
|
|
15
|
+
|
|
16
|
+
expect(newConv.rootMessageId).toBe(newConv.messages[0].id);
|
|
17
|
+
expect(newConv.messages[0].ancestors).toEqual([]);
|
|
18
|
+
expect(newConv.messages[1].ancestors).toEqual([newConv.messages[0].id]);
|
|
19
|
+
expect(newConv.messages[0].children).toEqual([newConv.messages[1].id]);
|
|
20
|
+
});
|
|
21
|
+
it("should work on empty conversations", async () => {
|
|
22
|
+
const conv = {
|
|
23
|
+
_id: new ObjectId(),
|
|
24
|
+
rootMessageId: undefined,
|
|
25
|
+
messages: [],
|
|
26
|
+
};
|
|
27
|
+
const newConv = convertLegacyConversation(conv);
|
|
28
|
+
expect(newConv.rootMessageId).toBe(undefined);
|
|
29
|
+
expect(newConv.messages).toEqual([]);
|
|
30
|
+
});
|
|
31
|
+
});
|