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,72 @@
|
|
|
1
|
+
import { Address4, Address6 } from "ip-address";
|
|
2
|
+
import { isIP } from "node:net";
|
|
3
|
+
|
|
4
|
+
const UNSAFE_IPV4_SUBNETS = [
|
|
5
|
+
"0.0.0.0/8",
|
|
6
|
+
"100.64.0.0/10",
|
|
7
|
+
"127.0.0.0/8",
|
|
8
|
+
"169.254.0.0/16",
|
|
9
|
+
"172.16.0.0/12",
|
|
10
|
+
"192.168.0.0/16",
|
|
11
|
+
].map((s) => new Address4(s));
|
|
12
|
+
|
|
13
|
+
function isUnsafeIp(address: string): boolean {
|
|
14
|
+
const family = isIP(address);
|
|
15
|
+
|
|
16
|
+
if (family === 4) {
|
|
17
|
+
const addr = new Address4(address);
|
|
18
|
+
return UNSAFE_IPV4_SUBNETS.some((subnet) => addr.isInSubnet(subnet));
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
if (family === 6) {
|
|
22
|
+
const addr = new Address6(address);
|
|
23
|
+
// Check IPv4-mapped IPv6 addresses (e.g. ::ffff:127.0.0.1)
|
|
24
|
+
if (addr.is4()) {
|
|
25
|
+
const v4 = addr.to4();
|
|
26
|
+
return UNSAFE_IPV4_SUBNETS.some((subnet) => v4.isInSubnet(subnet));
|
|
27
|
+
}
|
|
28
|
+
return addr.isLoopback() || addr.isLinkLocal();
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
return true; // Unknown format → block
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
/**
|
|
35
|
+
* Synchronous URL validation: checks protocol and hostname string.
|
|
36
|
+
*/
|
|
37
|
+
export function isValidUrl(urlString: string): boolean {
|
|
38
|
+
try {
|
|
39
|
+
const url = new URL(urlString.trim());
|
|
40
|
+
const hostname = url.hostname.toLowerCase();
|
|
41
|
+
// Allow HTTP for localhost/loopback (dev & local MCP bridge)
|
|
42
|
+
if (hostname === "localhost" || hostname === "127.0.0.1" || hostname === "::1") {
|
|
43
|
+
return url.protocol === "http:" || url.protocol === "https:";
|
|
44
|
+
}
|
|
45
|
+
// Allow HTTP for Docker-internal service names (no dots = private network)
|
|
46
|
+
if (!hostname.includes(".") && url.protocol === "http:") {
|
|
47
|
+
return true;
|
|
48
|
+
}
|
|
49
|
+
if (url.protocol !== "https:") {
|
|
50
|
+
return false;
|
|
51
|
+
}
|
|
52
|
+
// If the hostname is a raw IP literal, validate it
|
|
53
|
+
const cleanHostname = hostname.replace(/^\[|]$/g, "");
|
|
54
|
+
if (isIP(cleanHostname)) {
|
|
55
|
+
return !isUnsafeIp(cleanHostname);
|
|
56
|
+
}
|
|
57
|
+
return true;
|
|
58
|
+
} catch {
|
|
59
|
+
return false;
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
/**
|
|
64
|
+
* Assert that a resolved IP address is safe (not internal/private).
|
|
65
|
+
* Throws if the IP is internal. Used in undici's custom DNS lookup
|
|
66
|
+
* to validate IPs at connection time (prevents TOCTOU DNS rebinding).
|
|
67
|
+
*/
|
|
68
|
+
export function assertSafeIp(address: string, hostname: string): void {
|
|
69
|
+
if (isUnsafeIp(address)) {
|
|
70
|
+
throw new Error(`Resolved IP for ${hostname} is internal (${address})`);
|
|
71
|
+
}
|
|
72
|
+
}
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
import { z } from "zod";
|
|
2
|
+
import { config } from "$lib/server/config";
|
|
3
|
+
import JSON5 from "json5";
|
|
4
|
+
|
|
5
|
+
const sanitizeJSONEnv = (val: string, fallback: string) => {
|
|
6
|
+
const raw = (val ?? "").trim();
|
|
7
|
+
const unquoted = raw.startsWith("`") && raw.endsWith("`") ? raw.slice(1, -1) : raw;
|
|
8
|
+
return unquoted || fallback;
|
|
9
|
+
};
|
|
10
|
+
|
|
11
|
+
// RATE_LIMIT is the legacy way to define messages per minute limit
|
|
12
|
+
export const usageLimitsSchema = z
|
|
13
|
+
.object({
|
|
14
|
+
conversations: z.coerce.number().optional(), // how many conversations
|
|
15
|
+
messages: z.coerce.number().optional(), // how many messages in a conversation
|
|
16
|
+
messageLength: z.coerce.number().optional(), // how long can a message be before we cut it off
|
|
17
|
+
messagesPerMinute: z
|
|
18
|
+
.preprocess((val) => {
|
|
19
|
+
if (val === undefined) {
|
|
20
|
+
return config.RATE_LIMIT;
|
|
21
|
+
}
|
|
22
|
+
return val;
|
|
23
|
+
}, z.coerce.number().optional())
|
|
24
|
+
.optional(), // how many messages per minute
|
|
25
|
+
})
|
|
26
|
+
.optional();
|
|
27
|
+
|
|
28
|
+
export const usageLimits = usageLimitsSchema.parse(
|
|
29
|
+
JSON5.parse(sanitizeJSONEnv(config.USAGE_LIMITS, "{}"))
|
|
30
|
+
);
|
|
@@ -0,0 +1,175 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Autopilot Store — Svelte 5 runes-based store for managing autopilot Web Worker state.
|
|
3
|
+
*
|
|
4
|
+
* Provides reactive state for autopilot groups, tasks, and text content.
|
|
5
|
+
* Communicates with AutopilotWorker and DetailFetchWorker via postMessage.
|
|
6
|
+
*
|
|
7
|
+
* ADR-037 Part 2+3: Parallel Task UI + Web Workers
|
|
8
|
+
*/
|
|
9
|
+
|
|
10
|
+
import type { GroupState, AutopilotUIUpdate } from "$lib/workers/autopilotWorker";
|
|
11
|
+
import type { DetailWorkerOutgoing } from "$lib/workers/detailFetchWorker";
|
|
12
|
+
|
|
13
|
+
export interface AutopilotState {
|
|
14
|
+
active: boolean;
|
|
15
|
+
maxSteps: number;
|
|
16
|
+
groups: GroupState[];
|
|
17
|
+
textContent: string;
|
|
18
|
+
error: string | null;
|
|
19
|
+
totalSteps: number;
|
|
20
|
+
totalTasks: number;
|
|
21
|
+
duration: number;
|
|
22
|
+
paused: boolean;
|
|
23
|
+
pauseReason: string | null;
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
const defaultState: AutopilotState = {
|
|
27
|
+
active: false,
|
|
28
|
+
maxSteps: 20,
|
|
29
|
+
groups: [],
|
|
30
|
+
textContent: "",
|
|
31
|
+
error: null,
|
|
32
|
+
totalSteps: 0,
|
|
33
|
+
totalTasks: 0,
|
|
34
|
+
duration: 0,
|
|
35
|
+
paused: false,
|
|
36
|
+
pauseReason: null,
|
|
37
|
+
};
|
|
38
|
+
|
|
39
|
+
let state = $state<AutopilotState>({ ...defaultState });
|
|
40
|
+
|
|
41
|
+
let autopilotWorker: Worker | null = null;
|
|
42
|
+
let detailWorker: Worker | null = null;
|
|
43
|
+
const detailCallbacks = new Map<string, (content: string | null, error?: string) => void>();
|
|
44
|
+
|
|
45
|
+
async function ensureWorkers() {
|
|
46
|
+
if (typeof window === "undefined") return;
|
|
47
|
+
|
|
48
|
+
if (!autopilotWorker) {
|
|
49
|
+
const mod = await import("$lib/workers/autopilotWorker?worker");
|
|
50
|
+
autopilotWorker = new mod.default();
|
|
51
|
+
autopilotWorker.onmessage = handleWorkerMessage;
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
if (!detailWorker) {
|
|
55
|
+
const mod = await import("$lib/workers/detailFetchWorker?worker");
|
|
56
|
+
detailWorker = new mod.default();
|
|
57
|
+
detailWorker.onmessage = handleDetailMessage;
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
function handleWorkerMessage(e: MessageEvent) {
|
|
62
|
+
const msg = e.data;
|
|
63
|
+
|
|
64
|
+
switch (msg.type) {
|
|
65
|
+
case "batch_update":
|
|
66
|
+
state.groups = msg.groups;
|
|
67
|
+
for (const update of msg.updates as AutopilotUIUpdate[]) {
|
|
68
|
+
applyUpdate(update);
|
|
69
|
+
}
|
|
70
|
+
break;
|
|
71
|
+
|
|
72
|
+
case "text":
|
|
73
|
+
state.textContent += msg.content;
|
|
74
|
+
break;
|
|
75
|
+
|
|
76
|
+
case "done":
|
|
77
|
+
state.active = false;
|
|
78
|
+
state.groups = msg.groups;
|
|
79
|
+
break;
|
|
80
|
+
|
|
81
|
+
case "error":
|
|
82
|
+
state.active = false;
|
|
83
|
+
state.error = msg.error;
|
|
84
|
+
break;
|
|
85
|
+
|
|
86
|
+
case "stopped":
|
|
87
|
+
state.active = false;
|
|
88
|
+
state.groups = msg.groups;
|
|
89
|
+
break;
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
function applyUpdate(update: AutopilotUIUpdate) {
|
|
94
|
+
switch (update.type) {
|
|
95
|
+
case "start":
|
|
96
|
+
state.maxSteps = update.maxSteps;
|
|
97
|
+
break;
|
|
98
|
+
case "end":
|
|
99
|
+
state.totalSteps = update.totalSteps;
|
|
100
|
+
state.totalTasks = update.totalTasks;
|
|
101
|
+
state.duration = update.duration;
|
|
102
|
+
break;
|
|
103
|
+
case "text":
|
|
104
|
+
state.textContent += update.content;
|
|
105
|
+
break;
|
|
106
|
+
case "paused":
|
|
107
|
+
state.paused = true;
|
|
108
|
+
state.pauseReason = update.reason;
|
|
109
|
+
break;
|
|
110
|
+
case "error_event":
|
|
111
|
+
state.error = update.error;
|
|
112
|
+
break;
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
function handleDetailMessage(e: MessageEvent<DetailWorkerOutgoing>) {
|
|
117
|
+
const msg = e.data;
|
|
118
|
+
if (msg.type === "detail") {
|
|
119
|
+
const cb = detailCallbacks.get(msg.detailToken);
|
|
120
|
+
if (cb) {
|
|
121
|
+
cb(msg.content);
|
|
122
|
+
detailCallbacks.delete(msg.detailToken);
|
|
123
|
+
}
|
|
124
|
+
} else if (msg.type === "detail_error") {
|
|
125
|
+
const cb = detailCallbacks.get(msg.detailToken);
|
|
126
|
+
if (cb) {
|
|
127
|
+
cb(null, msg.error);
|
|
128
|
+
detailCallbacks.delete(msg.detailToken);
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
export function useAutopilot() {
|
|
134
|
+
return {
|
|
135
|
+
get state() {
|
|
136
|
+
return state;
|
|
137
|
+
},
|
|
138
|
+
|
|
139
|
+
async start(url: string, headers: Record<string, string>, body: unknown) {
|
|
140
|
+
await ensureWorkers();
|
|
141
|
+
Object.assign(state, { ...defaultState, active: true });
|
|
142
|
+
autopilotWorker?.postMessage({ type: "start", url, headers, body });
|
|
143
|
+
},
|
|
144
|
+
|
|
145
|
+
stop() {
|
|
146
|
+
autopilotWorker?.postMessage({ type: "stop" });
|
|
147
|
+
},
|
|
148
|
+
|
|
149
|
+
async fetchDetail(detailToken: string, bridgeUrl: string): Promise<string> {
|
|
150
|
+
await ensureWorkers();
|
|
151
|
+
return new Promise((resolve, reject) => {
|
|
152
|
+
detailCallbacks.set(detailToken, (content, error) => {
|
|
153
|
+
if (error) reject(new Error(error));
|
|
154
|
+
else resolve(content!);
|
|
155
|
+
});
|
|
156
|
+
detailWorker?.postMessage({ type: "fetch", detailToken, bridgeUrl });
|
|
157
|
+
});
|
|
158
|
+
},
|
|
159
|
+
|
|
160
|
+
prefetchDetail(detailToken: string, bridgeUrl: string) {
|
|
161
|
+
detailWorker?.postMessage({ type: "prefetch", detailToken, bridgeUrl });
|
|
162
|
+
},
|
|
163
|
+
|
|
164
|
+
evictDetail(detailToken: string) {
|
|
165
|
+
detailWorker?.postMessage({ type: "evict", detailToken });
|
|
166
|
+
},
|
|
167
|
+
|
|
168
|
+
destroy() {
|
|
169
|
+
autopilotWorker?.terminate();
|
|
170
|
+
detailWorker?.terminate();
|
|
171
|
+
autopilotWorker = null;
|
|
172
|
+
detailWorker = null;
|
|
173
|
+
},
|
|
174
|
+
};
|
|
175
|
+
}
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
export type BackgroundGeneration = {
|
|
2
|
+
id: string;
|
|
3
|
+
startedAt: number;
|
|
4
|
+
};
|
|
5
|
+
|
|
6
|
+
export const backgroundGenerationEntries = $state<BackgroundGeneration[]>([]);
|
|
7
|
+
|
|
8
|
+
export function addBackgroundGeneration(entry: BackgroundGeneration) {
|
|
9
|
+
const index = backgroundGenerationEntries.findIndex(({ id }) => id === entry.id);
|
|
10
|
+
|
|
11
|
+
if (index === -1) {
|
|
12
|
+
backgroundGenerationEntries.push(entry);
|
|
13
|
+
return;
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
backgroundGenerationEntries[index] = entry;
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
export function removeBackgroundGeneration(id: string) {
|
|
20
|
+
const index = backgroundGenerationEntries.findIndex((entry) => entry.id === id);
|
|
21
|
+
if (index === -1) return;
|
|
22
|
+
|
|
23
|
+
backgroundGenerationEntries.splice(index, 1);
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
export function clearBackgroundGenerations() {
|
|
27
|
+
backgroundGenerationEntries.length = 0;
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
export function hasBackgroundGeneration(id: string) {
|
|
31
|
+
return backgroundGenerationEntries.some((entry) => entry.id === id);
|
|
32
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export * from "./backgroundGenerations.svelte";
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
import { writable } from "svelte/store";
|
|
2
|
+
|
|
3
|
+
export const ERROR_MESSAGES = {
|
|
4
|
+
default: "Oops, something went wrong.",
|
|
5
|
+
authOnly: "You have to be logged in.",
|
|
6
|
+
rateLimited: "You are sending too many messages. Try again later.",
|
|
7
|
+
};
|
|
8
|
+
|
|
9
|
+
export const error = writable<string | undefined>(undefined);
|
|
@@ -0,0 +1,345 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* MCP Servers Store
|
|
3
|
+
* Manages base (env-configured) and custom (user-added) MCP servers
|
|
4
|
+
* Stores custom servers and selection state in browser localStorage
|
|
5
|
+
*/
|
|
6
|
+
|
|
7
|
+
import { writable, derived, get } from "svelte/store";
|
|
8
|
+
import { base } from "$app/paths";
|
|
9
|
+
import { env as publicEnv } from "$env/dynamic/public";
|
|
10
|
+
import { browser } from "$app/environment";
|
|
11
|
+
import type { MCPServer, ServerStatus, MCPTool } from "$lib/types/Tool";
|
|
12
|
+
|
|
13
|
+
// Namespace storage by app identity to avoid collisions across apps
|
|
14
|
+
function toKeyPart(s: string | undefined): string {
|
|
15
|
+
return (s || "").toLowerCase().replace(/[^a-z0-9_-]+/g, "-");
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
const appLabel = toKeyPart(publicEnv.PUBLIC_APP_ASSETS || publicEnv.PUBLIC_APP_NAME);
|
|
19
|
+
const baseLabel = toKeyPart(typeof base === "string" ? base : "");
|
|
20
|
+
// Final prefix format requested: "huggingchat:key" (no mcp:/chat)
|
|
21
|
+
const KEY_PREFIX = appLabel || baseLabel || "app";
|
|
22
|
+
|
|
23
|
+
const STORAGE_KEYS = {
|
|
24
|
+
CUSTOM_SERVERS: `${KEY_PREFIX}:mcp:custom-servers`,
|
|
25
|
+
SELECTED_IDS: `${KEY_PREFIX}:mcp:selected-ids`,
|
|
26
|
+
DISABLED_BASE_IDS: `${KEY_PREFIX}:mcp:disabled-base-ids`,
|
|
27
|
+
} as const;
|
|
28
|
+
|
|
29
|
+
// No migration needed per request — read/write only namespaced keys
|
|
30
|
+
|
|
31
|
+
// Load custom servers from localStorage
|
|
32
|
+
function loadCustomServers(): MCPServer[] {
|
|
33
|
+
if (!browser) return [];
|
|
34
|
+
|
|
35
|
+
try {
|
|
36
|
+
const json = localStorage.getItem(STORAGE_KEYS.CUSTOM_SERVERS);
|
|
37
|
+
return json ? JSON.parse(json) : [];
|
|
38
|
+
} catch (error) {
|
|
39
|
+
console.error("Failed to load custom MCP servers from localStorage:", error);
|
|
40
|
+
return [];
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
// Load selected server IDs from localStorage
|
|
45
|
+
function loadSelectedIds(): Set<string> {
|
|
46
|
+
if (!browser) return new Set();
|
|
47
|
+
|
|
48
|
+
try {
|
|
49
|
+
const json = localStorage.getItem(STORAGE_KEYS.SELECTED_IDS);
|
|
50
|
+
const ids: string[] = json ? JSON.parse(json) : [];
|
|
51
|
+
return new Set(ids);
|
|
52
|
+
} catch (error) {
|
|
53
|
+
console.error("Failed to load selected MCP server IDs from localStorage:", error);
|
|
54
|
+
return new Set();
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
// Save custom servers to localStorage
|
|
59
|
+
function saveCustomServers(servers: MCPServer[]) {
|
|
60
|
+
if (!browser) return;
|
|
61
|
+
|
|
62
|
+
try {
|
|
63
|
+
localStorage.setItem(STORAGE_KEYS.CUSTOM_SERVERS, JSON.stringify(servers));
|
|
64
|
+
} catch (error) {
|
|
65
|
+
console.error("Failed to save custom MCP servers to localStorage:", error);
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
// Save selected IDs to localStorage
|
|
70
|
+
function saveSelectedIds(ids: Set<string>) {
|
|
71
|
+
if (!browser) return;
|
|
72
|
+
|
|
73
|
+
try {
|
|
74
|
+
localStorage.setItem(STORAGE_KEYS.SELECTED_IDS, JSON.stringify([...ids]));
|
|
75
|
+
} catch (error) {
|
|
76
|
+
console.error("Failed to save selected MCP server IDs to localStorage:", error);
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
// Load disabled base server IDs from localStorage (empty set if missing or on error)
|
|
81
|
+
function loadDisabledBaseIds(): Set<string> {
|
|
82
|
+
if (!browser) return new Set();
|
|
83
|
+
|
|
84
|
+
try {
|
|
85
|
+
const json = localStorage.getItem(STORAGE_KEYS.DISABLED_BASE_IDS);
|
|
86
|
+
return new Set(json ? JSON.parse(json) : []);
|
|
87
|
+
} catch (error) {
|
|
88
|
+
console.error("Failed to load disabled base MCP server IDs from localStorage:", error);
|
|
89
|
+
return new Set();
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
// Save disabled base server IDs to localStorage
|
|
94
|
+
function saveDisabledBaseIds(ids: Set<string>) {
|
|
95
|
+
if (!browser) return;
|
|
96
|
+
|
|
97
|
+
try {
|
|
98
|
+
localStorage.setItem(STORAGE_KEYS.DISABLED_BASE_IDS, JSON.stringify([...ids]));
|
|
99
|
+
} catch (error) {
|
|
100
|
+
console.error("Failed to save disabled base MCP server IDs to localStorage:", error);
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
// Store for all servers (base + custom)
|
|
105
|
+
export const allMcpServers = writable<MCPServer[]>([]);
|
|
106
|
+
|
|
107
|
+
// Track if initial server load has completed
|
|
108
|
+
export const mcpServersLoaded = writable<boolean>(false);
|
|
109
|
+
|
|
110
|
+
// Store for selected server IDs
|
|
111
|
+
export const selectedServerIds = writable<Set<string>>(loadSelectedIds());
|
|
112
|
+
|
|
113
|
+
// Auto-persist selected IDs when they change
|
|
114
|
+
if (browser) {
|
|
115
|
+
selectedServerIds.subscribe((ids) => {
|
|
116
|
+
saveSelectedIds(ids);
|
|
117
|
+
});
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
// Derived store: only enabled servers
|
|
121
|
+
export const enabledServers = derived([allMcpServers, selectedServerIds], ([$all, $selected]) =>
|
|
122
|
+
$all.filter((s) => $selected.has(s.id))
|
|
123
|
+
);
|
|
124
|
+
|
|
125
|
+
// Derived store: count of enabled servers
|
|
126
|
+
export const enabledServersCount = derived(enabledServers, ($enabled) => $enabled.length);
|
|
127
|
+
|
|
128
|
+
// Derived store: true if all base servers are enabled
|
|
129
|
+
export const allBaseServersEnabled = derived(
|
|
130
|
+
[allMcpServers, selectedServerIds],
|
|
131
|
+
([$all, $selected]) => {
|
|
132
|
+
const baseServers = $all.filter((s) => s.type === "base");
|
|
133
|
+
return baseServers.length > 0 && baseServers.every((s) => $selected.has(s.id));
|
|
134
|
+
}
|
|
135
|
+
);
|
|
136
|
+
|
|
137
|
+
// Note: Authorization overlay (with user's HF token) for the Hugging Face MCP host
|
|
138
|
+
// is applied server-side when enabled via MCP_FORWARD_HF_USER_TOKEN.
|
|
139
|
+
|
|
140
|
+
/**
|
|
141
|
+
* Refresh base servers from API and merge with custom servers
|
|
142
|
+
*/
|
|
143
|
+
export async function refreshMcpServers() {
|
|
144
|
+
try {
|
|
145
|
+
const response = await fetch(`${base}/api/mcp/servers`);
|
|
146
|
+
if (!response.ok) {
|
|
147
|
+
throw new Error(`Failed to fetch base servers: ${response.statusText}`);
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
const baseServers: MCPServer[] = await response.json();
|
|
151
|
+
const customServers = loadCustomServers();
|
|
152
|
+
|
|
153
|
+
// Merge base and custom servers
|
|
154
|
+
const merged = [...baseServers, ...customServers];
|
|
155
|
+
allMcpServers.set(merged);
|
|
156
|
+
|
|
157
|
+
// Load disabled base servers
|
|
158
|
+
const disabledBaseIds = loadDisabledBaseIds();
|
|
159
|
+
|
|
160
|
+
// Auto-enable all base servers that aren't explicitly disabled
|
|
161
|
+
// Plus keep any custom servers that were previously selected
|
|
162
|
+
const validIds = new Set(merged.map((s) => s.id));
|
|
163
|
+
selectedServerIds.update(($currentIds) => {
|
|
164
|
+
const newSelection = new Set<string>();
|
|
165
|
+
|
|
166
|
+
// Add all base servers that aren't disabled
|
|
167
|
+
for (const server of baseServers) {
|
|
168
|
+
if (!disabledBaseIds.has(server.id)) {
|
|
169
|
+
newSelection.add(server.id);
|
|
170
|
+
}
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
// Keep custom servers that were selected and still exist
|
|
174
|
+
for (const id of $currentIds) {
|
|
175
|
+
if (validIds.has(id) && !id.startsWith("base-")) {
|
|
176
|
+
newSelection.add(id);
|
|
177
|
+
}
|
|
178
|
+
}
|
|
179
|
+
|
|
180
|
+
return newSelection;
|
|
181
|
+
});
|
|
182
|
+
mcpServersLoaded.set(true);
|
|
183
|
+
} catch (error) {
|
|
184
|
+
console.error("Failed to refresh MCP servers:", error);
|
|
185
|
+
// On error, just use custom servers
|
|
186
|
+
allMcpServers.set(loadCustomServers());
|
|
187
|
+
mcpServersLoaded.set(true);
|
|
188
|
+
}
|
|
189
|
+
}
|
|
190
|
+
|
|
191
|
+
/**
|
|
192
|
+
* Toggle a server on/off
|
|
193
|
+
*/
|
|
194
|
+
export function toggleServer(id: string) {
|
|
195
|
+
selectedServerIds.update(($ids) => {
|
|
196
|
+
const newSet = new Set($ids);
|
|
197
|
+
if (newSet.has(id)) {
|
|
198
|
+
newSet.delete(id);
|
|
199
|
+
// Track if this is a base server being disabled
|
|
200
|
+
if (id.startsWith("base-")) {
|
|
201
|
+
const disabled = loadDisabledBaseIds();
|
|
202
|
+
disabled.add(id);
|
|
203
|
+
saveDisabledBaseIds(disabled);
|
|
204
|
+
}
|
|
205
|
+
} else {
|
|
206
|
+
newSet.add(id);
|
|
207
|
+
// Remove from disabled if re-enabling a base server
|
|
208
|
+
if (id.startsWith("base-")) {
|
|
209
|
+
const disabled = loadDisabledBaseIds();
|
|
210
|
+
disabled.delete(id);
|
|
211
|
+
saveDisabledBaseIds(disabled);
|
|
212
|
+
}
|
|
213
|
+
}
|
|
214
|
+
return newSet;
|
|
215
|
+
});
|
|
216
|
+
}
|
|
217
|
+
|
|
218
|
+
/**
|
|
219
|
+
* Disable all MCP servers (marks all base servers as disabled)
|
|
220
|
+
*/
|
|
221
|
+
export function disableAllServers() {
|
|
222
|
+
// Get current base server IDs and mark them all as disabled
|
|
223
|
+
const servers = get(allMcpServers);
|
|
224
|
+
const baseServerIds = servers.filter((s) => s.type === "base").map((s) => s.id);
|
|
225
|
+
|
|
226
|
+
// Save all base servers as disabled
|
|
227
|
+
saveDisabledBaseIds(new Set(baseServerIds));
|
|
228
|
+
|
|
229
|
+
// Clear the selection
|
|
230
|
+
selectedServerIds.set(new Set());
|
|
231
|
+
}
|
|
232
|
+
|
|
233
|
+
/**
|
|
234
|
+
* Add a custom MCP server
|
|
235
|
+
*/
|
|
236
|
+
export function addCustomServer(server: Omit<MCPServer, "id" | "type" | "status">): string {
|
|
237
|
+
const newServer: MCPServer = {
|
|
238
|
+
...server,
|
|
239
|
+
id: crypto.randomUUID(),
|
|
240
|
+
type: "custom",
|
|
241
|
+
status: "disconnected",
|
|
242
|
+
};
|
|
243
|
+
|
|
244
|
+
const customServers = loadCustomServers();
|
|
245
|
+
customServers.push(newServer);
|
|
246
|
+
saveCustomServers(customServers);
|
|
247
|
+
|
|
248
|
+
// Refresh all servers to include the new one
|
|
249
|
+
refreshMcpServers();
|
|
250
|
+
|
|
251
|
+
return newServer.id;
|
|
252
|
+
}
|
|
253
|
+
|
|
254
|
+
/**
|
|
255
|
+
* Update an existing custom server
|
|
256
|
+
*/
|
|
257
|
+
export function updateCustomServer(id: string, updates: Partial<MCPServer>) {
|
|
258
|
+
const customServers = loadCustomServers();
|
|
259
|
+
const index = customServers.findIndex((s) => s.id === id);
|
|
260
|
+
|
|
261
|
+
if (index !== -1) {
|
|
262
|
+
customServers[index] = { ...customServers[index], ...updates };
|
|
263
|
+
saveCustomServers(customServers);
|
|
264
|
+
refreshMcpServers();
|
|
265
|
+
}
|
|
266
|
+
}
|
|
267
|
+
|
|
268
|
+
/**
|
|
269
|
+
* Delete a custom server
|
|
270
|
+
*/
|
|
271
|
+
export function deleteCustomServer(id: string) {
|
|
272
|
+
const customServers = loadCustomServers();
|
|
273
|
+
const filtered = customServers.filter((s) => s.id !== id);
|
|
274
|
+
saveCustomServers(filtered);
|
|
275
|
+
|
|
276
|
+
// Also remove from selected IDs
|
|
277
|
+
selectedServerIds.update(($ids) => {
|
|
278
|
+
const newSet = new Set($ids);
|
|
279
|
+
newSet.delete(id);
|
|
280
|
+
return newSet;
|
|
281
|
+
});
|
|
282
|
+
|
|
283
|
+
refreshMcpServers();
|
|
284
|
+
}
|
|
285
|
+
|
|
286
|
+
/**
|
|
287
|
+
* Update server status (from health check)
|
|
288
|
+
*/
|
|
289
|
+
export function updateServerStatus(
|
|
290
|
+
id: string,
|
|
291
|
+
status: ServerStatus,
|
|
292
|
+
errorMessage?: string,
|
|
293
|
+
tools?: MCPTool[],
|
|
294
|
+
authRequired?: boolean
|
|
295
|
+
) {
|
|
296
|
+
allMcpServers.update(($servers) =>
|
|
297
|
+
$servers.map((s) =>
|
|
298
|
+
s.id === id
|
|
299
|
+
? {
|
|
300
|
+
...s,
|
|
301
|
+
status,
|
|
302
|
+
errorMessage,
|
|
303
|
+
tools,
|
|
304
|
+
authRequired,
|
|
305
|
+
}
|
|
306
|
+
: s
|
|
307
|
+
)
|
|
308
|
+
);
|
|
309
|
+
}
|
|
310
|
+
|
|
311
|
+
/**
|
|
312
|
+
* Run health check on a server
|
|
313
|
+
*/
|
|
314
|
+
export async function healthCheckServer(
|
|
315
|
+
server: MCPServer
|
|
316
|
+
): Promise<{ ready: boolean; tools?: MCPTool[]; error?: string }> {
|
|
317
|
+
try {
|
|
318
|
+
updateServerStatus(server.id, "connecting");
|
|
319
|
+
|
|
320
|
+
const response = await fetch(`${base}/api/mcp/health`, {
|
|
321
|
+
method: "POST",
|
|
322
|
+
headers: { "Content-Type": "application/json" },
|
|
323
|
+
body: JSON.stringify({ url: server.url, headers: server.headers }),
|
|
324
|
+
});
|
|
325
|
+
|
|
326
|
+
const result = await response.json();
|
|
327
|
+
|
|
328
|
+
if (result.ready && result.tools) {
|
|
329
|
+
updateServerStatus(server.id, "connected", undefined, result.tools, false);
|
|
330
|
+
return { ready: true, tools: result.tools };
|
|
331
|
+
} else {
|
|
332
|
+
updateServerStatus(server.id, "error", result.error, undefined, Boolean(result.authRequired));
|
|
333
|
+
return { ready: false, error: result.error };
|
|
334
|
+
}
|
|
335
|
+
} catch (error) {
|
|
336
|
+
const errorMessage = error instanceof Error ? error.message : "Unknown error";
|
|
337
|
+
updateServerStatus(server.id, "error", errorMessage);
|
|
338
|
+
return { ready: false, error: errorMessage };
|
|
339
|
+
}
|
|
340
|
+
}
|
|
341
|
+
|
|
342
|
+
// Initialize on module load
|
|
343
|
+
if (browser) {
|
|
344
|
+
refreshMcpServers();
|
|
345
|
+
}
|