ruflo 3.5.2 → 3.5.4
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,92 @@
|
|
|
1
|
+
<script lang="ts">
|
|
2
|
+
import { requireAuthUser } from "$lib/utils/auth";
|
|
3
|
+
import CarbonImage from "~icons/carbon/image";
|
|
4
|
+
|
|
5
|
+
interface Props {
|
|
6
|
+
// import EosIconsLoading from "~icons/eos-icons/loading";
|
|
7
|
+
files: File[];
|
|
8
|
+
mimeTypes?: string[];
|
|
9
|
+
onDrag?: boolean;
|
|
10
|
+
onDragInner?: boolean;
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
let {
|
|
14
|
+
files = $bindable(),
|
|
15
|
+
mimeTypes = [],
|
|
16
|
+
onDrag = $bindable(false),
|
|
17
|
+
onDragInner = $bindable(false),
|
|
18
|
+
}: Props = $props();
|
|
19
|
+
|
|
20
|
+
async function dropHandle(event: DragEvent) {
|
|
21
|
+
event.preventDefault();
|
|
22
|
+
if (!requireAuthUser() && event.dataTransfer && event.dataTransfer.items) {
|
|
23
|
+
// Use DataTransferItemList interface to access the file(s)
|
|
24
|
+
if (files.length > 0) {
|
|
25
|
+
files = [];
|
|
26
|
+
}
|
|
27
|
+
if (event.dataTransfer.items[0].kind === "file") {
|
|
28
|
+
for (let i = 0; i < event.dataTransfer.items.length; i++) {
|
|
29
|
+
const file = event.dataTransfer.items[i].getAsFile();
|
|
30
|
+
|
|
31
|
+
if (file) {
|
|
32
|
+
// check if the file matches the mimeTypes
|
|
33
|
+
// else abort
|
|
34
|
+
if (
|
|
35
|
+
!mimeTypes.some((mimeType: string) => {
|
|
36
|
+
const [type, subtype] = mimeType.split("/");
|
|
37
|
+
const [fileType, fileSubtype] = file.type.split("/");
|
|
38
|
+
return (
|
|
39
|
+
(type === "*" || type === fileType) &&
|
|
40
|
+
(subtype === "*" || subtype === fileSubtype)
|
|
41
|
+
);
|
|
42
|
+
})
|
|
43
|
+
) {
|
|
44
|
+
setErrorMsg(
|
|
45
|
+
`Some file type not supported. Only allowed: ${mimeTypes.join(
|
|
46
|
+
", "
|
|
47
|
+
)}. Uploaded document is of type ${file.type}`
|
|
48
|
+
);
|
|
49
|
+
files = [];
|
|
50
|
+
return;
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
// if file is bigger than 10MB abort
|
|
54
|
+
if (file.size > 10 * 1024 * 1024) {
|
|
55
|
+
setErrorMsg("Some file is too big. (10MB max)");
|
|
56
|
+
files = [];
|
|
57
|
+
return;
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
// add the file to the files array
|
|
61
|
+
files = [...files, file];
|
|
62
|
+
|
|
63
|
+
// Tools removed: no settings update for document parser
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
onDrag = false;
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
function setErrorMsg(errorMsg: string) {
|
|
72
|
+
onDrag = false;
|
|
73
|
+
alert(errorMsg);
|
|
74
|
+
}
|
|
75
|
+
</script>
|
|
76
|
+
|
|
77
|
+
<div
|
|
78
|
+
id="dropzone"
|
|
79
|
+
role="form"
|
|
80
|
+
ondrop={dropHandle}
|
|
81
|
+
ondragenter={() => (onDragInner = true)}
|
|
82
|
+
ondragleave={() => (onDragInner = false)}
|
|
83
|
+
ondragover={(e) => {
|
|
84
|
+
e.preventDefault();
|
|
85
|
+
}}
|
|
86
|
+
class="relative flex h-28 w-full max-w-4xl flex-col items-center justify-center gap-1 rounded-xl border-2 border-dotted {onDragInner
|
|
87
|
+
? 'border-blue-200 !bg-blue-600/10 text-blue-600 *:pointer-events-none dark:border-blue-600 dark:bg-blue-600/20 dark:text-blue-600'
|
|
88
|
+
: 'bg-gray-100 text-gray-500 dark:border-gray-500 dark:bg-gray-700 dark:text-gray-400'}"
|
|
89
|
+
>
|
|
90
|
+
<CarbonImage class="text-xl" />
|
|
91
|
+
<p>Drop File to add to chat</p>
|
|
92
|
+
</div>
|
|
@@ -0,0 +1,66 @@
|
|
|
1
|
+
<script lang="ts">
|
|
2
|
+
import { onMount } from "svelte";
|
|
3
|
+
import Portal from "../Portal.svelte";
|
|
4
|
+
import CarbonClose from "~icons/carbon/close";
|
|
5
|
+
|
|
6
|
+
interface Props {
|
|
7
|
+
src: string;
|
|
8
|
+
onclose: () => void;
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
let { src, onclose }: Props = $props();
|
|
12
|
+
|
|
13
|
+
function handleKeydown(e: KeyboardEvent) {
|
|
14
|
+
if (e.key === "Escape") {
|
|
15
|
+
e.preventDefault();
|
|
16
|
+
e.stopPropagation();
|
|
17
|
+
onclose();
|
|
18
|
+
}
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
function handleOverlayClick(e: MouseEvent) {
|
|
22
|
+
// Close when clicking the overlay (not the image)
|
|
23
|
+
if (e.target === e.currentTarget) {
|
|
24
|
+
onclose();
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
onMount(() => {
|
|
29
|
+
// Prevent body scroll while lightbox is open
|
|
30
|
+
const originalOverflow = document.body.style.overflow;
|
|
31
|
+
document.body.style.overflow = "hidden";
|
|
32
|
+
|
|
33
|
+
return () => {
|
|
34
|
+
document.body.style.overflow = originalOverflow;
|
|
35
|
+
};
|
|
36
|
+
});
|
|
37
|
+
</script>
|
|
38
|
+
|
|
39
|
+
<svelte:window onkeydown={handleKeydown} />
|
|
40
|
+
|
|
41
|
+
<Portal>
|
|
42
|
+
<!-- svelte-ignore a11y_click_events_have_key_events -->
|
|
43
|
+
<!-- svelte-ignore a11y_no_static_element_interactions -->
|
|
44
|
+
<div
|
|
45
|
+
class="fixed inset-0 z-50 grid place-items-center bg-black/90 backdrop-blur-sm"
|
|
46
|
+
onclick={handleOverlayClick}
|
|
47
|
+
>
|
|
48
|
+
<!-- Close button -->
|
|
49
|
+
<button
|
|
50
|
+
class="absolute right-3 top-3 grid size-8 place-items-center rounded-full border border-white/25 bg-white/20 text-gray-300 hover:bg-white/30 sm:right-6 sm:top-6"
|
|
51
|
+
onclick={onclose}
|
|
52
|
+
aria-label="Close"
|
|
53
|
+
>
|
|
54
|
+
<CarbonClose />
|
|
55
|
+
</button>
|
|
56
|
+
|
|
57
|
+
<!-- Image with moon-landing's resize strategy -->
|
|
58
|
+
<!-- svelte-ignore a11y_no_noninteractive_element_interactions -->
|
|
59
|
+
<img
|
|
60
|
+
{src}
|
|
61
|
+
alt=""
|
|
62
|
+
class="h-auto max-h-[calc(100vh-160px)] w-auto max-w-full"
|
|
63
|
+
onclick={(e) => e.stopPropagation()}
|
|
64
|
+
/>
|
|
65
|
+
</div>
|
|
66
|
+
</Portal>
|
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
<script lang="ts">
|
|
2
|
+
import type { Token } from "$lib/utils/marked";
|
|
3
|
+
import CodeBlock from "../CodeBlock.svelte";
|
|
4
|
+
|
|
5
|
+
interface Props {
|
|
6
|
+
tokens: Token[];
|
|
7
|
+
loading?: boolean;
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
let { tokens, loading = false }: Props = $props();
|
|
11
|
+
|
|
12
|
+
// Derive rendered tokens for memoization
|
|
13
|
+
const renderedTokens = $derived(tokens);
|
|
14
|
+
</script>
|
|
15
|
+
|
|
16
|
+
{#each renderedTokens as token}
|
|
17
|
+
{#if token.type === "text"}
|
|
18
|
+
<!-- eslint-disable-next-line svelte/no-at-html-tags -->
|
|
19
|
+
{@html token.html}
|
|
20
|
+
{:else if token.type === "code"}
|
|
21
|
+
<CodeBlock code={token.code} rawCode={token.rawCode} loading={loading && !token.isClosed} />
|
|
22
|
+
{/if}
|
|
23
|
+
{/each}
|
|
@@ -0,0 +1,69 @@
|
|
|
1
|
+
<script lang="ts">
|
|
2
|
+
import { processBlocks, processBlocksSync, type BlockToken } from "$lib/utils/marked";
|
|
3
|
+
import MarkdownWorker from "$lib/workers/markdownWorker?worker";
|
|
4
|
+
import MarkdownBlock from "./MarkdownBlock.svelte";
|
|
5
|
+
import { browser } from "$app/environment";
|
|
6
|
+
|
|
7
|
+
import { onMount, onDestroy } from "svelte";
|
|
8
|
+
import { updateDebouncer } from "$lib/utils/updates";
|
|
9
|
+
|
|
10
|
+
interface Props {
|
|
11
|
+
content: string;
|
|
12
|
+
sources?: { title?: string; link: string }[];
|
|
13
|
+
loading?: boolean;
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
let { content, sources = [], loading = false }: Props = $props();
|
|
17
|
+
|
|
18
|
+
let blocks: BlockToken[] = $state(processBlocksSync(content, sources));
|
|
19
|
+
let worker: Worker | null = null;
|
|
20
|
+
let latestRequestId = 0;
|
|
21
|
+
|
|
22
|
+
function handleBlocks(result: BlockToken[], requestId: number) {
|
|
23
|
+
if (requestId !== latestRequestId) return;
|
|
24
|
+
blocks = result;
|
|
25
|
+
updateDebouncer.endRender();
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
$effect(() => {
|
|
29
|
+
if (!browser) {
|
|
30
|
+
blocks = processBlocksSync(content, sources);
|
|
31
|
+
return;
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
const requestId = ++latestRequestId;
|
|
35
|
+
|
|
36
|
+
if (worker) {
|
|
37
|
+
updateDebouncer.startRender();
|
|
38
|
+
worker.postMessage({ type: "process", content, sources, requestId });
|
|
39
|
+
return;
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
(async () => {
|
|
43
|
+
updateDebouncer.startRender();
|
|
44
|
+
const processed = await processBlocks(content, sources);
|
|
45
|
+
// Only apply if this is still the latest request
|
|
46
|
+
handleBlocks(processed, requestId);
|
|
47
|
+
})();
|
|
48
|
+
});
|
|
49
|
+
|
|
50
|
+
onMount(() => {
|
|
51
|
+
if (typeof Worker !== "undefined") {
|
|
52
|
+
worker = new MarkdownWorker();
|
|
53
|
+
worker.onmessage = (event: MessageEvent) => {
|
|
54
|
+
const data = event.data as { type?: string; blocks?: BlockToken[]; requestId?: number };
|
|
55
|
+
if (data?.type !== "processed" || !data.blocks || data.requestId === undefined) return;
|
|
56
|
+
handleBlocks(data.blocks, data.requestId);
|
|
57
|
+
};
|
|
58
|
+
}
|
|
59
|
+
});
|
|
60
|
+
|
|
61
|
+
onDestroy(() => {
|
|
62
|
+
worker?.terminate();
|
|
63
|
+
worker = null;
|
|
64
|
+
});
|
|
65
|
+
</script>
|
|
66
|
+
|
|
67
|
+
{#each blocks as block, index (loading && index === blocks.length - 1 ? `stream-${index}` : block.id)}
|
|
68
|
+
<MarkdownBlock tokens={block.tokens} {loading} />
|
|
69
|
+
{/each}
|
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
import MarkdownRenderer from "./MarkdownRenderer.svelte";
|
|
2
|
+
import { render } from "vitest-browser-svelte";
|
|
3
|
+
import { page } from "@vitest/browser/context";
|
|
4
|
+
|
|
5
|
+
import { describe, expect, it } from "vitest";
|
|
6
|
+
|
|
7
|
+
describe("MarkdownRenderer", () => {
|
|
8
|
+
it("renders", () => {
|
|
9
|
+
render(MarkdownRenderer, { content: "Hello, world!" });
|
|
10
|
+
expect(page.getByText("Hello, world!")).toBeInTheDocument();
|
|
11
|
+
});
|
|
12
|
+
it("renders headings", () => {
|
|
13
|
+
render(MarkdownRenderer, { content: "# Hello, world!" });
|
|
14
|
+
expect(page.getByRole("heading", { level: 1 })).toBeInTheDocument();
|
|
15
|
+
});
|
|
16
|
+
it("renders links", () => {
|
|
17
|
+
render(MarkdownRenderer, { content: "[Hello, world!](https://example.com)" });
|
|
18
|
+
const link = page.getByRole("link", { name: "Hello, world!" });
|
|
19
|
+
expect(link).toBeInTheDocument();
|
|
20
|
+
expect(link).toHaveAttribute("href", "https://example.com");
|
|
21
|
+
expect(link).toHaveAttribute("target", "_blank");
|
|
22
|
+
expect(link).toHaveAttribute("rel", "noreferrer");
|
|
23
|
+
});
|
|
24
|
+
it("renders inline codespans", () => {
|
|
25
|
+
render(MarkdownRenderer, { content: "`foobar`" });
|
|
26
|
+
expect(page.getByRole("code")).toHaveTextContent("foobar");
|
|
27
|
+
});
|
|
28
|
+
it("renders block codes", () => {
|
|
29
|
+
render(MarkdownRenderer, { content: "```foobar```" });
|
|
30
|
+
expect(page.getByRole("code")).toHaveTextContent("foobar");
|
|
31
|
+
});
|
|
32
|
+
it("doesnt render raw html directly", () => {
|
|
33
|
+
render(MarkdownRenderer, { content: "<button>Click me</button>" });
|
|
34
|
+
expect(page.getByRole("button").elements).toHaveLength(0);
|
|
35
|
+
// htmlparser2 escapes disallowed tags
|
|
36
|
+
expect(page.getByRole("paragraph")).toHaveTextContent("<button>Click me</button>");
|
|
37
|
+
});
|
|
38
|
+
it("renders latex", () => {
|
|
39
|
+
const { baseElement } = render(MarkdownRenderer, { content: "$(oo)^2$" });
|
|
40
|
+
expect(baseElement.querySelectorAll(".katex")).toHaveLength(1);
|
|
41
|
+
});
|
|
42
|
+
it("does not render latex in code blocks", () => {
|
|
43
|
+
const { baseElement } = render(MarkdownRenderer, { content: "```\n$(oo)^2$\n```" });
|
|
44
|
+
expect(baseElement.querySelectorAll(".katex")).toHaveLength(0);
|
|
45
|
+
});
|
|
46
|
+
it("does not render latex in inline codes", () => {
|
|
47
|
+
const { baseElement } = render(MarkdownRenderer, { content: "`$oo` and `$bar`" });
|
|
48
|
+
expect(baseElement.querySelectorAll(".katex")).toHaveLength(0);
|
|
49
|
+
});
|
|
50
|
+
it("does not render latex across multiple lines", () => {
|
|
51
|
+
const { baseElement } = render(MarkdownRenderer, { content: "* $oo \n* $aa" });
|
|
52
|
+
expect(baseElement.querySelectorAll(".katex")).toHaveLength(0);
|
|
53
|
+
});
|
|
54
|
+
it("renders latex with some < and > symbols", () => {
|
|
55
|
+
const { baseElement } = render(MarkdownRenderer, { content: "$foo < bar > baz$" });
|
|
56
|
+
expect(baseElement.querySelectorAll(".katex")).toHaveLength(1);
|
|
57
|
+
});
|
|
58
|
+
});
|
|
@@ -0,0 +1,103 @@
|
|
|
1
|
+
<script lang="ts">
|
|
2
|
+
import { onDestroy } from "svelte";
|
|
3
|
+
|
|
4
|
+
let { animating = false, classNames = "" } = $props();
|
|
5
|
+
|
|
6
|
+
let blobAnim: SVGAnimateElement | undefined = $state();
|
|
7
|
+
let svgEl: SVGSVGElement | undefined = $state();
|
|
8
|
+
|
|
9
|
+
// Only trigger begin/end on transitions, and pause when not animating
|
|
10
|
+
let prevAnimating: boolean | undefined = undefined;
|
|
11
|
+
let prevBlobAnim: SVGAnimateElement | undefined = undefined;
|
|
12
|
+
|
|
13
|
+
$effect(() => {
|
|
14
|
+
if (!blobAnim) return;
|
|
15
|
+
const blobChanged = blobAnim !== prevBlobAnim;
|
|
16
|
+
const animChanged = animating !== prevAnimating;
|
|
17
|
+
if (!(blobChanged || animChanged)) return;
|
|
18
|
+
|
|
19
|
+
if (animating) {
|
|
20
|
+
// Resume animations and start once
|
|
21
|
+
svgEl?.unpauseAnimations?.();
|
|
22
|
+
blobAnim.beginElement();
|
|
23
|
+
} else {
|
|
24
|
+
// Stop current run and pause so it cannot restart from queued begins
|
|
25
|
+
blobAnim.endElement();
|
|
26
|
+
svgEl?.pauseAnimations?.();
|
|
27
|
+
}
|
|
28
|
+
prevAnimating = animating;
|
|
29
|
+
prevBlobAnim = blobAnim;
|
|
30
|
+
});
|
|
31
|
+
|
|
32
|
+
onDestroy(() => {
|
|
33
|
+
blobAnim?.endElement();
|
|
34
|
+
svgEl?.pauseAnimations?.();
|
|
35
|
+
});
|
|
36
|
+
</script>
|
|
37
|
+
|
|
38
|
+
<svg
|
|
39
|
+
bind:this={svgEl}
|
|
40
|
+
class={classNames}
|
|
41
|
+
id="ball"
|
|
42
|
+
width="1em"
|
|
43
|
+
height="1em"
|
|
44
|
+
viewBox="0 0 12 12"
|
|
45
|
+
fill="none"
|
|
46
|
+
xmlns="http://www.w3.org/2000/svg"
|
|
47
|
+
aria-label="Ball mask"
|
|
48
|
+
>
|
|
49
|
+
<g clip-path="url(#a)">
|
|
50
|
+
<!-- circular mask -->
|
|
51
|
+
<path d="M12 6A6 6 0 1 0 0 6a6 6 0 0 0 12 0Z" fill="#fff" />
|
|
52
|
+
<mask id="b" style="mask-type:luminance" x="0" y="0" width="12" height="12">
|
|
53
|
+
<path d="M12 6A6 6 0 1 0 0 6a6 6 0 0 0 12 0Z" fill="#fff" />
|
|
54
|
+
</mask>
|
|
55
|
+
|
|
56
|
+
<!-- the blurred black shape inside the circular mask -->
|
|
57
|
+
<g filter="url(#c)" mask="url(#b)">
|
|
58
|
+
<!-- BASE state (normalized to absolute L commands) -->
|
|
59
|
+
<path id="blob" fill="#000" d="M11 1 L8 -4 L3 -8 L-6 6 L3 12 L7 11 L6 2 L11 1 Z">
|
|
60
|
+
<!-- MORPH: base -> mid -> far -> mid -> base -->
|
|
61
|
+
<animate
|
|
62
|
+
bind:this={blobAnim}
|
|
63
|
+
attributeName="d"
|
|
64
|
+
begin="indefinite"
|
|
65
|
+
end="indefinite"
|
|
66
|
+
dur="3.2s"
|
|
67
|
+
repeatCount="indefinite"
|
|
68
|
+
fill="remove"
|
|
69
|
+
calcMode="spline"
|
|
70
|
+
keyTimes="0; .33; .66; .9; 1"
|
|
71
|
+
keySplines="
|
|
72
|
+
.4 0 .2 1;
|
|
73
|
+
.4 0 .2 1;
|
|
74
|
+
.4 0 .2 1;
|
|
75
|
+
.4 0 .2 1"
|
|
76
|
+
values="
|
|
77
|
+
M11 1 L8 -4 L3 -8 L-6 6 L3 12 L7 11 L6 2 L11 1 Z;
|
|
78
|
+
M11 1 L8 -4 L3 -8 L-6 6 L3 12 L5 9 L7 4 L11 1 Z;
|
|
79
|
+
M11 1 L8 -4 L3 -8 L-6 6 L3 12 L3 6 L5 1 L11 1 Z;
|
|
80
|
+
M11 1 L8 -4 L3 -8 L-6 6 L3 12 L5 9 L7 4 L11 1 Z;
|
|
81
|
+
M11 1 L8 -4 L3 -8 L-6 6 L3 12 L7 11 L6 2 L11 1 Z"
|
|
82
|
+
/>
|
|
83
|
+
</path>
|
|
84
|
+
</g>
|
|
85
|
+
</g>
|
|
86
|
+
|
|
87
|
+
<defs>
|
|
88
|
+
<clipPath id="a"><path fill="#fff" d="M0 0h12v12H0z" /></clipPath>
|
|
89
|
+
<filter
|
|
90
|
+
id="c"
|
|
91
|
+
x="-9.4"
|
|
92
|
+
y="-10.8"
|
|
93
|
+
width="23.8"
|
|
94
|
+
height="26"
|
|
95
|
+
filterUnits="userSpaceOnUse"
|
|
96
|
+
color-interpolation-filters="sRGB"
|
|
97
|
+
>
|
|
98
|
+
<feFlood flood-opacity="0" result="BackgroundImageFix" />
|
|
99
|
+
<feBlend in="SourceGraphic" in2="BackgroundImageFix" result="shape" />
|
|
100
|
+
<feGaussianBlur stdDeviation="1.6" />
|
|
101
|
+
</filter>
|
|
102
|
+
</defs>
|
|
103
|
+
</svg>
|
|
@@ -0,0 +1,64 @@
|
|
|
1
|
+
<script lang="ts">
|
|
2
|
+
import { invalidateAll } from "$app/navigation";
|
|
3
|
+
import { page } from "$app/state";
|
|
4
|
+
import { base } from "$app/paths";
|
|
5
|
+
import type { Model } from "$lib/types/Model";
|
|
6
|
+
|
|
7
|
+
interface Props {
|
|
8
|
+
models: Model[];
|
|
9
|
+
currentModel: Model;
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
let { models, currentModel }: Props = $props();
|
|
13
|
+
|
|
14
|
+
let selectedModelId = $state(
|
|
15
|
+
models.map((m) => m.id).includes(currentModel.id) ? currentModel.id : models[0].id
|
|
16
|
+
);
|
|
17
|
+
|
|
18
|
+
async function handleModelChange() {
|
|
19
|
+
if (!page.params.id) return;
|
|
20
|
+
|
|
21
|
+
try {
|
|
22
|
+
const response = await fetch(`${base}/conversation/${page.params.id}`, {
|
|
23
|
+
method: "PATCH",
|
|
24
|
+
headers: {
|
|
25
|
+
"Content-Type": "application/json",
|
|
26
|
+
},
|
|
27
|
+
body: JSON.stringify({ model: selectedModelId }),
|
|
28
|
+
});
|
|
29
|
+
|
|
30
|
+
if (!response.ok) {
|
|
31
|
+
throw new Error("Failed to update model");
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
await invalidateAll();
|
|
35
|
+
} catch (error) {
|
|
36
|
+
console.error(error);
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
</script>
|
|
40
|
+
|
|
41
|
+
<div
|
|
42
|
+
class="mx-auto mt-0 flex w-fit flex-col items-center justify-center gap-2 rounded-lg border border-gray-200 bg-gray-500/20 p-4 dark:border-gray-800"
|
|
43
|
+
>
|
|
44
|
+
<span>
|
|
45
|
+
This model is no longer available. Switch to a new one to continue this conversation:
|
|
46
|
+
</span>
|
|
47
|
+
<div class="flex items-center space-x-2">
|
|
48
|
+
<select
|
|
49
|
+
bind:value={selectedModelId}
|
|
50
|
+
class="rounded-md bg-gray-100 px-2 py-1 dark:bg-gray-900 max-sm:max-w-32"
|
|
51
|
+
>
|
|
52
|
+
{#each models as model}
|
|
53
|
+
<option value={model.id}>{model.name}</option>
|
|
54
|
+
{/each}
|
|
55
|
+
</select>
|
|
56
|
+
<button
|
|
57
|
+
onclick={handleModelChange}
|
|
58
|
+
disabled={selectedModelId === currentModel.id}
|
|
59
|
+
class="rounded-md bg-gray-100 px-2 py-1 dark:bg-gray-900"
|
|
60
|
+
>
|
|
61
|
+
Accept
|
|
62
|
+
</button>
|
|
63
|
+
</div>
|
|
64
|
+
</div>
|
|
@@ -0,0 +1,81 @@
|
|
|
1
|
+
<script lang="ts">
|
|
2
|
+
import MarkdownRenderer from "./MarkdownRenderer.svelte";
|
|
3
|
+
import BlockWrapper from "./BlockWrapper.svelte";
|
|
4
|
+
|
|
5
|
+
interface Props {
|
|
6
|
+
content: string;
|
|
7
|
+
loading?: boolean;
|
|
8
|
+
hasNext?: boolean;
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
let { content, loading = false, hasNext = false }: Props = $props();
|
|
12
|
+
let isOpen = $state(false);
|
|
13
|
+
let wasLoading = $state(false);
|
|
14
|
+
let initialized = $state(false);
|
|
15
|
+
|
|
16
|
+
// Track loading transitions to auto-expand/collapse
|
|
17
|
+
$effect(() => {
|
|
18
|
+
// Auto-expand on first render if already loading
|
|
19
|
+
if (!initialized) {
|
|
20
|
+
initialized = true;
|
|
21
|
+
if (loading) {
|
|
22
|
+
isOpen = true;
|
|
23
|
+
wasLoading = true;
|
|
24
|
+
return;
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
if (loading && !wasLoading) {
|
|
29
|
+
// Loading started - auto-expand
|
|
30
|
+
isOpen = true;
|
|
31
|
+
} else if (!loading && wasLoading) {
|
|
32
|
+
// Loading finished - auto-collapse
|
|
33
|
+
isOpen = false;
|
|
34
|
+
}
|
|
35
|
+
wasLoading = loading;
|
|
36
|
+
});
|
|
37
|
+
</script>
|
|
38
|
+
|
|
39
|
+
{#snippet icon()}
|
|
40
|
+
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 32 32">
|
|
41
|
+
<path
|
|
42
|
+
class="stroke-gray-500 dark:stroke-gray-400"
|
|
43
|
+
style="stroke-width: 1.9; fill: none; stroke-linecap: round; stroke-linejoin: round;"
|
|
44
|
+
d="M16 6v3.33M16 6c0-2.65 3.25-4.3 5.4-2.62 1.2.95 1.6 2.65.95 4.04a3.63 3.63 0 0 1 4.61.16 3.45 3.45 0 0 1 .46 4.37 5.32 5.32 0 0 1 1.87 4.75c-.22 1.66-1.39 3.6-3.07 4.14M16 6c0-2.65-3.25-4.3-5.4-2.62a3.37 3.37 0 0 0-.95 4.04 3.65 3.65 0 0 0-4.6.16 3.37 3.37 0 0 0-.49 4.27 5.57 5.57 0 0 0-1.85 4.85 5.3 5.3 0 0 0 3.07 4.15M16 9.33v17.34m0-17.34c0 2.18 1.82 4 4 4m6.22 7.5c.67 1.3.56 2.91-.27 4.11a4.05 4.05 0 0 1-4.62 1.5c0 1.53-1.05 2.9-2.66 2.9A2.7 2.7 0 0 1 16 26.66m10.22-5.83a4.05 4.05 0 0 0-3.55-2.17m-16.9 2.18a4.05 4.05 0 0 0 .28 4.1c1 1.44 2.92 2.09 4.59 1.5 0 1.52 1.12 2.88 2.7 2.88A2.7 2.7 0 0 0 16 26.67M5.78 20.85a4.04 4.04 0 0 1 3.55-2.18"
|
|
45
|
+
/>
|
|
46
|
+
</svg>
|
|
47
|
+
{/snippet}
|
|
48
|
+
|
|
49
|
+
<BlockWrapper
|
|
50
|
+
{icon}
|
|
51
|
+
{hasNext}
|
|
52
|
+
iconBg="bg-gray-100 dark:bg-gray-700"
|
|
53
|
+
iconRing="ring-gray-200 dark:ring-gray-600"
|
|
54
|
+
>
|
|
55
|
+
<!-- Collapsed view (clickable to expand) -->
|
|
56
|
+
<button
|
|
57
|
+
type="button"
|
|
58
|
+
class="group/text w-full cursor-pointer text-left"
|
|
59
|
+
onclick={() => (isOpen = !isOpen)}
|
|
60
|
+
>
|
|
61
|
+
{#if isOpen}
|
|
62
|
+
<!-- Expanded: show full content -->
|
|
63
|
+
<div
|
|
64
|
+
class="prose prose-sm max-w-none text-sm leading-relaxed text-gray-500 dark:prose-invert dark:text-gray-400"
|
|
65
|
+
>
|
|
66
|
+
<MarkdownRenderer {content} {loading} />
|
|
67
|
+
</div>
|
|
68
|
+
{:else}
|
|
69
|
+
<!-- Collapsed: 2-line preview (plain text, strip markdown) -->
|
|
70
|
+
<div
|
|
71
|
+
class="line-clamp-2 max-h-[3.25em] text-sm leading-relaxed text-gray-500 dark:text-gray-400"
|
|
72
|
+
class:animate-pulse={loading}
|
|
73
|
+
>
|
|
74
|
+
{content
|
|
75
|
+
.replace(/[#*`~[\]]/g, "")
|
|
76
|
+
.replace(/\n+/g, " ")
|
|
77
|
+
.trim()}
|
|
78
|
+
</div>
|
|
79
|
+
{/if}
|
|
80
|
+
</button>
|
|
81
|
+
</BlockWrapper>
|
|
@@ -0,0 +1,88 @@
|
|
|
1
|
+
<script lang="ts">
|
|
2
|
+
import type { MessageToolUpdate } from "$lib/types/MessageUpdate";
|
|
3
|
+
import ToolUpdate from "./ToolUpdate.svelte";
|
|
4
|
+
import CarbonChevronDown from "~icons/carbon/chevron-down";
|
|
5
|
+
|
|
6
|
+
interface ToolBlock {
|
|
7
|
+
uuid: string;
|
|
8
|
+
updates: MessageToolUpdate[];
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
interface Props {
|
|
12
|
+
step: number;
|
|
13
|
+
tools: ToolBlock[];
|
|
14
|
+
loading?: boolean;
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
let { step, tools, loading = false }: Props = $props();
|
|
18
|
+
|
|
19
|
+
let isCollapsed = $state(false);
|
|
20
|
+
|
|
21
|
+
let allDone = $derived(
|
|
22
|
+
tools.every((t) =>
|
|
23
|
+
t.updates.some(
|
|
24
|
+
(u) =>
|
|
25
|
+
("subtype" in u && u.subtype === "result") || ("subtype" in u && u.subtype === "error")
|
|
26
|
+
)
|
|
27
|
+
)
|
|
28
|
+
);
|
|
29
|
+
|
|
30
|
+
// Auto-collapse completed groups after 1.5s
|
|
31
|
+
$effect(() => {
|
|
32
|
+
if (allDone && !loading) {
|
|
33
|
+
const timer = setTimeout(() => {
|
|
34
|
+
isCollapsed = true;
|
|
35
|
+
}, 1500);
|
|
36
|
+
return () => clearTimeout(timer);
|
|
37
|
+
}
|
|
38
|
+
});
|
|
39
|
+
</script>
|
|
40
|
+
|
|
41
|
+
<div
|
|
42
|
+
class="my-2 rounded-lg border {allDone
|
|
43
|
+
? 'border-emerald-200 bg-emerald-50/50 dark:border-emerald-800/50 dark:bg-emerald-950/20'
|
|
44
|
+
: 'border-blue-200 bg-blue-50/50 dark:border-blue-800/50 dark:bg-blue-950/20'}"
|
|
45
|
+
data-exclude-from-copy
|
|
46
|
+
>
|
|
47
|
+
<!-- Header -->
|
|
48
|
+
<button
|
|
49
|
+
onclick={() => (isCollapsed = !isCollapsed)}
|
|
50
|
+
class="flex w-full items-center gap-2 px-3 py-1.5 text-xs font-medium {allDone
|
|
51
|
+
? 'text-emerald-700 dark:text-emerald-400'
|
|
52
|
+
: 'text-blue-700 dark:text-blue-400'}"
|
|
53
|
+
>
|
|
54
|
+
<span
|
|
55
|
+
class="flex size-5 items-center justify-center rounded-full text-[10px] font-bold text-white {allDone
|
|
56
|
+
? 'bg-emerald-500'
|
|
57
|
+
: 'bg-blue-500'}"
|
|
58
|
+
>
|
|
59
|
+
{step}
|
|
60
|
+
</span>
|
|
61
|
+
|
|
62
|
+
{#if loading && !allDone}
|
|
63
|
+
<span class="inline-block size-3 animate-spin rounded-full border-2 border-blue-400 border-t-transparent"></span>
|
|
64
|
+
<span>Running {tools.length} tool{tools.length > 1 ? "s" : ""} in parallel...</span>
|
|
65
|
+
{:else if allDone}
|
|
66
|
+
<span>Step {step} — {tools.length} tool{tools.length > 1 ? "s" : ""} completed</span>
|
|
67
|
+
{:else}
|
|
68
|
+
<span>Step {step} — {tools.length} tool{tools.length > 1 ? "s" : ""}</span>
|
|
69
|
+
{/if}
|
|
70
|
+
|
|
71
|
+
<CarbonChevronDown
|
|
72
|
+
class="ml-auto size-3.5 transition-transform {isCollapsed ? '-rotate-90' : ''}"
|
|
73
|
+
/>
|
|
74
|
+
</button>
|
|
75
|
+
|
|
76
|
+
<!-- Content -->
|
|
77
|
+
{#if !isCollapsed}
|
|
78
|
+
<div class="space-y-1 px-2 pb-2">
|
|
79
|
+
{#each tools as tool, i}
|
|
80
|
+
<ToolUpdate
|
|
81
|
+
tool={tool.updates}
|
|
82
|
+
{loading}
|
|
83
|
+
hasNext={i < tools.length - 1}
|
|
84
|
+
/>
|
|
85
|
+
{/each}
|
|
86
|
+
</div>
|
|
87
|
+
{/if}
|
|
88
|
+
</div>
|