@mseep/open-computer-use 1.0.0
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/.coderabbit.yaml +25 -0
- package/.dockerignore +95 -0
- package/.env.example +137 -0
- package/.githooks/pre-commit +68 -0
- package/.github/CODEOWNERS +125 -0
- package/.github/ISSUE_TEMPLATE/adr-proposal.md +41 -0
- package/.github/ISSUE_TEMPLATE/bug-report.md +49 -0
- package/.github/ISSUE_TEMPLATE/component-proposal.md +38 -0
- package/.github/ISSUE_TEMPLATE/config.yml +15 -0
- package/.github/ISSUE_TEMPLATE/dependency-proposal.md +59 -0
- package/.github/ISSUE_TEMPLATE/feature_request.md +15 -0
- package/.github/ISSUE_TEMPLATE/nfr-proposal.md +44 -0
- package/.github/PULL_REQUEST_TEMPLATE.md +15 -0
- package/.github/codeql/codeql-config.yml +11 -0
- package/.github/codeql/extensions/security-models/python-sanitizers.model.yml +17 -0
- package/.github/codeql/extensions/security-models/qlpack.yml +7 -0
- package/.github/dependabot.yml +23 -0
- package/.github/security-exceptions.yml +23 -0
- package/.github/workflows/build.yml +420 -0
- package/.github/workflows/codeql.yml +33 -0
- package/.github/workflows/contracts-lint.yml +90 -0
- package/.github/workflows/docs-lint.yml +151 -0
- package/.github/workflows/helm.yml +131 -0
- package/.github/workflows/identity-lint.yml +30 -0
- package/.github/workflows/release-chart.yml +177 -0
- package/.github/workflows/release.yml +95 -0
- package/.github/workflows/security.yml +332 -0
- package/.github/workflows/stale.yml +31 -0
- package/.github/workflows/supply-chain.yml +242 -0
- package/.gitleaks.toml +53 -0
- package/.markdownlint.yaml +51 -0
- package/.semgrepignore +85 -0
- package/.vale/styles/Architecture/ap13-data-class-substrate.yml +12 -0
- package/.vale/styles/Architecture/banned-phrases.yml +23 -0
- package/.vale/styles/Architecture/banned-vocab.yml +23 -0
- package/.vale/styles/Architecture/marketing-tone.yml +19 -0
- package/.vale.ini +18 -0
- package/CHANGELOG.md +411 -0
- package/CLAUDE.md +218 -0
- package/CONTRIBUTING.md +82 -0
- package/Dockerfile +676 -0
- package/LICENSE +98 -0
- package/LICENSE-APACHE +202 -0
- package/LICENSE-MIT +21 -0
- package/NOTICE +36 -0
- package/README.md +516 -0
- package/SECURITY.md +45 -0
- package/THIRD-PARTY-LICENSES.md +14 -0
- package/apt-packages.txt +108 -0
- package/computer-use-server/.dockerignore +13 -0
- package/computer-use-server/Dockerfile +44 -0
- package/computer-use-server/README.md +84 -0
- package/computer-use-server/app.py +1544 -0
- package/computer-use-server/bin/list-subagent-models +449 -0
- package/computer-use-server/cli-defaults/README.md +31 -0
- package/computer-use-server/cli-defaults/codex.json +7 -0
- package/computer-use-server/cli-defaults/opencode.json +18 -0
- package/computer-use-server/cli_adapters/__init__.py +46 -0
- package/computer-use-server/cli_adapters/claude.py +163 -0
- package/computer-use-server/cli_adapters/codex.py +163 -0
- package/computer-use-server/cli_adapters/opencode.py +169 -0
- package/computer-use-server/cli_adapters/result.py +34 -0
- package/computer-use-server/cli_runtime.py +316 -0
- package/computer-use-server/context_vars.py +24 -0
- package/computer-use-server/docker_manager.py +1100 -0
- package/computer-use-server/docs_html.py +12 -0
- package/computer-use-server/mcp_resources.py +170 -0
- package/computer-use-server/mcp_tools.py +1430 -0
- package/computer-use-server/requirements.txt +17 -0
- package/computer-use-server/security.py +50 -0
- package/computer-use-server/skill_manager.py +664 -0
- package/computer-use-server/static/browser-viewer.js +445 -0
- package/computer-use-server/static/chart.umd.js +14 -0
- package/computer-use-server/static/docs.html +203 -0
- package/computer-use-server/static/github-dark.min.css +10 -0
- package/computer-use-server/static/github.min.css +10 -0
- package/computer-use-server/static/highlight.min.js +1213 -0
- package/computer-use-server/static/highlightjs-line-numbers.min.js +1 -0
- package/computer-use-server/static/icons.js +74 -0
- package/computer-use-server/static/jszip.min.js +13 -0
- package/computer-use-server/static/katex/auto-render.min.js +1 -0
- package/computer-use-server/static/katex/fonts/KaTeX_AMS-Regular.ttf +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_AMS-Regular.woff +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_AMS-Regular.woff2 +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Caligraphic-Bold.ttf +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Caligraphic-Bold.woff +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Caligraphic-Bold.woff2 +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Caligraphic-Regular.ttf +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Caligraphic-Regular.woff +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Caligraphic-Regular.woff2 +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Fraktur-Bold.ttf +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Fraktur-Bold.woff +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Fraktur-Bold.woff2 +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Fraktur-Regular.ttf +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Fraktur-Regular.woff +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Fraktur-Regular.woff2 +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Main-Bold.ttf +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Main-Bold.woff +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Main-Bold.woff2 +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Main-BoldItalic.ttf +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Main-BoldItalic.woff +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Main-BoldItalic.woff2 +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Main-Italic.ttf +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Main-Italic.woff +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Main-Italic.woff2 +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Main-Regular.ttf +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Main-Regular.woff +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Main-Regular.woff2 +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Math-BoldItalic.ttf +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Math-BoldItalic.woff +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Math-BoldItalic.woff2 +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Math-Italic.ttf +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Math-Italic.woff +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Math-Italic.woff2 +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_SansSerif-Bold.ttf +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_SansSerif-Bold.woff +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_SansSerif-Bold.woff2 +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_SansSerif-Italic.ttf +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_SansSerif-Italic.woff +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_SansSerif-Italic.woff2 +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_SansSerif-Regular.ttf +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_SansSerif-Regular.woff +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_SansSerif-Regular.woff2 +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Script-Regular.ttf +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Script-Regular.woff +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Script-Regular.woff2 +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Size1-Regular.ttf +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Size1-Regular.woff +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Size1-Regular.woff2 +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Size2-Regular.ttf +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Size2-Regular.woff +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Size2-Regular.woff2 +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Size3-Regular.ttf +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Size3-Regular.woff +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Size3-Regular.woff2 +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Size4-Regular.ttf +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Size4-Regular.woff +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Size4-Regular.woff2 +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Typewriter-Regular.ttf +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Typewriter-Regular.woff +0 -0
- package/computer-use-server/static/katex/fonts/KaTeX_Typewriter-Regular.woff2 +0 -0
- package/computer-use-server/static/katex/katex.min.css +1 -0
- package/computer-use-server/static/katex/katex.min.js +1 -0
- package/computer-use-server/static/locale.js +242 -0
- package/computer-use-server/static/mammoth.browser.min.js +21 -0
- package/computer-use-server/static/marked.min.js +6 -0
- package/computer-use-server/static/mermaid.min.js +2811 -0
- package/computer-use-server/static/pdf.min.js +22 -0
- package/computer-use-server/static/pdf.worker.min.js +22 -0
- package/computer-use-server/static/pptxviewjs.min.js +1 -0
- package/computer-use-server/static/preact-htm.min.js +1 -0
- package/computer-use-server/static/preview.css +1030 -0
- package/computer-use-server/static/preview.js +1522 -0
- package/computer-use-server/static/xlsx.full.min.js +22 -0
- package/computer-use-server/static/xterm-addon-fit.min.js +2 -0
- package/computer-use-server/static/xterm-addon-web-links.min.js +2 -0
- package/computer-use-server/static/xterm.css +218 -0
- package/computer-use-server/static/xterm.min.js +2 -0
- package/computer-use-server/system_prompt.py +761 -0
- package/computer-use-server/uploads.py +82 -0
- package/contracts/README.md +53 -0
- package/contracts/audit/audit-fanin.asyncapi.yaml +407 -0
- package/contracts/exec/exec-channel.schema.json +240 -0
- package/contracts/mcp/2025-06-18/ocu-constraints.schema.json +178 -0
- package/contracts/storage/file-artifact-api.schema.json +390 -0
- package/contracts/storage/file-ops.schema.json +217 -0
- package/contracts/storage/mount-config.schema.json +197 -0
- package/cron/Dockerfile +15 -0
- package/cron/cleanup-quick.sh +21 -0
- package/cron/cleanup.sh +127 -0
- package/data/outputs/.gitkeep +0 -0
- package/data/uploads/.gitkeep +0 -0
- package/docker-compose.test.yml +54 -0
- package/docker-compose.webui.yml +77 -0
- package/docker-compose.yml +96 -0
- package/docs/CLOUD.md +29 -0
- package/docs/COMPARISON.md +128 -0
- package/docs/DOCKER.md +469 -0
- package/docs/DYNAMIC-SKILLS.md +77 -0
- package/docs/FEATURES.md +100 -0
- package/docs/INSTALL.md +111 -0
- package/docs/KNOWN-BUGS.md +86 -0
- package/docs/MCP.md +320 -0
- package/docs/SCREENSHOTS.md +39 -0
- package/docs/SKILLS-USER-GUIDE.md +86 -0
- package/docs/SKILLS.md +483 -0
- package/docs/TERMINAL-TAB.md +56 -0
- package/docs/architecture/02-trust-boundaries.md +224 -0
- package/docs/architecture/03-c4-context.md +61 -0
- package/docs/architecture/04-bounded-contexts.md +119 -0
- package/docs/architecture/05-c4-container.md +88 -0
- package/docs/architecture/06-threat-model.md +172 -0
- package/docs/architecture/08-contracts.md +105 -0
- package/docs/architecture/MANIFESTO.md +38 -0
- package/docs/architecture/PROCESS.md +64 -0
- package/docs/architecture/README.md +37 -0
- package/docs/architecture/adr/0000-template.md +65 -0
- package/docs/architecture/adr/0001-layer-0-gate-legacy-exclusion.md +75 -0
- package/docs/architecture/adr/0002-session-view-descriptor.md +57 -0
- package/docs/architecture/adr/0003-sandbox-runtime-tier-ladder.md +63 -0
- package/docs/architecture/adr/0004-operator-authentication-substrate.md +63 -0
- package/docs/architecture/adr/0005-egress-credential-delivery-envoy-sds.md +62 -0
- package/docs/architecture/adr/0006-egress-forward-proxy-substrate.md +65 -0
- package/docs/architecture/adr/0007-egress-auth-mechanism.md +72 -0
- package/docs/architecture/adr/0008-session-egress-attribution.md +59 -0
- package/docs/architecture/adr/0009-audit-pipeline-pluggable-by-contract.md +76 -0
- package/docs/architecture/adr/0010-storage-backend-pluggable-adapter.md +60 -0
- package/docs/architecture/adr/0011-storage-egress-lane.md +67 -0
- package/docs/architecture/adr/0012-implementation-language.md +67 -0
- package/docs/architecture/adr/0020-sandbox-image-provisioning.md +82 -0
- package/docs/architecture/adr/README.md +53 -0
- package/docs/architecture/compliance/.gitkeep +0 -0
- package/docs/architecture/components/00-overview.md +42 -0
- package/docs/architecture/components/0000-template.md +50 -0
- package/docs/architecture/components/01-mcp-gateway.md +80 -0
- package/docs/architecture/components/02-control-operator-api.md +80 -0
- package/docs/architecture/components/04-storage-broker.md +104 -0
- package/docs/architecture/components/05-session-sandbox.md +93 -0
- package/docs/architecture/components/06-egress-trust-edge.md +95 -0
- package/docs/architecture/components/07-audit-pipeline.md +110 -0
- package/docs/architecture/diagrams/.gitkeep +0 -0
- package/docs/architecture/diagrams/02-trust-boundaries.mmd +111 -0
- package/docs/architecture/diagrams/06-threat-model.mmd +41 -0
- package/docs/architecture/diagrams/08-contracts.mmd +47 -0
- package/docs/architecture/diagrams/c4-container.mmd +59 -0
- package/docs/architecture/diagrams/c4-context.mmd +46 -0
- package/docs/architecture/glossary.md +172 -0
- package/docs/architecture/manifesto/.gitkeep +0 -0
- package/docs/architecture/manifesto/01-audience-and-buyer.md +57 -0
- package/docs/architecture/manifesto/02-nfrs.md +325 -0
- package/docs/architecture/manifesto/03-non-negotiables.md +35 -0
- package/docs/architecture/manifesto/04-non-goals.md +23 -0
- package/docs/architecture/manifesto/05-licensing-posture.md +61 -0
- package/docs/architecture/manifesto/06-starter-mode-policy.md +49 -0
- package/docs/architecture/manifesto/07-governance.md +60 -0
- package/docs/architecture/primitives-backlog.md +51 -0
- package/docs/architecture.svg +117 -0
- package/docs/claude-code-gateway.md +173 -0
- package/docs/cli-config-templates.md +240 -0
- package/docs/data-flow.svg +72 -0
- package/docs/demo-landing-page.gif +0 -0
- package/docs/demo-qwen-trending.gif +0 -0
- package/docs/dynamic-skills.svg +77 -0
- package/docs/file-flow.svg +126 -0
- package/docs/future-architecture/README.md +152 -0
- package/docs/future-architecture/adr/0001-control-plane-language-go.md +80 -0
- package/docs/future-architecture/adr/0002-guest-agent-language-go.md +84 -0
- package/docs/future-architecture/adr/0003-docker-poc-first-then-k8s.md +37 -0
- package/docs/future-architecture/adr/0004-pluggable-runtime-via-runtimeclass.md +34 -0
- package/docs/future-architecture/adr/0005-mcp-as-control-plane-gateway.md +34 -0
- package/docs/future-architecture/adr/0006-no-agpl-no-bsl-dependencies.md +41 -0
- package/docs/future-architecture/adr/0007-superseded-by-future-architecture.md +37 -0
- package/docs/future-architecture/adr/0008-internal-grpc-external-rest-mcp.md +106 -0
- package/docs/future-architecture/adr/0009-external-protocol-dialects.md +94 -0
- package/docs/future-architecture/adr/0010-lambda-as-inspiration-not-runtime.md +86 -0
- package/docs/future-architecture/adr/0011-kata-as-first-class-dind-runtime.md +84 -0
- package/docs/future-architecture/antipatterns.md +552 -0
- package/docs/future-architecture/architecture/01-layers.md +109 -0
- package/docs/future-architecture/architecture/02-layer4-control-plane.md +122 -0
- package/docs/future-architecture/architecture/03-layer3-providers.md +174 -0
- package/docs/future-architecture/architecture/04-layer2-runtimes.md +114 -0
- package/docs/future-architecture/architecture/04b-credential-broker.md +153 -0
- package/docs/future-architecture/architecture/05-layer1-guest-agent.md +138 -0
- package/docs/future-architecture/architecture/06-storage.md +134 -0
- package/docs/future-architecture/architecture/07-security.md +194 -0
- package/docs/future-architecture/architecture/08-networking.md +149 -0
- package/docs/future-architecture/architecture/09-templates.md +122 -0
- package/docs/future-architecture/architecture/10-observability.md +121 -0
- package/docs/future-architecture/design-notes.md +72 -0
- package/docs/future-architecture/gaps.md +281 -0
- package/docs/future-architecture/phase-template.md +123 -0
- package/docs/future-architecture/references.md +225 -0
- package/docs/future-architecture/research/01-kata-containers.md +100 -0
- package/docs/future-architecture/research/02-e2b-infra.md +133 -0
- package/docs/future-architecture/research/03-coder.md +115 -0
- package/docs/future-architecture/research/04-cloud-hypervisor.md +99 -0
- package/docs/future-architecture/research/05-firecracker.md +114 -0
- package/docs/future-architecture/research/06-agent-sandbox.md +142 -0
- package/docs/future-architecture/research/07-chromedp.md +78 -0
- package/docs/future-architecture/research/08-microsandbox.md +78 -0
- package/docs/future-architecture/research/09-agentbox.md +135 -0
- package/docs/future-architecture/research/10-sysbox.md +100 -0
- package/docs/future-architecture/research/11-firecracker-containerd.md +93 -0
- package/docs/future-architecture/research/12-docker-socket-proxy.md +59 -0
- package/docs/future-architecture/research/14-e2b-desktop-and-surf.md +107 -0
- package/docs/future-architecture/research/18-open-webui-terminals-observed.md +135 -0
- package/docs/future-architecture/research/bank-buyer.md +96 -0
- package/docs/future-architecture/research/enthusiast-audience.md +106 -0
- package/docs/future-architecture/research/proof-uipath-anthropic-2026-05.md +76 -0
- package/docs/future-architecture/research/widemoat-thesis-advisor.md +124 -0
- package/docs/future-architecture/roadmap.md +438 -0
- package/docs/kata-runtime.md +267 -0
- package/docs/kubernetes.md +86 -0
- package/docs/logo.png +0 -0
- package/docs/multi-cli.md +161 -0
- package/docs/openwebui-filter.md +134 -0
- package/docs/roadmap/implementation-roadmap.md +104 -0
- package/docs/sandbox-contents.svg +229 -0
- package/docs/screenshots/01-create-document.png +0 -0
- package/docs/screenshots/02-file-preview.png +0 -0
- package/docs/screenshots/03-browser-viewer.png +0 -0
- package/docs/screenshots/04-sub-agent-terminal.png +0 -0
- package/docs/screenshots/05-chat-overview.png +0 -0
- package/docs/screenshots/06-sub-agent-dashboard.png +0 -0
- package/docs/screenshots/07-frontend-design-skill.png +0 -0
- package/docs/screenshots/08-pptx-skill.png +0 -0
- package/docs/screenshots/09-skill-creator.png +0 -0
- package/docs/screenshots/10-data-chart.png +0 -0
- package/docs/shared-browser.svg +102 -0
- package/docs/system-prompt.md +113 -0
- package/docs/terminal-flow.svg +69 -0
- package/examples/helm/README.md +20 -0
- package/examples/helm/standalone/values.yaml +49 -0
- package/examples/helm/with-open-webui/README.md +99 -0
- package/examples/helm/with-open-webui/values-computer-use.yaml +32 -0
- package/examples/helm/with-open-webui/values-open-webui.yaml +67 -0
- package/fonts/NotoEmoji-Regular.ttf +0 -0
- package/helm/computer-use-server/.helmignore +17 -0
- package/helm/computer-use-server/Chart.yaml +32 -0
- package/helm/computer-use-server/README.md +211 -0
- package/helm/computer-use-server/templates/NOTES.txt +66 -0
- package/helm/computer-use-server/templates/_helpers.tpl +115 -0
- package/helm/computer-use-server/templates/configmap-dind-init.yaml +82 -0
- package/helm/computer-use-server/templates/configmap.yaml +18 -0
- package/helm/computer-use-server/templates/deployment.yaml +248 -0
- package/helm/computer-use-server/templates/ingress.yaml +38 -0
- package/helm/computer-use-server/templates/networkpolicy.yaml +50 -0
- package/helm/computer-use-server/templates/pdb.yaml +16 -0
- package/helm/computer-use-server/templates/pvc-data.yaml +20 -0
- package/helm/computer-use-server/templates/pvc-skills-cache.yaml +20 -0
- package/helm/computer-use-server/templates/pvc-user-data.yaml +20 -0
- package/helm/computer-use-server/templates/pvc-var-lib-docker.yaml +27 -0
- package/helm/computer-use-server/templates/secret.yaml +23 -0
- package/helm/computer-use-server/templates/service.yaml +22 -0
- package/helm/computer-use-server/templates/serviceaccount.yaml +15 -0
- package/helm/computer-use-server/templates/tests/test-health.yaml +23 -0
- package/helm/computer-use-server/values.schema.json +183 -0
- package/helm/computer-use-server/values.yaml +297 -0
- package/lychee.toml +36 -0
- package/openwebui/Dockerfile +52 -0
- package/openwebui/README.md +38 -0
- package/openwebui/functions/README.md +48 -0
- package/openwebui/functions/computer_link_filter.py +487 -0
- package/openwebui/init.sh +305 -0
- package/openwebui/patches/README.md +44 -0
- package/openwebui/patches/fix_artifacts_auto_show.py +441 -0
- package/openwebui/patches/fix_attached_files_position.py +87 -0
- package/openwebui/patches/fix_large_tool_args.py +156 -0
- package/openwebui/patches/fix_large_tool_results.py +289 -0
- package/openwebui/patches/fix_preview_url_detection.py +230 -0
- package/openwebui/patches/fix_skip_embedding_chat_files.py +229 -0
- package/openwebui/patches/fix_skip_rag_files_native_fc.py +100 -0
- package/openwebui/patches/fix_tool_loop_errors.py +510 -0
- package/package.json +39 -0
- package/requirements.txt +112 -0
- package/scripts/check-config.sh +141 -0
- package/scripts/docs-lint/ai-slop-detector.sh +202 -0
- package/scripts/docs-lint/architecture-tree-whitelist.sh +131 -0
- package/scripts/docs-lint/ascii-diagram-detector.sh +58 -0
- package/scripts/docs-lint/front-matter-validator.sh +97 -0
- package/scripts/docs-lint/gitignored-ref-detector.sh +122 -0
- package/scripts/docs-lint/identity-email-detector.sh +48 -0
- package/scripts/docs-lint/test-linters.sh +354 -0
- package/scripts/docs-lint/wc-budget.sh +61 -0
- package/scripts/githooks/pre-push +75 -0
- package/server.json +13 -0
- package/settings-wrapper/Dockerfile +9 -0
- package/settings-wrapper/README.md +119 -0
- package/settings-wrapper/app.py +113 -0
- package/settings-wrapper/requirements.txt +2 -0
- package/settings-wrapper/skills.json +25 -0
- package/skills/README.md +46 -0
- package/skills/examples/algorithmic-art/SKILL.md +405 -0
- package/skills/examples/algorithmic-art/templates/generator_template.js +223 -0
- package/skills/examples/algorithmic-art/templates/viewer.html +601 -0
- package/skills/examples/artifacts-builder/SKILL.md +74 -0
- package/skills/examples/artifacts-builder/scripts/bundle-artifact.sh +54 -0
- package/skills/examples/artifacts-builder/scripts/init-artifact.sh +322 -0
- package/skills/examples/artifacts-builder/scripts/shadcn-components.tar.gz +0 -0
- package/skills/examples/canvas-design/LICENSE.txt +202 -0
- package/skills/examples/canvas-design/SKILL.md +130 -0
- package/skills/examples/canvas-design/canvas-fonts/ArsenalSC-OFL.txt +93 -0
- package/skills/examples/canvas-design/canvas-fonts/ArsenalSC-Regular.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/BigShoulders-Bold.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/BigShoulders-OFL.txt +93 -0
- package/skills/examples/canvas-design/canvas-fonts/BigShoulders-Regular.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/Boldonse-OFL.txt +93 -0
- package/skills/examples/canvas-design/canvas-fonts/Boldonse-Regular.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/BricolageGrotesque-Bold.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/BricolageGrotesque-OFL.txt +93 -0
- package/skills/examples/canvas-design/canvas-fonts/BricolageGrotesque-Regular.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/CrimsonPro-Bold.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/CrimsonPro-Italic.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/CrimsonPro-OFL.txt +93 -0
- package/skills/examples/canvas-design/canvas-fonts/CrimsonPro-Regular.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/DMMono-OFL.txt +93 -0
- package/skills/examples/canvas-design/canvas-fonts/DMMono-Regular.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/EricaOne-OFL.txt +94 -0
- package/skills/examples/canvas-design/canvas-fonts/EricaOne-Regular.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/GeistMono-Bold.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/GeistMono-OFL.txt +93 -0
- package/skills/examples/canvas-design/canvas-fonts/GeistMono-Regular.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/Gloock-OFL.txt +93 -0
- package/skills/examples/canvas-design/canvas-fonts/Gloock-Regular.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/IBMPlexMono-Bold.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/IBMPlexMono-OFL.txt +93 -0
- package/skills/examples/canvas-design/canvas-fonts/IBMPlexMono-Regular.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/IBMPlexSerif-Bold.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/IBMPlexSerif-BoldItalic.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/IBMPlexSerif-Italic.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/IBMPlexSerif-Regular.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/InstrumentSans-Bold.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/InstrumentSans-BoldItalic.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/InstrumentSans-Italic.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/InstrumentSans-OFL.txt +93 -0
- package/skills/examples/canvas-design/canvas-fonts/InstrumentSans-Regular.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/InstrumentSerif-Italic.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/InstrumentSerif-Regular.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/Italiana-OFL.txt +93 -0
- package/skills/examples/canvas-design/canvas-fonts/Italiana-Regular.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/JetBrainsMono-Bold.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/JetBrainsMono-OFL.txt +93 -0
- package/skills/examples/canvas-design/canvas-fonts/JetBrainsMono-Regular.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/Jura-Light.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/Jura-Medium.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/Jura-OFL.txt +93 -0
- package/skills/examples/canvas-design/canvas-fonts/LibreBaskerville-OFL.txt +93 -0
- package/skills/examples/canvas-design/canvas-fonts/LibreBaskerville-Regular.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/Lora-Bold.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/Lora-BoldItalic.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/Lora-Italic.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/Lora-OFL.txt +93 -0
- package/skills/examples/canvas-design/canvas-fonts/Lora-Regular.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/NationalPark-Bold.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/NationalPark-OFL.txt +93 -0
- package/skills/examples/canvas-design/canvas-fonts/NationalPark-Regular.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/NothingYouCouldDo-OFL.txt +93 -0
- package/skills/examples/canvas-design/canvas-fonts/NothingYouCouldDo-Regular.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/Outfit-Bold.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/Outfit-OFL.txt +93 -0
- package/skills/examples/canvas-design/canvas-fonts/Outfit-Regular.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/PixelifySans-Medium.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/PixelifySans-OFL.txt +93 -0
- package/skills/examples/canvas-design/canvas-fonts/PoiretOne-OFL.txt +93 -0
- package/skills/examples/canvas-design/canvas-fonts/PoiretOne-Regular.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/RedHatMono-Bold.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/RedHatMono-OFL.txt +93 -0
- package/skills/examples/canvas-design/canvas-fonts/RedHatMono-Regular.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/Silkscreen-OFL.txt +93 -0
- package/skills/examples/canvas-design/canvas-fonts/Silkscreen-Regular.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/SmoochSans-Medium.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/SmoochSans-OFL.txt +93 -0
- package/skills/examples/canvas-design/canvas-fonts/Tektur-Medium.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/Tektur-OFL.txt +93 -0
- package/skills/examples/canvas-design/canvas-fonts/Tektur-Regular.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/WorkSans-Bold.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/WorkSans-BoldItalic.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/WorkSans-Italic.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/WorkSans-OFL.txt +93 -0
- package/skills/examples/canvas-design/canvas-fonts/WorkSans-Regular.ttf +0 -0
- package/skills/examples/canvas-design/canvas-fonts/YoungSerif-OFL.txt +93 -0
- package/skills/examples/canvas-design/canvas-fonts/YoungSerif-Regular.ttf +0 -0
- package/skills/examples/copy-editing/SKILL.md +447 -0
- package/skills/examples/copy-editing/evals/evals.json +89 -0
- package/skills/examples/copy-editing/references/plain-english-alternatives.md +394 -0
- package/skills/examples/internal-comms/LICENSE.txt +202 -0
- package/skills/examples/internal-comms/SKILL.md +32 -0
- package/skills/examples/internal-comms/examples/3p-updates.md +47 -0
- package/skills/examples/internal-comms/examples/company-newsletter.md +65 -0
- package/skills/examples/internal-comms/examples/faq-answers.md +30 -0
- package/skills/examples/internal-comms/examples/general-comms.md +16 -0
- package/skills/examples/mcp-builder/SKILL.md +328 -0
- package/skills/examples/mcp-builder/reference/evaluation.md +602 -0
- package/skills/examples/mcp-builder/reference/mcp_best_practices.md +915 -0
- package/skills/examples/mcp-builder/reference/node_mcp_server.md +916 -0
- package/skills/examples/mcp-builder/reference/python_mcp_server.md +752 -0
- package/skills/examples/mcp-builder/scripts/connections.py +151 -0
- package/skills/examples/mcp-builder/scripts/evaluation.py +373 -0
- package/skills/examples/mcp-builder/scripts/example_evaluation.xml +22 -0
- package/skills/examples/mcp-builder/scripts/requirements.txt +2 -0
- package/skills/examples/product-marketing-context/SKILL.md +241 -0
- package/skills/examples/product-marketing-context/evals/evals.json +85 -0
- package/skills/examples/single-cell-rna-qc/SKILL.md +175 -0
- package/skills/examples/single-cell-rna-qc/references/scverse_qc_guidelines.md +186 -0
- package/skills/examples/single-cell-rna-qc/scripts/qc_analysis.py +232 -0
- package/skills/examples/single-cell-rna-qc/scripts/qc_core.py +233 -0
- package/skills/examples/single-cell-rna-qc/scripts/qc_plotting.py +235 -0
- package/skills/examples/skill-creator/SKILL.md +355 -0
- package/skills/examples/skill-creator/references/output-patterns.md +82 -0
- package/skills/examples/skill-creator/references/workflows.md +28 -0
- package/skills/examples/skill-creator/scripts/init_skill.py +303 -0
- package/skills/examples/skill-creator/scripts/package_skill.py +110 -0
- package/skills/examples/skill-creator/scripts/quick_validate.py +95 -0
- package/skills/examples/slack-gif-creator/SKILL.md +254 -0
- package/skills/examples/slack-gif-creator/core/easing.py +234 -0
- package/skills/examples/slack-gif-creator/core/frame_composer.py +176 -0
- package/skills/examples/slack-gif-creator/core/gif_builder.py +269 -0
- package/skills/examples/slack-gif-creator/core/validators.py +136 -0
- package/skills/examples/slack-gif-creator/requirements.txt +4 -0
- package/skills/examples/social-content/SKILL.md +278 -0
- package/skills/examples/social-content/evals/evals.json +92 -0
- package/skills/examples/social-content/references/platforms.md +170 -0
- package/skills/examples/social-content/references/post-templates.md +177 -0
- package/skills/examples/social-content/references/reverse-engineering.md +195 -0
- package/skills/examples/theme-factory/SKILL.md +59 -0
- package/skills/examples/theme-factory/theme-showcase.pdf +0 -0
- package/skills/examples/theme-factory/themes/arctic-frost.md +19 -0
- package/skills/examples/theme-factory/themes/botanical-garden.md +19 -0
- package/skills/examples/theme-factory/themes/desert-rose.md +19 -0
- package/skills/examples/theme-factory/themes/forest-canopy.md +19 -0
- package/skills/examples/theme-factory/themes/golden-hour.md +19 -0
- package/skills/examples/theme-factory/themes/midnight-galaxy.md +19 -0
- package/skills/examples/theme-factory/themes/modern-minimalist.md +19 -0
- package/skills/examples/theme-factory/themes/ocean-depths.md +19 -0
- package/skills/examples/theme-factory/themes/sunset-boulevard.md +19 -0
- package/skills/examples/theme-factory/themes/tech-innovation.md +19 -0
- package/skills/examples/web-artifacts-builder/LICENSE.txt +202 -0
- package/skills/examples/web-artifacts-builder/SKILL.md +74 -0
- package/skills/examples/web-artifacts-builder/scripts/bundle-artifact.sh +54 -0
- package/skills/examples/web-artifacts-builder/scripts/init-artifact.sh +322 -0
- package/skills/examples/web-artifacts-builder/scripts/shadcn-components.tar.gz +0 -0
- package/skills/examples/writing-skills/SKILL.md +655 -0
- package/skills/examples/writing-skills/anthropic-best-practices.md +1150 -0
- package/skills/examples/writing-skills/examples/CLAUDE_MD_TESTING.md +189 -0
- package/skills/examples/writing-skills/graphviz-conventions.dot +172 -0
- package/skills/examples/writing-skills/persuasion-principles.md +187 -0
- package/skills/examples/writing-skills/render-graphs.js +168 -0
- package/skills/examples/writing-skills/testing-skills-with-subagents.md +384 -0
- package/skills/public/describe-image/SKILL.md +105 -0
- package/skills/public/describe-image/scripts/describe.py +389 -0
- package/skills/public/doc-coauthoring/SKILL.md +375 -0
- package/skills/public/docx/LICENSE.txt +30 -0
- package/skills/public/docx/SKILL.md +199 -0
- package/skills/public/docx/docx-js.md +350 -0
- package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-chart.xsd +1499 -0
- package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-chartDrawing.xsd +146 -0
- package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-diagram.xsd +1085 -0
- package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-lockedCanvas.xsd +11 -0
- package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-main.xsd +3081 -0
- package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-picture.xsd +23 -0
- package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-spreadsheetDrawing.xsd +185 -0
- package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-wordprocessingDrawing.xsd +287 -0
- package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/pml.xsd +1676 -0
- package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-additionalCharacteristics.xsd +28 -0
- package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-bibliography.xsd +144 -0
- package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-commonSimpleTypes.xsd +174 -0
- package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-customXmlDataProperties.xsd +25 -0
- package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-customXmlSchemaProperties.xsd +18 -0
- package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesCustom.xsd +59 -0
- package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesExtended.xsd +56 -0
- package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesVariantTypes.xsd +195 -0
- package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-math.xsd +582 -0
- package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-relationshipReference.xsd +25 -0
- package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/sml.xsd +4439 -0
- package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/vml-main.xsd +570 -0
- package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/vml-officeDrawing.xsd +509 -0
- package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/vml-presentationDrawing.xsd +12 -0
- package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/vml-spreadsheetDrawing.xsd +108 -0
- package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/vml-wordprocessingDrawing.xsd +96 -0
- package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/wml.xsd +3646 -0
- package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/xml.xsd +116 -0
- package/skills/public/docx/ooxml/schemas/ecma/fouth-edition/opc-contentTypes.xsd +42 -0
- package/skills/public/docx/ooxml/schemas/ecma/fouth-edition/opc-coreProperties.xsd +50 -0
- package/skills/public/docx/ooxml/schemas/ecma/fouth-edition/opc-digSig.xsd +49 -0
- package/skills/public/docx/ooxml/schemas/ecma/fouth-edition/opc-relationships.xsd +33 -0
- package/skills/public/docx/ooxml/schemas/mce/mc.xsd +75 -0
- package/skills/public/docx/ooxml/schemas/microsoft/wml-2010.xsd +560 -0
- package/skills/public/docx/ooxml/schemas/microsoft/wml-2012.xsd +67 -0
- package/skills/public/docx/ooxml/schemas/microsoft/wml-2018.xsd +14 -0
- package/skills/public/docx/ooxml/schemas/microsoft/wml-cex-2018.xsd +20 -0
- package/skills/public/docx/ooxml/schemas/microsoft/wml-cid-2016.xsd +13 -0
- package/skills/public/docx/ooxml/schemas/microsoft/wml-sdtdatahash-2020.xsd +4 -0
- package/skills/public/docx/ooxml/schemas/microsoft/wml-symex-2015.xsd +8 -0
- package/skills/public/docx/ooxml/scripts/pack.py +159 -0
- package/skills/public/docx/ooxml/scripts/unpack.py +29 -0
- package/skills/public/docx/ooxml/scripts/validate.py +69 -0
- package/skills/public/docx/ooxml/scripts/validation/__init__.py +15 -0
- package/skills/public/docx/ooxml/scripts/validation/base.py +951 -0
- package/skills/public/docx/ooxml/scripts/validation/docx.py +274 -0
- package/skills/public/docx/ooxml/scripts/validation/pptx.py +315 -0
- package/skills/public/docx/ooxml/scripts/validation/redlining.py +279 -0
- package/skills/public/docx/ooxml.md +632 -0
- package/skills/public/docx/scripts/__init__.py +1 -0
- package/skills/public/docx/scripts/document.py +1292 -0
- package/skills/public/docx/scripts/templates/comments.xml +3 -0
- package/skills/public/docx/scripts/templates/commentsExtended.xml +3 -0
- package/skills/public/docx/scripts/templates/commentsExtensible.xml +3 -0
- package/skills/public/docx/scripts/templates/commentsIds.xml +3 -0
- package/skills/public/docx/scripts/templates/people.xml +3 -0
- package/skills/public/docx/scripts/utilities.py +374 -0
- package/skills/public/file-reading/LICENSE.txt +30 -0
- package/skills/public/file-reading/SKILL.md +350 -0
- package/skills/public/frontend-design/LICENSE.txt +177 -0
- package/skills/public/frontend-design/SKILL.md +42 -0
- package/skills/public/gitlab-explorer/SKILL.md +174 -0
- package/skills/public/gitlab-explorer/references/git-commands.md +323 -0
- package/skills/public/gitlab-explorer/references/glab-commands.md +282 -0
- package/skills/public/gitlab-explorer/scripts/check_gitlab_auth.sh +109 -0
- package/skills/public/pdf/FORMS.md +205 -0
- package/skills/public/pdf/REFERENCE.md +612 -0
- package/skills/public/pdf/SKILL.md +364 -0
- package/skills/public/pdf/scripts/check_bounding_boxes.py +70 -0
- package/skills/public/pdf/scripts/check_bounding_boxes_test.py +226 -0
- package/skills/public/pdf/scripts/check_fillable_fields.py +12 -0
- package/skills/public/pdf/scripts/convert_pdf_to_images.py +35 -0
- package/skills/public/pdf/scripts/create_validation_image.py +41 -0
- package/skills/public/pdf/scripts/extract_form_field_info.py +152 -0
- package/skills/public/pdf/scripts/fill_fillable_fields.py +114 -0
- package/skills/public/pdf/scripts/fill_pdf_form_with_annotations.py +108 -0
- package/skills/public/pdf-reading/LICENSE.txt +30 -0
- package/skills/public/pdf-reading/REFERENCE.md +196 -0
- package/skills/public/pdf-reading/SKILL.md +305 -0
- package/skills/public/playwright-cli/SKILL.md +278 -0
- package/skills/public/playwright-cli/references/request-mocking.md +87 -0
- package/skills/public/playwright-cli/references/running-code.md +232 -0
- package/skills/public/playwright-cli/references/session-management.md +169 -0
- package/skills/public/playwright-cli/references/storage-state.md +275 -0
- package/skills/public/playwright-cli/references/test-generation.md +88 -0
- package/skills/public/playwright-cli/references/tracing.md +139 -0
- package/skills/public/playwright-cli/references/video-recording.md +43 -0
- package/skills/public/pptx/LICENSE.txt +30 -0
- package/skills/public/pptx/SKILL.md +484 -0
- package/skills/public/pptx/css.md +335 -0
- package/skills/public/pptx/html2pptx.md +893 -0
- package/skills/public/pptx/html2pptx.tgz +0 -0
- package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-chart.xsd +1499 -0
- package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-chartDrawing.xsd +146 -0
- package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-diagram.xsd +1085 -0
- package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-lockedCanvas.xsd +11 -0
- package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-main.xsd +3081 -0
- package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-picture.xsd +23 -0
- package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-spreadsheetDrawing.xsd +185 -0
- package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-wordprocessingDrawing.xsd +287 -0
- package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/pml.xsd +1676 -0
- package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-additionalCharacteristics.xsd +28 -0
- package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-bibliography.xsd +144 -0
- package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-commonSimpleTypes.xsd +174 -0
- package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-customXmlDataProperties.xsd +25 -0
- package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-customXmlSchemaProperties.xsd +18 -0
- package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesCustom.xsd +59 -0
- package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesExtended.xsd +56 -0
- package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesVariantTypes.xsd +195 -0
- package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-math.xsd +582 -0
- package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-relationshipReference.xsd +25 -0
- package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/sml.xsd +4439 -0
- package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/vml-main.xsd +570 -0
- package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/vml-officeDrawing.xsd +509 -0
- package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/vml-presentationDrawing.xsd +12 -0
- package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/vml-spreadsheetDrawing.xsd +108 -0
- package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/vml-wordprocessingDrawing.xsd +96 -0
- package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/wml.xsd +3646 -0
- package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/xml.xsd +116 -0
- package/skills/public/pptx/ooxml/schemas/ecma/fouth-edition/opc-contentTypes.xsd +42 -0
- package/skills/public/pptx/ooxml/schemas/ecma/fouth-edition/opc-coreProperties.xsd +50 -0
- package/skills/public/pptx/ooxml/schemas/ecma/fouth-edition/opc-digSig.xsd +49 -0
- package/skills/public/pptx/ooxml/schemas/ecma/fouth-edition/opc-relationships.xsd +33 -0
- package/skills/public/pptx/ooxml/schemas/mce/mc.xsd +75 -0
- package/skills/public/pptx/ooxml/schemas/microsoft/wml-2010.xsd +560 -0
- package/skills/public/pptx/ooxml/schemas/microsoft/wml-2012.xsd +67 -0
- package/skills/public/pptx/ooxml/schemas/microsoft/wml-2018.xsd +14 -0
- package/skills/public/pptx/ooxml/schemas/microsoft/wml-cex-2018.xsd +20 -0
- package/skills/public/pptx/ooxml/schemas/microsoft/wml-cid-2016.xsd +13 -0
- package/skills/public/pptx/ooxml/schemas/microsoft/wml-sdtdatahash-2020.xsd +4 -0
- package/skills/public/pptx/ooxml/schemas/microsoft/wml-symex-2015.xsd +8 -0
- package/skills/public/pptx/ooxml/scripts/pack.py +159 -0
- package/skills/public/pptx/ooxml/scripts/unpack.py +29 -0
- package/skills/public/pptx/ooxml/scripts/validate.py +69 -0
- package/skills/public/pptx/ooxml/scripts/validation/__init__.py +15 -0
- package/skills/public/pptx/ooxml/scripts/validation/base.py +951 -0
- package/skills/public/pptx/ooxml/scripts/validation/docx.py +274 -0
- package/skills/public/pptx/ooxml/scripts/validation/pptx.py +315 -0
- package/skills/public/pptx/ooxml/scripts/validation/redlining.py +279 -0
- package/skills/public/pptx/ooxml.md +427 -0
- package/skills/public/pptx/scripts/inventory.py +1020 -0
- package/skills/public/pptx/scripts/rearrange.py +231 -0
- package/skills/public/pptx/scripts/replace.py +385 -0
- package/skills/public/pptx/scripts/thumbnail.py +450 -0
- package/skills/public/skill-creator/SKILL.md +356 -0
- package/skills/public/skill-creator/references/output-patterns.md +82 -0
- package/skills/public/skill-creator/references/workflows.md +28 -0
- package/skills/public/skill-creator/scripts/init_skill.py +303 -0
- package/skills/public/skill-creator/scripts/package_skill.py +110 -0
- package/skills/public/skill-creator/scripts/quick_validate.py +95 -0
- package/skills/public/sub-agent/SKILL.md +186 -0
- package/skills/public/sub-agent/references/security-review.md +153 -0
- package/skills/public/sub-agent/references/usage.md +207 -0
- package/skills/public/sub-agent/scripts/list_subagent_models.sh +22 -0
- package/skills/public/test-driven-development/SKILL.md +371 -0
- package/skills/public/test-driven-development/testing-anti-patterns.md +299 -0
- package/skills/public/webapp-testing/LICENSE.txt +202 -0
- package/skills/public/webapp-testing/SKILL.md +96 -0
- package/skills/public/webapp-testing/examples/console_logging.py +35 -0
- package/skills/public/webapp-testing/examples/element_discovery.py +40 -0
- package/skills/public/webapp-testing/examples/static_html_automation.py +33 -0
- package/skills/public/webapp-testing/scripts/with_server.py +106 -0
- package/skills/public/xlsx/LICENSE.txt +30 -0
- package/skills/public/xlsx/SKILL.md +316 -0
- package/skills/public/xlsx/preview_data.py +93 -0
- package/skills/public/xlsx/recalc.py +178 -0
- package/tests/README.md +42 -0
- package/tests/fixtures/cli/claude_v0.9.2.0_argv.json +46 -0
- package/tests/fixtures/cli/claude_v0.9.2.0_stdout.json +32 -0
- package/tests/fixtures/cli/codex_run.jsonl +4 -0
- package/tests/fixtures/cli/opencode_run.jsonl +6 -0
- package/tests/integration/README.md +56 -0
- package/tests/integration/conftest.py +280 -0
- package/tests/integration/pytest.ini +13 -0
- package/tests/integration/test_mcp_auth.py +85 -0
- package/tests/integration/test_mcp_tools.py +101 -0
- package/tests/integration/test_workspace_lifecycle.py +125 -0
- package/tests/orchestrator/mock_llm_server.py +343 -0
- package/tests/orchestrator/test_cli_adapters.py +566 -0
- package/tests/orchestrator/test_cli_adapters_live.py +527 -0
- package/tests/orchestrator/test_cli_runtime.py +451 -0
- package/tests/orchestrator/test_docker_manager.py +302 -0
- package/tests/orchestrator/test_dynamic_instructions.py +69 -0
- package/tests/orchestrator/test_mcp_resources.py +140 -0
- package/tests/orchestrator/test_mcp_tools.py +224 -0
- package/tests/orchestrator/test_passthrough_isolation.py +201 -0
- package/tests/orchestrator/test_readme_in_container.py +76 -0
- package/tests/orchestrator/test_render_cache.py +84 -0
- package/tests/orchestrator/test_runtime_cli_endpoint.py +108 -0
- package/tests/orchestrator/test_single_user_mode.py +212 -0
- package/tests/orchestrator/test_startup_warnings.py +123 -0
- package/tests/orchestrator/test_sub_agent_dispatch.py +327 -0
- package/tests/orchestrator/test_subagent_claude_compat.py +367 -0
- package/tests/orchestrator/test_system_prompt_endpoint.py +191 -0
- package/tests/orchestrator/test_tool_descriptions.py +52 -0
- package/tests/orchestrator/test_view_image.py +201 -0
- package/tests/patches/conftest.py +30 -0
- package/tests/patches/fixtures/__init__.py +10 -0
- package/tests/patches/fixtures/middleware_v0.9.1.py +5057 -0
- package/tests/patches/fixtures/middleware_v0.9.2.py +5120 -0
- package/tests/patches/fixtures/retrieval_v0.9.1.py +2684 -0
- package/tests/patches/fixtures/retrieval_v0.9.2.py +2700 -0
- package/tests/patches/test_fix_attached_files_position.py +118 -0
- package/tests/patches/test_fix_large_tool_args.py +130 -0
- package/tests/patches/test_fix_large_tool_results.py +531 -0
- package/tests/patches/test_fix_skip_embedding_chat_files.py +160 -0
- package/tests/patches/test_fix_skip_rag_files_native_fc.py +120 -0
- package/tests/patches/test_fix_tool_loop_errors.py +128 -0
- package/tests/security/test_path_traversal_app.py +132 -0
- package/tests/security/test_path_traversal_docker.py +36 -0
- package/tests/security/test_path_traversal_settings.py +87 -0
- package/tests/security/test_safe_path_util.py +166 -0
- package/tests/security/test_xss_preview.py +46 -0
- package/tests/test-default-model-resolution.py +136 -0
- package/tests/test-docker-image.sh +358 -0
- package/tests/test-list-subagent-models.sh +421 -0
- package/tests/test-mcp-endpoint-live.sh +92 -0
- package/tests/test-mcp-native-surface.sh +213 -0
- package/tests/test-no-cyrillic.sh +135 -0
- package/tests/test-opencode-error-mapping.py +130 -0
- package/tests/test-pr88-skills.sh +305 -0
- package/tests/test-project-structure.sh +202 -0
- package/tests/test-single-user-mode.sh +269 -0
- package/tests/test-skill-no-hardcoded-models.sh +65 -0
- package/tests/test-subagent-cli-surface.py +137 -0
- package/tests/test-subagent-runtime.sh +109 -0
- package/tests/test_codex_toml_converter.py +204 -0
- package/tests/test_default_resolver_no_legacy_global.py +159 -0
- package/tests/test_filter.py +648 -0
- package/tests/test_init_sh_unchanged.sh +49 -0
- package/tests/test_opencode_alias_map_drop.py +144 -0
- package/tests/test_requirements.py +91 -0
- package/tests/test_subagent_docstring.py +193 -0
- package/tests/test_tools.py +34 -0
- package/vendor/extract-text/README.md +46 -0
- package/vendor/extract-text/extract-text +0 -0
|
@@ -0,0 +1,1522 @@
|
|
|
1
|
+
// SPDX-License-Identifier: FSL-1.1-Apache-2.0
|
|
2
|
+
// Copyright (c) 2025 Open Computer Use Contributors
|
|
3
|
+
// =============================================================================
|
|
4
|
+
// Preview SPA — Preact + HTM
|
|
5
|
+
// =============================================================================
|
|
6
|
+
|
|
7
|
+
import { html, render, useState, useEffect, useRef, useCallback, useMemo } from '/static/preact-htm.min.js';
|
|
8
|
+
import { icon, fileIcon, fileIconLarge } from '/static/icons.js';
|
|
9
|
+
import { BrowserViewer } from '/static/browser-viewer.js';
|
|
10
|
+
import { t, LANG } from '/static/locale.js';
|
|
11
|
+
|
|
12
|
+
const { apiUrl: API_URL, filesBase: FILES_BASE, chatId: CHAT_ID } = window.__CONFIG__;
|
|
13
|
+
|
|
14
|
+
// =============================================================================
|
|
15
|
+
// Utilities
|
|
16
|
+
// =============================================================================
|
|
17
|
+
|
|
18
|
+
function escapeHtml(text) {
|
|
19
|
+
const div = document.createElement('div');
|
|
20
|
+
div.textContent = text;
|
|
21
|
+
return div.innerHTML;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
function formatSize(bytes) {
|
|
25
|
+
if (bytes < 1024) return bytes + ' B';
|
|
26
|
+
if (bytes < 1024 * 1024) return (bytes / 1024).toFixed(1) + ' KB';
|
|
27
|
+
return (bytes / (1024 * 1024)).toFixed(1) + ' MB';
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
const _loadedScripts = new Set();
|
|
31
|
+
function loadScript(url) {
|
|
32
|
+
if (_loadedScripts.has(url)) return Promise.resolve();
|
|
33
|
+
return new Promise((resolve, reject) => {
|
|
34
|
+
const s = document.createElement('script');
|
|
35
|
+
s.src = url;
|
|
36
|
+
s.onload = () => { _loadedScripts.add(url); resolve(); };
|
|
37
|
+
s.onerror = reject;
|
|
38
|
+
document.head.appendChild(s);
|
|
39
|
+
});
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
function normalizePath(path) {
|
|
43
|
+
const parts = path.split('/');
|
|
44
|
+
const result = [];
|
|
45
|
+
for (let i = 0; i < parts.length; i++) {
|
|
46
|
+
if (parts[i] === '.' || parts[i] === '') continue;
|
|
47
|
+
if (parts[i] === '..') { result.pop(); }
|
|
48
|
+
else { result.push(parts[i]); }
|
|
49
|
+
}
|
|
50
|
+
return result.join('/');
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
function parseCSV(text, delimiter) {
|
|
54
|
+
const rows = [];
|
|
55
|
+
let row = [];
|
|
56
|
+
let cell = '';
|
|
57
|
+
let inQuotes = false;
|
|
58
|
+
for (let i = 0; i < text.length; i++) {
|
|
59
|
+
const ch = text[i];
|
|
60
|
+
if (inQuotes) {
|
|
61
|
+
if (ch === '"') {
|
|
62
|
+
if (i + 1 < text.length && text[i + 1] === '"') { cell += '"'; i++; }
|
|
63
|
+
else { inQuotes = false; }
|
|
64
|
+
} else { cell += ch; }
|
|
65
|
+
} else {
|
|
66
|
+
if (ch === '"') { inQuotes = true; }
|
|
67
|
+
else if (ch === delimiter) { row.push(cell); cell = ''; }
|
|
68
|
+
else if (ch === '\n') {
|
|
69
|
+
row.push(cell); cell = '';
|
|
70
|
+
if (row.length > 0) rows.push(row);
|
|
71
|
+
row = [];
|
|
72
|
+
} else if (ch !== '\r') { cell += ch; }
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
if (cell || row.length > 0) { row.push(cell); rows.push(row); }
|
|
76
|
+
return rows;
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
function showToast(text) {
|
|
80
|
+
const msg = document.createElement('div');
|
|
81
|
+
msg.className = 'toast';
|
|
82
|
+
msg.textContent = text;
|
|
83
|
+
document.body.appendChild(msg);
|
|
84
|
+
setTimeout(() => msg.remove(), 1500);
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
function copyText(text) {
|
|
88
|
+
// Try modern clipboard API first, fallback to execCommand for iframe sandbox
|
|
89
|
+
if (navigator.clipboard && navigator.clipboard.writeText) {
|
|
90
|
+
return navigator.clipboard.writeText(text).then(
|
|
91
|
+
() => true,
|
|
92
|
+
() => copyTextFallback(text)
|
|
93
|
+
);
|
|
94
|
+
}
|
|
95
|
+
return Promise.resolve(copyTextFallback(text));
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
function copyTextFallback(text) {
|
|
99
|
+
const ta = document.createElement('textarea');
|
|
100
|
+
ta.value = text;
|
|
101
|
+
ta.style.cssText = 'position:fixed;left:-9999px;top:-9999px';
|
|
102
|
+
document.body.appendChild(ta);
|
|
103
|
+
ta.select();
|
|
104
|
+
try { document.execCommand('copy'); return true; }
|
|
105
|
+
catch { return false; }
|
|
106
|
+
finally { ta.remove(); }
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
// =============================================================================
|
|
110
|
+
// Link interception for HTML previews and Markdown
|
|
111
|
+
// =============================================================================
|
|
112
|
+
|
|
113
|
+
function handleLinkClick(href, resolvedUrl, files, selectedFile, onSelectFile) {
|
|
114
|
+
if (!href) return;
|
|
115
|
+
if (href.startsWith('#')) {
|
|
116
|
+
const targetId = decodeURIComponent(href.substring(1));
|
|
117
|
+
let el = document.getElementById(targetId);
|
|
118
|
+
if (!el) { try { el = document.querySelector(href); } catch(e) {} }
|
|
119
|
+
if (el) el.scrollIntoView({ behavior: 'smooth' });
|
|
120
|
+
return;
|
|
121
|
+
}
|
|
122
|
+
if (href.startsWith('http://') || href.startsWith('https://') || href.startsWith('//')) {
|
|
123
|
+
try {
|
|
124
|
+
const linkUrl = new URL(href, window.location.origin);
|
|
125
|
+
if (linkUrl.origin === window.location.origin) {
|
|
126
|
+
window.open(href, '_blank', 'noopener');
|
|
127
|
+
} else {
|
|
128
|
+
_showExternalLinkDialog(href);
|
|
129
|
+
}
|
|
130
|
+
} catch (e) { /* invalid URL, ignore */ }
|
|
131
|
+
return;
|
|
132
|
+
}
|
|
133
|
+
let filePath = null;
|
|
134
|
+
const filesBaseWithSlash = FILES_BASE + '/';
|
|
135
|
+
if (resolvedUrl) {
|
|
136
|
+
try {
|
|
137
|
+
const url = new URL(resolvedUrl, window.location.origin);
|
|
138
|
+
if (url.pathname.startsWith(filesBaseWithSlash)) {
|
|
139
|
+
filePath = decodeURIComponent(url.pathname.substring(filesBaseWithSlash.length));
|
|
140
|
+
}
|
|
141
|
+
} catch(e) {}
|
|
142
|
+
}
|
|
143
|
+
if (!filePath) {
|
|
144
|
+
const currentDir = selectedFile && selectedFile.path.includes('/')
|
|
145
|
+
? selectedFile.path.substring(0, selectedFile.path.lastIndexOf('/'))
|
|
146
|
+
: '';
|
|
147
|
+
filePath = currentDir ? currentDir + '/' + href : href;
|
|
148
|
+
filePath = normalizePath(filePath);
|
|
149
|
+
}
|
|
150
|
+
if (filePath) {
|
|
151
|
+
filePath = filePath.split('?')[0].split('#')[0];
|
|
152
|
+
let targetFile = files.find(f => f.path === filePath);
|
|
153
|
+
if (!targetFile) {
|
|
154
|
+
const lp = filePath.toLowerCase();
|
|
155
|
+
targetFile = files.find(f => f.path.toLowerCase() === lp);
|
|
156
|
+
}
|
|
157
|
+
if (targetFile) { onSelectFile(targetFile); return; }
|
|
158
|
+
}
|
|
159
|
+
if (resolvedUrl) {
|
|
160
|
+
try {
|
|
161
|
+
const rUrl = new URL(resolvedUrl, window.location.origin);
|
|
162
|
+
if (rUrl.origin !== window.location.origin) {
|
|
163
|
+
_showExternalLinkDialog(resolvedUrl);
|
|
164
|
+
return;
|
|
165
|
+
}
|
|
166
|
+
} catch (e) { /* invalid URL, ignore */ }
|
|
167
|
+
window.open(resolvedUrl, '_blank', 'noopener');
|
|
168
|
+
}
|
|
169
|
+
}
|
|
170
|
+
|
|
171
|
+
function _showExternalLinkDialog(href) {
|
|
172
|
+
const existing = document.getElementById('__ext_link_dialog');
|
|
173
|
+
if (existing) existing.remove();
|
|
174
|
+
const overlay = document.createElement('div');
|
|
175
|
+
overlay.id = '__ext_link_dialog';
|
|
176
|
+
overlay.style.cssText = 'position:fixed;inset:0;background:rgba(0,0,0,.5);display:flex;align-items:center;justify-content:center;z-index:99999';
|
|
177
|
+
const box = document.createElement('div');
|
|
178
|
+
box.style.cssText = 'background:var(--bg-primary,#fff);color:var(--text-primary,#000);border-radius:8px;padding:20px;max-width:420px;word-break:break-all;font-family:system-ui,sans-serif';
|
|
179
|
+
box.innerHTML = '<p style="margin:0 0 8px;font-weight:600">Open external link?</p>'
|
|
180
|
+
+ '<p style="margin:0 0 16px;font-size:13px;opacity:.8">' + href.replace(/</g, '<') + '</p>';
|
|
181
|
+
const btns = document.createElement('div');
|
|
182
|
+
btns.style.cssText = 'display:flex;gap:8px;justify-content:flex-end';
|
|
183
|
+
const cancel = document.createElement('button');
|
|
184
|
+
cancel.textContent = 'Cancel';
|
|
185
|
+
cancel.style.cssText = 'padding:6px 16px;border:1px solid #ccc;border-radius:4px;background:transparent;cursor:pointer';
|
|
186
|
+
cancel.onclick = () => overlay.remove();
|
|
187
|
+
const open = document.createElement('button');
|
|
188
|
+
open.textContent = 'Open';
|
|
189
|
+
open.style.cssText = 'padding:6px 16px;border:none;border-radius:4px;background:#2563eb;color:#fff;cursor:pointer';
|
|
190
|
+
open.onclick = () => { overlay.remove(); window.open(href, '_blank', 'noopener,noreferrer'); };
|
|
191
|
+
btns.append(cancel, open);
|
|
192
|
+
box.appendChild(btns);
|
|
193
|
+
overlay.appendChild(box);
|
|
194
|
+
overlay.onclick = (e) => { if (e.target === overlay) overlay.remove(); };
|
|
195
|
+
document.body.appendChild(overlay);
|
|
196
|
+
}
|
|
197
|
+
|
|
198
|
+
// =============================================================================
|
|
199
|
+
// Preview Renderers (imperative DOM — these render into a container ref)
|
|
200
|
+
// =============================================================================
|
|
201
|
+
|
|
202
|
+
async function renderHtmlPreview(container, file) {
|
|
203
|
+
try {
|
|
204
|
+
const resp = await fetch(file.url);
|
|
205
|
+
let text = await resp.text();
|
|
206
|
+
const fileDir = file.path.includes('/') ? file.path.substring(0, file.path.lastIndexOf('/')) : '';
|
|
207
|
+
const baseUrl = fileDir ? FILES_BASE + '/' + fileDir + '/' : FILES_BASE + '/';
|
|
208
|
+
const baseTag = `<base href="${baseUrl}">`;
|
|
209
|
+
const linkInterceptScript = '<scr' + 'ipt>'
|
|
210
|
+
+ '(function(){'
|
|
211
|
+
+ 'document.addEventListener("click",function(e){'
|
|
212
|
+
+ 'var a=e.target.closest("a");'
|
|
213
|
+
+ 'if(!a)return;'
|
|
214
|
+
+ 'var href=a.getAttribute("href");'
|
|
215
|
+
+ 'if(!href)return;'
|
|
216
|
+
+ 'e.preventDefault();'
|
|
217
|
+
+ 'e.stopPropagation();'
|
|
218
|
+
+ 'window.parent.postMessage({'
|
|
219
|
+
+ 'type:"iframe-link-click",'
|
|
220
|
+
+ 'href:href,'
|
|
221
|
+
+ 'resolvedUrl:a.href'
|
|
222
|
+
+ '},"*");'
|
|
223
|
+
+ '},true);'
|
|
224
|
+
+ '})();'
|
|
225
|
+
+ '</scr' + 'ipt>';
|
|
226
|
+
const injection = baseTag + linkInterceptScript;
|
|
227
|
+
if (text.includes('<head>')) {
|
|
228
|
+
text = text.replace('<head>', '<head>' + injection);
|
|
229
|
+
} else if (text.includes('<html>')) {
|
|
230
|
+
text = text.replace('<html>', '<html><head>' + injection + '</head>');
|
|
231
|
+
} else {
|
|
232
|
+
text = injection + text;
|
|
233
|
+
}
|
|
234
|
+
const iframe = document.createElement('iframe');
|
|
235
|
+
iframe.srcdoc = text;
|
|
236
|
+
container.innerHTML = '';
|
|
237
|
+
container.appendChild(iframe);
|
|
238
|
+
} catch {
|
|
239
|
+
const iframe = document.createElement('iframe');
|
|
240
|
+
iframe.src = file.url;
|
|
241
|
+
container.innerHTML = '';
|
|
242
|
+
container.appendChild(iframe);
|
|
243
|
+
}
|
|
244
|
+
}
|
|
245
|
+
|
|
246
|
+
async function renderPdfPreview(container, file) {
|
|
247
|
+
container.innerHTML = `<div class="pdf-container" id="pdfContainer"><div class="empty-state"><div class="spinner"></div><p class="loading-text">${t('loading_pdf')}</p></div></div>`;
|
|
248
|
+
try {
|
|
249
|
+
await loadScript('/static/pdf.min.js');
|
|
250
|
+
const pdfjsLib = window.pdfjsLib;
|
|
251
|
+
if (!pdfjsLib) throw new Error('pdf.js not loaded');
|
|
252
|
+
pdfjsLib.GlobalWorkerOptions.workerSrc = '/static/pdf.worker.min.js';
|
|
253
|
+
const pdf = await pdfjsLib.getDocument(file.url).promise;
|
|
254
|
+
const pdfContainer = container.querySelector('#pdfContainer');
|
|
255
|
+
pdfContainer.innerHTML = '';
|
|
256
|
+
const maxPages = Math.min(pdf.numPages, 30);
|
|
257
|
+
for (let i = 1; i <= maxPages; i++) {
|
|
258
|
+
const page = await pdf.getPage(i);
|
|
259
|
+
const viewport = page.getViewport({ scale: 1.5 });
|
|
260
|
+
const canvas = document.createElement('canvas');
|
|
261
|
+
canvas.width = viewport.width;
|
|
262
|
+
canvas.height = viewport.height;
|
|
263
|
+
await page.render({ canvasContext: canvas.getContext('2d'), viewport }).promise;
|
|
264
|
+
pdfContainer.appendChild(canvas);
|
|
265
|
+
}
|
|
266
|
+
if (pdf.numPages > maxPages) {
|
|
267
|
+
const p = document.createElement('p');
|
|
268
|
+
p.className = 'truncation-notice';
|
|
269
|
+
p.textContent = t('showing_pages', { max: maxPages, total: pdf.numPages });
|
|
270
|
+
pdfContainer.appendChild(p);
|
|
271
|
+
}
|
|
272
|
+
} catch (err) {
|
|
273
|
+
console.error('PDF render error:', err);
|
|
274
|
+
container.innerHTML = `<iframe src="${escapeHtml(file.url)}"></iframe>`;
|
|
275
|
+
}
|
|
276
|
+
}
|
|
277
|
+
|
|
278
|
+
async function renderMarkdownPreview(container, file, files, onSelectFile) {
|
|
279
|
+
try {
|
|
280
|
+
const resp = await fetch(file.url);
|
|
281
|
+
let text = await resp.text();
|
|
282
|
+
if (text.length > 500000) text = text.substring(0, 500000) + '\n\n... (truncated)';
|
|
283
|
+
const renderer = new marked.Renderer();
|
|
284
|
+
const origImage = renderer.image.bind(renderer);
|
|
285
|
+
const origLink = renderer.link.bind(renderer);
|
|
286
|
+
renderer.heading = function(token) {
|
|
287
|
+
let text = token.text;
|
|
288
|
+
let prev;
|
|
289
|
+
do { prev = text; text = text.replace(/<[^>]*>/g, ''); } while (text !== prev);
|
|
290
|
+
const slug = text.toLowerCase()
|
|
291
|
+
.replace(/[^\w\u0400-\u04ff\s-]/g, '')
|
|
292
|
+
.replace(/\s+/g, '-');
|
|
293
|
+
return `<h${token.depth} id="${slug}">${token.text}</h${token.depth}>\n`;
|
|
294
|
+
};
|
|
295
|
+
function resolveUrl(href) {
|
|
296
|
+
if (href && !href.startsWith('http') && !href.startsWith('//') && !href.startsWith('data:') && !href.startsWith('#')) {
|
|
297
|
+
const fileDir = file.path.includes('/') ? file.path.substring(0, file.path.lastIndexOf('/')) : '';
|
|
298
|
+
const base = fileDir ? FILES_BASE + '/' + fileDir : FILES_BASE;
|
|
299
|
+
return base + '/' + href;
|
|
300
|
+
}
|
|
301
|
+
return href;
|
|
302
|
+
}
|
|
303
|
+
renderer.image = function(token) { token.href = resolveUrl(token.href); return origImage(token); };
|
|
304
|
+
renderer.link = function(token) { token.href = resolveUrl(token.href); return origLink(token); };
|
|
305
|
+
const htmlContent = marked.parse(text, { renderer });
|
|
306
|
+
container.innerHTML = `<div class="markdown-body">${htmlContent}</div>`;
|
|
307
|
+
const mdBody = container.querySelector('.markdown-body');
|
|
308
|
+
|
|
309
|
+
// Link interception
|
|
310
|
+
mdBody.addEventListener('click', function(e) {
|
|
311
|
+
const a = e.target.closest('a');
|
|
312
|
+
if (!a) return;
|
|
313
|
+
e.preventDefault();
|
|
314
|
+
handleLinkClick(a.getAttribute('href'), a.href, files, file, onSelectFile);
|
|
315
|
+
});
|
|
316
|
+
|
|
317
|
+
// Syntax highlighting
|
|
318
|
+
mdBody.querySelectorAll('pre code').forEach(el => {
|
|
319
|
+
if (!el.classList.contains('language-mermaid')) hljs.highlightElement(el);
|
|
320
|
+
});
|
|
321
|
+
|
|
322
|
+
// Mermaid
|
|
323
|
+
const mermaidBlocks = mdBody.querySelectorAll('pre code.language-mermaid');
|
|
324
|
+
if (mermaidBlocks.length > 0) {
|
|
325
|
+
try {
|
|
326
|
+
await loadScript('/static/mermaid.min.js');
|
|
327
|
+
mermaid.initialize({ startOnLoad: false, theme: window.matchMedia('(prefers-color-scheme: dark)').matches ? 'dark' : 'default' });
|
|
328
|
+
mermaidBlocks.forEach(codeEl => {
|
|
329
|
+
const pre = codeEl.parentElement;
|
|
330
|
+
const div = document.createElement('div');
|
|
331
|
+
div.className = 'mermaid';
|
|
332
|
+
div.textContent = codeEl.textContent;
|
|
333
|
+
pre.replaceWith(div);
|
|
334
|
+
});
|
|
335
|
+
await mermaid.run({ querySelector: '.mermaid' });
|
|
336
|
+
} catch (e) { console.warn('Mermaid:', e); }
|
|
337
|
+
}
|
|
338
|
+
|
|
339
|
+
// KaTeX
|
|
340
|
+
try {
|
|
341
|
+
await loadScript('/static/katex/katex.min.js');
|
|
342
|
+
await loadScript('/static/katex/auto-render.min.js');
|
|
343
|
+
renderMathInElement(mdBody, {
|
|
344
|
+
delimiters: [
|
|
345
|
+
{ left: '$$', right: '$$', display: true },
|
|
346
|
+
{ left: '$', right: '$', display: false }
|
|
347
|
+
],
|
|
348
|
+
throwOnError: false
|
|
349
|
+
});
|
|
350
|
+
} catch (e) { console.warn('KaTeX:', e); }
|
|
351
|
+
} catch (err) {
|
|
352
|
+
console.error('Markdown render error:', err);
|
|
353
|
+
container.innerHTML = `<div class="empty-state"><p>${t('load_fail')}</p></div>`;
|
|
354
|
+
}
|
|
355
|
+
}
|
|
356
|
+
|
|
357
|
+
async function renderCodePreview(container, file) {
|
|
358
|
+
try {
|
|
359
|
+
const resp = await fetch(file.url);
|
|
360
|
+
let text = await resp.text();
|
|
361
|
+
if (text.length > 200000) text = text.substring(0, 200000) + '\n... (truncated)';
|
|
362
|
+
const ext = file.name.split('.').pop() || '';
|
|
363
|
+
container.innerHTML = `<pre><code class="language-${ext}">${escapeHtml(text)}</code></pre>`;
|
|
364
|
+
container.querySelectorAll('pre code').forEach(el => {
|
|
365
|
+
hljs.highlightElement(el);
|
|
366
|
+
if (typeof hljs.lineNumbersBlock === 'function') hljs.lineNumbersBlock(el);
|
|
367
|
+
});
|
|
368
|
+
} catch {
|
|
369
|
+
container.innerHTML = `<div class="empty-state"><p>${t('load_fail')}</p></div>`;
|
|
370
|
+
}
|
|
371
|
+
}
|
|
372
|
+
|
|
373
|
+
async function renderSpreadsheetPreview(container, file) {
|
|
374
|
+
try {
|
|
375
|
+
const resp = await fetch(file.url);
|
|
376
|
+
let text = await resp.text();
|
|
377
|
+
if (text.length > 500000) text = text.substring(0, 500000);
|
|
378
|
+
const ext = file.name.split('.').pop().toLowerCase();
|
|
379
|
+
const delimiter = ext === 'tsv' ? '\t' : ',';
|
|
380
|
+
const rows = parseCSV(text, delimiter);
|
|
381
|
+
let tableHtml = '<table><thead><tr>';
|
|
382
|
+
if (rows.length > 0) {
|
|
383
|
+
rows[0].forEach(cell => { tableHtml += `<th>${escapeHtml(cell)}</th>`; });
|
|
384
|
+
tableHtml += '</tr></thead><tbody>';
|
|
385
|
+
for (let i = 1; i < Math.min(rows.length, 1000); i++) {
|
|
386
|
+
tableHtml += '<tr>';
|
|
387
|
+
rows[i].forEach(cell => { tableHtml += `<td>${escapeHtml(cell)}</td>`; });
|
|
388
|
+
tableHtml += '</tr>';
|
|
389
|
+
}
|
|
390
|
+
tableHtml += '</tbody></table>';
|
|
391
|
+
if (rows.length > 1000) {
|
|
392
|
+
tableHtml += `<p class="truncation-notice">${t('showing_rows', { n: rows.length })}</p>`;
|
|
393
|
+
}
|
|
394
|
+
}
|
|
395
|
+
container.innerHTML = `<div class="data-table-wrap">${tableHtml}</div>`;
|
|
396
|
+
} catch {
|
|
397
|
+
container.innerHTML = `<div class="empty-state"><p>${t('load_fail')}</p></div>`;
|
|
398
|
+
}
|
|
399
|
+
}
|
|
400
|
+
|
|
401
|
+
async function renderDocxPreview(container, file) {
|
|
402
|
+
container.innerHTML = '<div class="markdown-body" id="docxContainer"><div class="spinner" style="margin:20px auto"></div></div>';
|
|
403
|
+
try {
|
|
404
|
+
await loadScript('/static/mammoth.browser.min.js');
|
|
405
|
+
const resp = await fetch(file.url);
|
|
406
|
+
const arrayBuffer = await resp.arrayBuffer();
|
|
407
|
+
const result = await mammoth.convertToHtml({ arrayBuffer });
|
|
408
|
+
container.querySelector('#docxContainer').innerHTML = result.value;
|
|
409
|
+
} catch (err) {
|
|
410
|
+
console.error('DOCX render error:', err);
|
|
411
|
+
renderDownloadFallback(container, file, 'fileText');
|
|
412
|
+
}
|
|
413
|
+
}
|
|
414
|
+
|
|
415
|
+
async function renderXlsxPreview(container, file) {
|
|
416
|
+
container.innerHTML = '<div class="data-table-wrap" id="xlsxContainer"><div class="spinner" style="margin:20px auto"></div></div>';
|
|
417
|
+
try {
|
|
418
|
+
await loadScript('/static/xlsx.full.min.js');
|
|
419
|
+
const resp = await fetch(file.url);
|
|
420
|
+
const arrayBuffer = await resp.arrayBuffer();
|
|
421
|
+
const workbook = XLSX.read(arrayBuffer, { type: 'array' });
|
|
422
|
+
let html = '';
|
|
423
|
+
if (workbook.SheetNames.length > 1) {
|
|
424
|
+
html += '<div class="sheet-tabs">';
|
|
425
|
+
workbook.SheetNames.forEach((name, i) => {
|
|
426
|
+
html += `<button class="sheet-tab${i === 0 ? ' active' : ''}" data-sheet="${i}">${escapeHtml(name)}</button>`;
|
|
427
|
+
});
|
|
428
|
+
html += '</div>';
|
|
429
|
+
}
|
|
430
|
+
html += '<div id="sheetContent"></div>';
|
|
431
|
+
const xlsxContainer = container.querySelector('#xlsxContainer');
|
|
432
|
+
xlsxContainer.innerHTML = html;
|
|
433
|
+
function renderSheet(index) {
|
|
434
|
+
const sheet = workbook.Sheets[workbook.SheetNames[index]];
|
|
435
|
+
xlsxContainer.querySelector('#sheetContent').innerHTML = XLSX.utils.sheet_to_html(sheet, { editable: false });
|
|
436
|
+
}
|
|
437
|
+
renderSheet(0);
|
|
438
|
+
xlsxContainer.querySelectorAll('.sheet-tab').forEach(btn => {
|
|
439
|
+
btn.addEventListener('click', function() {
|
|
440
|
+
xlsxContainer.querySelectorAll('.sheet-tab').forEach(b => b.classList.remove('active'));
|
|
441
|
+
this.classList.add('active');
|
|
442
|
+
renderSheet(parseInt(this.getAttribute('data-sheet')));
|
|
443
|
+
});
|
|
444
|
+
});
|
|
445
|
+
} catch (err) {
|
|
446
|
+
console.error('XLSX render error:', err);
|
|
447
|
+
renderDownloadFallback(container, file, 'fileSpreadsheet');
|
|
448
|
+
}
|
|
449
|
+
}
|
|
450
|
+
|
|
451
|
+
async function renderPptxPreview(container, file) {
|
|
452
|
+
container.innerHTML = `<div class="empty-state" id="pptxLoading"><div class="spinner"></div><p class="loading-text">${t('loading')}</p></div><div class="pptx-container" id="pptxContainer" style="display:none"></div>`;
|
|
453
|
+
try {
|
|
454
|
+
await loadScript('/static/jszip.min.js');
|
|
455
|
+
await loadScript('/static/chart.umd.js');
|
|
456
|
+
await loadScript('/static/pptxviewjs.min.js');
|
|
457
|
+
const pptxResp = await fetch(file.url);
|
|
458
|
+
const pptxBuf = await pptxResp.arrayBuffer();
|
|
459
|
+
const pptxContainer = container.querySelector('#pptxContainer');
|
|
460
|
+
const pptxWidth = container.clientWidth;
|
|
461
|
+
const viewer = new PptxViewJS.PPTXViewer();
|
|
462
|
+
await viewer.loadFile(pptxBuf);
|
|
463
|
+
const slideCount = viewer.getSlideCount();
|
|
464
|
+
for (let i = 0; i < slideCount; i++) {
|
|
465
|
+
const canvas = document.createElement('canvas');
|
|
466
|
+
canvas.width = pptxWidth;
|
|
467
|
+
canvas.height = Math.round(pptxWidth * 9 / 16);
|
|
468
|
+
canvas.className = 'pptx-slide';
|
|
469
|
+
pptxContainer.appendChild(canvas);
|
|
470
|
+
viewer.setCanvas(canvas);
|
|
471
|
+
await viewer.goToSlide(i);
|
|
472
|
+
await viewer.render();
|
|
473
|
+
}
|
|
474
|
+
container.querySelector('#pptxLoading')?.remove();
|
|
475
|
+
pptxContainer.style.display = '';
|
|
476
|
+
} catch (err) {
|
|
477
|
+
console.error('PPTX render error:', err);
|
|
478
|
+
renderDownloadFallback(container, file, 'filePresentation', t('pptx_fail'));
|
|
479
|
+
}
|
|
480
|
+
}
|
|
481
|
+
|
|
482
|
+
async function renderDrawioPreview(container, file) {
|
|
483
|
+
container.innerHTML = `<div class="empty-state"><div class="spinner"></div><p class="loading-text">${t('loading')}</p></div>`;
|
|
484
|
+
try {
|
|
485
|
+
const drawioResp = await fetch(file.url);
|
|
486
|
+
const drawioXml = await drawioResp.text();
|
|
487
|
+
const mxDiv = document.createElement('div');
|
|
488
|
+
mxDiv.className = 'mxgraph';
|
|
489
|
+
mxDiv.style.cssText = 'max-width:100%;margin:0 auto;background:#fff;padding:20px;border-radius:8px;';
|
|
490
|
+
mxDiv.setAttribute('data-mxgraph', JSON.stringify({ xml: drawioXml }));
|
|
491
|
+
container.innerHTML = '';
|
|
492
|
+
container.style.display = 'flex';
|
|
493
|
+
container.style.justifyContent = 'center';
|
|
494
|
+
container.style.alignItems = 'flex-start';
|
|
495
|
+
container.appendChild(mxDiv);
|
|
496
|
+
const viewerUrl = 'https://viewer.diagrams.net/js/viewer-static.min.js';
|
|
497
|
+
if (_loadedScripts.has(viewerUrl) && window.GraphViewer) {
|
|
498
|
+
GraphViewer.processElements();
|
|
499
|
+
} else {
|
|
500
|
+
await loadScript(viewerUrl);
|
|
501
|
+
}
|
|
502
|
+
} catch (err) {
|
|
503
|
+
console.error('Draw.io render error:', err);
|
|
504
|
+
renderDownloadFallback(container, file, 'diagram', t('drawio_fail'));
|
|
505
|
+
}
|
|
506
|
+
}
|
|
507
|
+
|
|
508
|
+
function renderDownloadFallback(container, file, iconType, errorMsg) {
|
|
509
|
+
container.innerHTML = `<div class="download-prompt">
|
|
510
|
+
<div class="dl-icon">${fileIconLarge(file.type)}</div>
|
|
511
|
+
<div class="dl-name">${escapeHtml(file.name)}</div>
|
|
512
|
+
<div class="dl-size">${formatSize(file.size)}</div>
|
|
513
|
+
${errorMsg ? `<div class="dl-error">${escapeHtml(errorMsg)}</div>` : ''}
|
|
514
|
+
<a class="btn" href="${escapeHtml(file.url)}" download>${icon('download')} ${t('download')}</a>
|
|
515
|
+
</div>`;
|
|
516
|
+
}
|
|
517
|
+
|
|
518
|
+
function renderPreviewContent(container, file, files, onSelectFile) {
|
|
519
|
+
switch (file.type) {
|
|
520
|
+
case 'html': renderHtmlPreview(container, file); break;
|
|
521
|
+
case 'image':
|
|
522
|
+
container.innerHTML = `<img src="${escapeHtml(file.url)}" alt="${escapeHtml(file.name)}">`;
|
|
523
|
+
break;
|
|
524
|
+
case 'pdf': renderPdfPreview(container, file); break;
|
|
525
|
+
case 'markdown': renderMarkdownPreview(container, file, files, onSelectFile); break;
|
|
526
|
+
case 'code':
|
|
527
|
+
case 'text': renderCodePreview(container, file); break;
|
|
528
|
+
case 'spreadsheet': renderSpreadsheetPreview(container, file); break;
|
|
529
|
+
case 'docx': renderDocxPreview(container, file); break;
|
|
530
|
+
case 'xlsx': renderXlsxPreview(container, file); break;
|
|
531
|
+
case 'pptx': renderPptxPreview(container, file); break;
|
|
532
|
+
case 'drawio': renderDrawioPreview(container, file); break;
|
|
533
|
+
case 'audio':
|
|
534
|
+
container.innerHTML = `<div class="media-container">
|
|
535
|
+
<div class="media-icon">${icon('music', 48)}</div>
|
|
536
|
+
<div class="media-name">${escapeHtml(file.name)}</div>
|
|
537
|
+
<div class="media-size">${formatSize(file.size)}</div>
|
|
538
|
+
<audio controls preload="metadata" src="${escapeHtml(file.url)}">${t('audio_unsupported')}</audio>
|
|
539
|
+
<a class="btn" href="${escapeHtml(file.url)}" download>${icon('download')} ${t('download')}</a>
|
|
540
|
+
</div>`;
|
|
541
|
+
break;
|
|
542
|
+
case 'video':
|
|
543
|
+
container.innerHTML = `<div class="media-container">
|
|
544
|
+
<video controls preload="metadata" src="${escapeHtml(file.url)}">${t('video_unsupported')}</video>
|
|
545
|
+
<a class="btn" href="${escapeHtml(file.url)}" download>${icon('download')} ${t('download')}</a>
|
|
546
|
+
</div>`;
|
|
547
|
+
break;
|
|
548
|
+
default:
|
|
549
|
+
renderDownloadFallback(container, file);
|
|
550
|
+
}
|
|
551
|
+
}
|
|
552
|
+
|
|
553
|
+
// =============================================================================
|
|
554
|
+
// Components
|
|
555
|
+
// =============================================================================
|
|
556
|
+
|
|
557
|
+
function ViewTabs({ currentView, onSwitch, browserActive, terminalActive }) {
|
|
558
|
+
return html`
|
|
559
|
+
<div class="view-tabs">
|
|
560
|
+
<button class="view-tab ${currentView === 'files' ? 'active' : ''}"
|
|
561
|
+
onClick=${() => onSwitch('files')}>
|
|
562
|
+
<span class="icon-inline" dangerouslySetInnerHTML=${{ __html: icon('folder') }}></span> ${t('files')}
|
|
563
|
+
</button>
|
|
564
|
+
<button class="view-tab ${currentView === 'browser' ? 'active' : ''}"
|
|
565
|
+
onClick=${() => onSwitch('browser')}>
|
|
566
|
+
<span class="icon-inline" dangerouslySetInnerHTML=${{ __html: icon('globe') }}></span> ${t('browser')}
|
|
567
|
+
${browserActive && html`<span class="tab-dot"></span>`}
|
|
568
|
+
</button>
|
|
569
|
+
<button class="view-tab ${currentView === 'terminal' ? 'active' : ''}"
|
|
570
|
+
onClick=${() => onSwitch('terminal')}>
|
|
571
|
+
<span class="icon-inline" dangerouslySetInnerHTML=${{ __html: icon('terminal') }}></span> ${t('subagent')}
|
|
572
|
+
<span class="beta-badge">${t('beta')}</span>
|
|
573
|
+
${terminalActive && html`<span class="tab-dot"></span>`}
|
|
574
|
+
</button>
|
|
575
|
+
</div>
|
|
576
|
+
`;
|
|
577
|
+
}
|
|
578
|
+
|
|
579
|
+
function FileSelector({ files, selectedFile, seenFiles, onSelect }) {
|
|
580
|
+
const [open, setOpen] = useState(false);
|
|
581
|
+
const [dropdownPath, setDropdownPath] = useState('');
|
|
582
|
+
const ref = useRef(null);
|
|
583
|
+
|
|
584
|
+
// Close on click outside
|
|
585
|
+
useEffect(() => {
|
|
586
|
+
const handler = (e) => {
|
|
587
|
+
if (ref.current && !ref.current.contains(e.target)) setOpen(false);
|
|
588
|
+
};
|
|
589
|
+
document.addEventListener('click', handler);
|
|
590
|
+
return () => document.removeEventListener('click', handler);
|
|
591
|
+
}, []);
|
|
592
|
+
|
|
593
|
+
const prefix = dropdownPath ? dropdownPath + '/' : '';
|
|
594
|
+
const folders = new Set();
|
|
595
|
+
const currentFiles = [];
|
|
596
|
+
for (const f of files) {
|
|
597
|
+
if (prefix && !f.path.startsWith(prefix)) continue;
|
|
598
|
+
const rest = prefix ? f.path.substring(prefix.length) : f.path;
|
|
599
|
+
if (rest.includes('/')) { folders.add(rest.substring(0, rest.indexOf('/'))); }
|
|
600
|
+
else { currentFiles.push(f); }
|
|
601
|
+
}
|
|
602
|
+
|
|
603
|
+
const selectedIcon = selectedFile ? fileIcon(selectedFile.type) : icon('folderOpen');
|
|
604
|
+
const hasNew = files.some(f => !seenFiles.has(f.path));
|
|
605
|
+
|
|
606
|
+
return html`
|
|
607
|
+
<div class="file-selector" ref=${ref}>
|
|
608
|
+
<button class="file-selector-btn ${open ? 'open' : ''}" type="button"
|
|
609
|
+
onClick=${(e) => { e.stopPropagation(); setOpen(!open); setDropdownPath(''); }}>
|
|
610
|
+
<span class="icon-inline" dangerouslySetInnerHTML=${{ __html: selectedIcon }}></span>
|
|
611
|
+
<span class="selector-name">${selectedFile ? selectedFile.name : t('no_file_selected')}</span>
|
|
612
|
+
<span class="selector-count">${files.length}</span>
|
|
613
|
+
${hasNew && html`<span class="selector-new-dot"></span>`}
|
|
614
|
+
<span class="selector-arrow icon-inline" dangerouslySetInnerHTML=${{ __html: icon('chevronDown') }}></span>
|
|
615
|
+
</button>
|
|
616
|
+
<ul class="dropdown-menu ${open ? 'open' : ''}">
|
|
617
|
+
${dropdownPath && html`
|
|
618
|
+
<li class="dropdown-item back" onClick=${(e) => {
|
|
619
|
+
e.stopPropagation();
|
|
620
|
+
setDropdownPath(dropdownPath.includes('/') ? dropdownPath.substring(0, dropdownPath.lastIndexOf('/')) : '');
|
|
621
|
+
}}>
|
|
622
|
+
<span class="item-icon icon-inline" dangerouslySetInnerHTML=${{ __html: icon('arrowLeft') }}></span>
|
|
623
|
+
<span class="item-name">..</span>
|
|
624
|
+
</li>
|
|
625
|
+
`}
|
|
626
|
+
${[...folders].sort().map(folder => html`
|
|
627
|
+
<li class="dropdown-item folder" onClick=${(e) => { e.stopPropagation(); setDropdownPath(prefix + folder); }}>
|
|
628
|
+
<span class="item-icon icon-inline" dangerouslySetInnerHTML=${{ __html: icon('folder') }}></span>
|
|
629
|
+
<span class="item-name">${folder}</span>
|
|
630
|
+
<span class="item-chevron icon-inline" dangerouslySetInnerHTML=${{ __html: icon('chevronRight') }}></span>
|
|
631
|
+
</li>
|
|
632
|
+
`)}
|
|
633
|
+
${currentFiles.map(f => html`
|
|
634
|
+
<li class="dropdown-item ${selectedFile && f.path === selectedFile.path ? 'active' : ''}"
|
|
635
|
+
onClick=${() => { onSelect(f); setOpen(false); }}>
|
|
636
|
+
<span class="item-icon icon-inline" dangerouslySetInnerHTML=${{ __html: fileIcon(f.type) }}></span>
|
|
637
|
+
<span class="item-name">${f.name}</span>
|
|
638
|
+
<span class="item-size">${formatSize(f.size)}</span>
|
|
639
|
+
${!seenFiles.has(f.path) && html`<span class="badge-new"></span>`}
|
|
640
|
+
</li>
|
|
641
|
+
`)}
|
|
642
|
+
</ul>
|
|
643
|
+
</div>
|
|
644
|
+
`;
|
|
645
|
+
}
|
|
646
|
+
|
|
647
|
+
function FilesView({ files, selectedFile, onSelectFile }) {
|
|
648
|
+
const containerRef = useRef(null);
|
|
649
|
+
const prevFileRef = useRef(null);
|
|
650
|
+
|
|
651
|
+
useEffect(() => {
|
|
652
|
+
if (!containerRef.current) return;
|
|
653
|
+
if (selectedFile && selectedFile !== prevFileRef.current) {
|
|
654
|
+
prevFileRef.current = selectedFile;
|
|
655
|
+
renderPreviewContent(containerRef.current, selectedFile, files, onSelectFile);
|
|
656
|
+
}
|
|
657
|
+
}, [selectedFile, files]);
|
|
658
|
+
|
|
659
|
+
if (!selectedFile) {
|
|
660
|
+
return html`
|
|
661
|
+
<div class="preview">
|
|
662
|
+
<div class="empty-state">
|
|
663
|
+
<div class="empty-icon" dangerouslySetInnerHTML=${{ __html: icon('folder', 48) }}></div>
|
|
664
|
+
<div class="empty-title">${t('no_files_yet')}</div>
|
|
665
|
+
<div class="empty-desc" dangerouslySetInnerHTML=${{ __html: t('no_files_desc') }}></div>
|
|
666
|
+
<div class="empty-example">${t('no_files_example')}</div>
|
|
667
|
+
</div>
|
|
668
|
+
</div>
|
|
669
|
+
`;
|
|
670
|
+
}
|
|
671
|
+
|
|
672
|
+
return html`<div class="preview" ref=${containerRef}></div>`;
|
|
673
|
+
}
|
|
674
|
+
|
|
675
|
+
function BrowserView({ chatId, browserActive, onBrowserViewerRef }) {
|
|
676
|
+
const canvasRef = useRef(null);
|
|
677
|
+
const urlBarRef = useRef(null);
|
|
678
|
+
const viewerRef = useRef(null);
|
|
679
|
+
const [connecting, setConnecting] = useState(false);
|
|
680
|
+
const [connected, setConnected] = useState(false);
|
|
681
|
+
const [error, setError] = useState(false);
|
|
682
|
+
|
|
683
|
+
useEffect(() => {
|
|
684
|
+
if (!browserActive) {
|
|
685
|
+
if (viewerRef.current) { viewerRef.current.disconnect(); viewerRef.current = null; }
|
|
686
|
+
setConnected(false);
|
|
687
|
+
setConnecting(false);
|
|
688
|
+
setError(false);
|
|
689
|
+
return;
|
|
690
|
+
}
|
|
691
|
+
if (viewerRef.current) return;
|
|
692
|
+
|
|
693
|
+
setConnecting(true);
|
|
694
|
+
setError(false);
|
|
695
|
+
const canvas = canvasRef.current;
|
|
696
|
+
if (!canvas) return;
|
|
697
|
+
const viewer = new BrowserViewer(canvas, chatId);
|
|
698
|
+
viewerRef.current = viewer;
|
|
699
|
+
if (onBrowserViewerRef) onBrowserViewerRef(viewer);
|
|
700
|
+
viewer.connect().then(ok => {
|
|
701
|
+
setConnecting(false);
|
|
702
|
+
if (ok) {
|
|
703
|
+
setConnected(true);
|
|
704
|
+
canvas.focus();
|
|
705
|
+
} else {
|
|
706
|
+
setError(true);
|
|
707
|
+
viewerRef.current = null;
|
|
708
|
+
}
|
|
709
|
+
});
|
|
710
|
+
|
|
711
|
+
return () => {
|
|
712
|
+
if (viewerRef.current) { viewerRef.current.disconnect(); viewerRef.current = null; }
|
|
713
|
+
};
|
|
714
|
+
}, [browserActive]);
|
|
715
|
+
|
|
716
|
+
if (!browserActive) {
|
|
717
|
+
return html`
|
|
718
|
+
<div class="browser-panel">
|
|
719
|
+
<div class="empty-state">
|
|
720
|
+
<div class="empty-icon" dangerouslySetInnerHTML=${{ __html: icon('globe', 48) }}></div>
|
|
721
|
+
<div class="empty-title">${t('browser_title')}</div>
|
|
722
|
+
<div class="empty-desc" dangerouslySetInnerHTML=${{ __html: t('browser_desc') }}></div>
|
|
723
|
+
<div class="empty-example" dangerouslySetInnerHTML=${{ __html: t('browser_example') }}></div>
|
|
724
|
+
</div>
|
|
725
|
+
</div>
|
|
726
|
+
`;
|
|
727
|
+
}
|
|
728
|
+
|
|
729
|
+
return html`
|
|
730
|
+
<div class="browser-panel">
|
|
731
|
+
${connecting && html`<div class="browser-connecting"><div class="spinner"></div> ${t('loading_browser')}</div>`}
|
|
732
|
+
${error && html`<div class="browser-connecting">${t('browser_connect_fail')}</div>`}
|
|
733
|
+
<canvas ref=${canvasRef} tabindex="0" style="display:${connected ? '' : 'none'};flex:1;width:100%;object-fit:contain;cursor:pointer;background:var(--bg-primary)"></canvas>
|
|
734
|
+
<div ref=${urlBarRef} class="browser-url-bar" id="browserUrlBar" style="display:${connected ? '' : 'none'}"></div>
|
|
735
|
+
</div>
|
|
736
|
+
`;
|
|
737
|
+
}
|
|
738
|
+
|
|
739
|
+
function TerminalDashboard({ chatId, dangerousMode, onToggleDangerous, onStartSession, onResumeSession }) {
|
|
740
|
+
const [data, setData] = useState(null);
|
|
741
|
+
const [loading, setLoading] = useState(true);
|
|
742
|
+
|
|
743
|
+
const fetchData = useCallback(async () => {
|
|
744
|
+
setLoading(true);
|
|
745
|
+
try {
|
|
746
|
+
const [sResp, sessResp, procResp, uplResp] = await Promise.all([
|
|
747
|
+
fetch(`/terminal/${chatId}/status?_t=${Date.now()}`),
|
|
748
|
+
fetch(`/terminal/${chatId}/sessions?_t=${Date.now()}`),
|
|
749
|
+
fetch(`/terminal/${chatId}/processes?_t=${Date.now()}`),
|
|
750
|
+
fetch(`/api/uploads/${chatId}/list?_t=${Date.now()}`),
|
|
751
|
+
]);
|
|
752
|
+
setData({
|
|
753
|
+
status: await sResp.json(),
|
|
754
|
+
sessions: await sessResp.json(),
|
|
755
|
+
processes: await procResp.json(),
|
|
756
|
+
uploads: await uplResp.json(),
|
|
757
|
+
});
|
|
758
|
+
} catch(e) {
|
|
759
|
+
setData({ status: { active: false }, sessions: { sessions: [] }, processes: { processes: [] }, uploads: { files: [], total: 0 } });
|
|
760
|
+
}
|
|
761
|
+
setLoading(false);
|
|
762
|
+
}, [chatId]);
|
|
763
|
+
|
|
764
|
+
useEffect(() => { fetchData(); }, []);
|
|
765
|
+
|
|
766
|
+
const uploadFile = useCallback((evt) => {
|
|
767
|
+
if (evt) evt.stopPropagation();
|
|
768
|
+
const input = document.createElement('input');
|
|
769
|
+
input.type = 'file';
|
|
770
|
+
input.multiple = true;
|
|
771
|
+
input.style.display = 'none';
|
|
772
|
+
document.body.appendChild(input);
|
|
773
|
+
input.addEventListener('change', async () => {
|
|
774
|
+
for (const file of input.files) {
|
|
775
|
+
const formData = new FormData();
|
|
776
|
+
formData.append('file', file);
|
|
777
|
+
await fetch(`/api/uploads/${chatId}/${encodeURIComponent(file.name)}`, { method: 'POST', body: formData });
|
|
778
|
+
}
|
|
779
|
+
input.remove();
|
|
780
|
+
fetchData();
|
|
781
|
+
});
|
|
782
|
+
input.click();
|
|
783
|
+
}, [chatId]);
|
|
784
|
+
|
|
785
|
+
const killProcess = useCallback(async (pid) => {
|
|
786
|
+
try { await fetch(`/terminal/${chatId}/processes/${pid}/kill`, { method: 'POST' }); } catch(e) {}
|
|
787
|
+
fetchData();
|
|
788
|
+
}, [chatId]);
|
|
789
|
+
|
|
790
|
+
const copyPath = useCallback((path) => {
|
|
791
|
+
copyText(path).then(() => showToast(t('copied')));
|
|
792
|
+
}, []);
|
|
793
|
+
|
|
794
|
+
if (loading || !data) {
|
|
795
|
+
return html`<div class="dash-scroll" style="display:flex;align-items:center;justify-content:center;height:100%"><div class="spinner"></div></div>`;
|
|
796
|
+
}
|
|
797
|
+
|
|
798
|
+
const { status, sessions, processes, uploads } = data;
|
|
799
|
+
const hasProcesses = processes.processes.length > 0;
|
|
800
|
+
const hasSessions = sessions.sessions.length > 0;
|
|
801
|
+
const hasUploads = uploads.files && uploads.files.length > 0;
|
|
802
|
+
|
|
803
|
+
return html`
|
|
804
|
+
<div class="dash-scroll">
|
|
805
|
+
<div class="dash-hero">
|
|
806
|
+
<h3>Claude Code <span class="badge">${t('beta')}</span></h3>
|
|
807
|
+
<p>${t('hero_desc')}</p>
|
|
808
|
+
<p class="warn">${t('pd_warning')}</p>
|
|
809
|
+
<div class="dangerous-toggle-row ${dangerousMode ? 'active' : ''}" onClick=${() => onToggleDangerous(!dangerousMode)}>
|
|
810
|
+
<div class="dangerous-toggle-track ${dangerousMode ? 'on' : ''}">
|
|
811
|
+
<div class="dangerous-toggle-knob"></div>
|
|
812
|
+
</div>
|
|
813
|
+
<div class="dangerous-toggle-label">
|
|
814
|
+
<span class="dangerous-toggle-title">${t('skip_perms')}</span>
|
|
815
|
+
${dangerousMode && html`<span class="dangerous-toggle-warn">${t('dangerous_warn')}</span>`}
|
|
816
|
+
</div>
|
|
817
|
+
</div>
|
|
818
|
+
<div class="dash-actions">
|
|
819
|
+
<button class="dash-btn-primary" onClick=${() => onStartSession()}>
|
|
820
|
+
<span class="icon-inline" dangerouslySetInnerHTML=${{ __html: icon('play') }}></span> ${t('open_terminal')}
|
|
821
|
+
</button>
|
|
822
|
+
<button class="dash-btn-secondary" onClick=${uploadFile}>
|
|
823
|
+
<span class="icon-inline" dangerouslySetInnerHTML=${{ __html: icon('upload') }}></span> ${t('upload_file')}
|
|
824
|
+
</button>
|
|
825
|
+
<a class="dash-link" href="https://github.com/Yambr/open-computer-use/blob/main/docs/TERMINAL-TAB.md" target="_blank">
|
|
826
|
+
<span class="icon-inline" dangerouslySetInnerHTML=${{ __html: icon('book') }}></span> ${t('how_to_use')}
|
|
827
|
+
</a>
|
|
828
|
+
</div>
|
|
829
|
+
</div>
|
|
830
|
+
|
|
831
|
+
${hasProcesses && html`
|
|
832
|
+
<div class="dash-card">
|
|
833
|
+
<h4><span class="icon-inline" dangerouslySetInnerHTML=${{ __html: icon('settings') }}></span> ${t('running_now')}</h4>
|
|
834
|
+
${processes.processes.map(p => {
|
|
835
|
+
const mins = p.elapsed_minutes || 0;
|
|
836
|
+
const timeStr = mins < 1 ? t('less_than_min') : mins + t('min_suffix');
|
|
837
|
+
return html`
|
|
838
|
+
<div class="dash-process">
|
|
839
|
+
<span class="dot"></span>
|
|
840
|
+
<span>${t('claude_running')} · ${timeStr}</span>
|
|
841
|
+
<button class="action-btn kill-btn" onClick=${() => killProcess(p.pid)}>${t('stop')}</button>
|
|
842
|
+
</div>
|
|
843
|
+
`;
|
|
844
|
+
})}
|
|
845
|
+
</div>
|
|
846
|
+
`}
|
|
847
|
+
|
|
848
|
+
${hasUploads && html`
|
|
849
|
+
<div class="dash-card">
|
|
850
|
+
<h4><span class="icon-inline" dangerouslySetInnerHTML=${{ __html: icon('paperclip') }}></span> ${t('uploaded_files')}</h4>
|
|
851
|
+
<table class="dash-table">
|
|
852
|
+
<thead><tr><th>${t('th_file')}</th><th>${t('th_size')}</th><th></th></tr></thead>
|
|
853
|
+
<tbody>
|
|
854
|
+
${uploads.files.map(f => html`
|
|
855
|
+
<tr>
|
|
856
|
+
<td>${f.name}</td>
|
|
857
|
+
<td style="white-space:nowrap">${formatSize(f.size)}</td>
|
|
858
|
+
<td><button class="action-btn" onClick=${() => copyPath(f.container_path)} title=${f.container_path}>
|
|
859
|
+
<span class="icon-inline" dangerouslySetInnerHTML=${{ __html: icon('copy') }}></span> ${t('copy_path')}
|
|
860
|
+
</button></td>
|
|
861
|
+
</tr>
|
|
862
|
+
`)}
|
|
863
|
+
</tbody>
|
|
864
|
+
</table>
|
|
865
|
+
</div>
|
|
866
|
+
`}
|
|
867
|
+
|
|
868
|
+
${hasSessions && html`
|
|
869
|
+
<div class="dash-card">
|
|
870
|
+
<h4><span class="icon-inline" dangerouslySetInnerHTML=${{ __html: icon('fileText') }}></span> ${t('prev_sessions')}</h4>
|
|
871
|
+
<table class="dash-table">
|
|
872
|
+
<thead><tr><th>${t('th_task')}</th><th>${t('th_date')}</th><th></th></tr></thead>
|
|
873
|
+
<tbody>
|
|
874
|
+
${sessions.sessions.map(s => {
|
|
875
|
+
const dt = s.timestamp ? new Date(s.timestamp * 1000).toLocaleDateString(LANG === 'ru' ? 'ru-RU' : 'en-US', { day: 'numeric', month: 'short' }) : '';
|
|
876
|
+
const name = (s.label || s.session_id.substring(0, 16) + '...').substring(0, 50);
|
|
877
|
+
return html`
|
|
878
|
+
<tr>
|
|
879
|
+
<td title=${s.session_id}>${name}</td>
|
|
880
|
+
<td>${dt}</td>
|
|
881
|
+
<td><button class="action-btn" onClick=${() => onResumeSession(s.session_id)}>${t('resume')}</button></td>
|
|
882
|
+
</tr>
|
|
883
|
+
`;
|
|
884
|
+
})}
|
|
885
|
+
</tbody>
|
|
886
|
+
</table>
|
|
887
|
+
</div>
|
|
888
|
+
`}
|
|
889
|
+
|
|
890
|
+
${!status.active && !hasSessions && !hasProcesses && html`
|
|
891
|
+
<div class="dash-hint">${t('dash_hint')}</div>
|
|
892
|
+
`}
|
|
893
|
+
</div>
|
|
894
|
+
`;
|
|
895
|
+
}
|
|
896
|
+
|
|
897
|
+
function TerminalSession({ chatId, resumeId, dangerousMode, onBack }) {
|
|
898
|
+
const containerRef = useRef(null);
|
|
899
|
+
const xtermRef = useRef(null);
|
|
900
|
+
const fitAddonRef = useRef(null);
|
|
901
|
+
const wsRef = useRef(null);
|
|
902
|
+
const reconnectAttemptsRef = useRef(0);
|
|
903
|
+
const dangerousModeRef = useRef(dangerousMode);
|
|
904
|
+
useEffect(() => { dangerousModeRef.current = dangerousMode; }, [dangerousMode]);
|
|
905
|
+
const [startError, setStartError] = useState(null);
|
|
906
|
+
const [selectMode, setSelectMode] = useState(false);
|
|
907
|
+
|
|
908
|
+
const connectWs = useCallback((rId) => {
|
|
909
|
+
const proto = location.protocol === 'https:' ? 'wss:' : 'ws:';
|
|
910
|
+
const ws = new WebSocket(`${proto}//${location.host}/terminal/${chatId}/ws`, ['tty']);
|
|
911
|
+
ws.binaryType = 'arraybuffer';
|
|
912
|
+
wsRef.current = ws;
|
|
913
|
+
|
|
914
|
+
let receivedData = false;
|
|
915
|
+
ws.onopen = () => {
|
|
916
|
+
const dims = fitAddonRef.current ? fitAddonRef.current.proposeDimensions() : { cols: 80, rows: 24 };
|
|
917
|
+
ws.send(JSON.stringify({ authToken: '', columns: dims.cols || 80, rows: dims.rows || 24 }));
|
|
918
|
+
if (rId) {
|
|
919
|
+
// Check if Claude Code is running, then send appropriate resume sequence
|
|
920
|
+
(async () => {
|
|
921
|
+
const dangerous = dangerousModeRef.current;
|
|
922
|
+
const flagSuffix = dangerous ? ' --dangerously-skip-permissions' : '';
|
|
923
|
+
try {
|
|
924
|
+
const resp = await fetch(`/terminal/${chatId}/processes?_t=${Date.now()}`);
|
|
925
|
+
const data = await resp.json();
|
|
926
|
+
const hasClaudeRunning = data.processes && data.processes.length > 0;
|
|
927
|
+
if (hasClaudeRunning) {
|
|
928
|
+
// Claude Code active — double Ctrl+C to exit, then resume
|
|
929
|
+
ws.send(new TextEncoder().encode('\x30\x03')); // Ctrl+C #1
|
|
930
|
+
await new Promise(r => setTimeout(r, 500));
|
|
931
|
+
ws.send(new TextEncoder().encode('\x30\x03')); // Ctrl+C #2
|
|
932
|
+
await new Promise(r => setTimeout(r, 1500));
|
|
933
|
+
} else {
|
|
934
|
+
// No Claude Code — just wait for bash prompt
|
|
935
|
+
await new Promise(r => setTimeout(r, 1000));
|
|
936
|
+
}
|
|
937
|
+
ws.send(new TextEncoder().encode('\x30claude --resume ' + rId + flagSuffix + '\r'));
|
|
938
|
+
} catch(e) {
|
|
939
|
+
// Fallback: just send resume after delay
|
|
940
|
+
await new Promise(r => setTimeout(r, 2000));
|
|
941
|
+
ws.send(new TextEncoder().encode('\x30claude --resume ' + rId + (dangerousModeRef.current ? ' --dangerously-skip-permissions' : '') + '\r'));
|
|
942
|
+
}
|
|
943
|
+
})();
|
|
944
|
+
} else if (dangerousModeRef.current) {
|
|
945
|
+
// New session in dangerous mode — NO_AUTOSTART=1 prevented .bashrc autostart,
|
|
946
|
+
// so we wait for the bash prompt and inject the command with the flag.
|
|
947
|
+
// For reconnects where claude is already running — leave it undisturbed.
|
|
948
|
+
(async () => {
|
|
949
|
+
await new Promise(r => setTimeout(r, 1500));
|
|
950
|
+
try {
|
|
951
|
+
const resp = await fetch(`/terminal/${chatId}/processes?_t=${Date.now()}`);
|
|
952
|
+
const data = await resp.json();
|
|
953
|
+
const hasClaudeRunning = data.processes && data.processes.length > 0;
|
|
954
|
+
if (!hasClaudeRunning && ws.readyState === WebSocket.OPEN) {
|
|
955
|
+
ws.send(new TextEncoder().encode('\x30claude --dangerously-skip-permissions\r'));
|
|
956
|
+
}
|
|
957
|
+
} catch(e) {
|
|
958
|
+
if (ws.readyState === WebSocket.OPEN)
|
|
959
|
+
ws.send(new TextEncoder().encode('\x30claude --dangerously-skip-permissions\r'));
|
|
960
|
+
}
|
|
961
|
+
})();
|
|
962
|
+
}
|
|
963
|
+
};
|
|
964
|
+
|
|
965
|
+
ws.onmessage = (ev) => {
|
|
966
|
+
if (!receivedData) { receivedData = true; reconnectAttemptsRef.current = 0; }
|
|
967
|
+
if (ev.data instanceof ArrayBuffer && xtermRef.current) {
|
|
968
|
+
const data = new Uint8Array(ev.data);
|
|
969
|
+
if (data.length > 1) {
|
|
970
|
+
const type = data[0];
|
|
971
|
+
if (type === 0x30) { xtermRef.current.write(data.slice(1)); }
|
|
972
|
+
else if (type === 0x31 || type === 0x32) { /* ignore title/prefs */ }
|
|
973
|
+
else { xtermRef.current.write(data); }
|
|
974
|
+
}
|
|
975
|
+
}
|
|
976
|
+
};
|
|
977
|
+
|
|
978
|
+
ws.onclose = () => {
|
|
979
|
+
if (xtermRef.current) {
|
|
980
|
+
if (reconnectAttemptsRef.current < 3) {
|
|
981
|
+
const delay = 1000 * Math.pow(2, reconnectAttemptsRef.current);
|
|
982
|
+
reconnectAttemptsRef.current++;
|
|
983
|
+
setTimeout(() => connectWs(), delay);
|
|
984
|
+
} else {
|
|
985
|
+
xtermRef.current.write('\r\n\x1b[90m' + t('session_ended') + '\x1b[0m\r\n');
|
|
986
|
+
setTimeout(() => onBack(), 2000);
|
|
987
|
+
}
|
|
988
|
+
}
|
|
989
|
+
};
|
|
990
|
+
}, [chatId, onBack]);
|
|
991
|
+
|
|
992
|
+
const [ready, setReady] = useState(false);
|
|
993
|
+
|
|
994
|
+
// Phase 1: start ttyd (if not already running)
|
|
995
|
+
useEffect(() => {
|
|
996
|
+
let cancelled = false;
|
|
997
|
+
(async () => {
|
|
998
|
+
try {
|
|
999
|
+
const resp = await fetch(`/terminal/${chatId}/start-ttyd`, {
|
|
1000
|
+
method: 'POST',
|
|
1001
|
+
headers: { 'Content-Type': 'application/json' },
|
|
1002
|
+
body: JSON.stringify({ dangerous_mode: dangerousModeRef.current }),
|
|
1003
|
+
});
|
|
1004
|
+
if (!resp.ok) {
|
|
1005
|
+
if (!cancelled) {
|
|
1006
|
+
// Check if container is stopped (can be restarted) or removed with meta (can be resurrected)
|
|
1007
|
+
try {
|
|
1008
|
+
const statusResp = await fetch(`/terminal/${chatId}/status?_t=${Date.now()}`);
|
|
1009
|
+
const statusData = await statusResp.json();
|
|
1010
|
+
if (statusData.container_stopped) {
|
|
1011
|
+
setStartError('__stopped__');
|
|
1012
|
+
} else if (statusData.meta_exists) {
|
|
1013
|
+
setStartError('__meta_exists__');
|
|
1014
|
+
} else {
|
|
1015
|
+
setStartError(t('terminal_fail'));
|
|
1016
|
+
}
|
|
1017
|
+
} catch { setStartError(t('terminal_fail')); }
|
|
1018
|
+
}
|
|
1019
|
+
return;
|
|
1020
|
+
}
|
|
1021
|
+
const data = await resp.json();
|
|
1022
|
+
// If already running, shorter wait (just need WS connection)
|
|
1023
|
+
const wait = data.already_running ? 500 : 1500;
|
|
1024
|
+
await new Promise(r => setTimeout(r, wait));
|
|
1025
|
+
} catch(e) { if (!cancelled) setStartError(t('server_fail')); return; }
|
|
1026
|
+
if (!cancelled) setReady(true);
|
|
1027
|
+
})();
|
|
1028
|
+
return () => { cancelled = true; };
|
|
1029
|
+
}, [chatId]);
|
|
1030
|
+
|
|
1031
|
+
// Phase 2: init xterm + connect WS (only after ttyd is ready)
|
|
1032
|
+
useEffect(() => {
|
|
1033
|
+
if (!ready || !containerRef.current) return;
|
|
1034
|
+
|
|
1035
|
+
const term = new Terminal({
|
|
1036
|
+
cursorBlink: true,
|
|
1037
|
+
fontSize: 13,
|
|
1038
|
+
fontFamily: "'JetBrains Mono', 'Fira Code', Monaco, monospace",
|
|
1039
|
+
scrollback: 10000,
|
|
1040
|
+
fastScrollModifier: 'shift',
|
|
1041
|
+
rightClickSelectsWord: true,
|
|
1042
|
+
macOptionClickForcesSelection: true,
|
|
1043
|
+
theme: {
|
|
1044
|
+
background: '#1a1b26',
|
|
1045
|
+
foreground: '#c0caf5',
|
|
1046
|
+
cursor: '#c0caf5',
|
|
1047
|
+
selectionBackground: '#33467c',
|
|
1048
|
+
},
|
|
1049
|
+
});
|
|
1050
|
+
// Copy: Cmd+C (Mac), Ctrl+Shift+C (Linux) — handle directly in key handler, don't send to terminal
|
|
1051
|
+
// Paste: Cmd+V (Mac), Ctrl+V / Ctrl+Shift+V (Windows/Linux) — read clipboard and paste
|
|
1052
|
+
term.attachCustomKeyEventHandler((ev) => {
|
|
1053
|
+
if (ev.type !== 'keydown') return true;
|
|
1054
|
+
const isCopy = (ev.metaKey && ev.key === 'c') ||
|
|
1055
|
+
(ev.ctrlKey && ev.shiftKey && ev.key === 'C');
|
|
1056
|
+
const isPaste = (ev.metaKey && ev.key === 'v') ||
|
|
1057
|
+
(ev.ctrlKey && ev.key === 'v');
|
|
1058
|
+
if (ev.metaKey && ev.key === 'a') {
|
|
1059
|
+
term.selectAll();
|
|
1060
|
+
return false;
|
|
1061
|
+
}
|
|
1062
|
+
if (isCopy) {
|
|
1063
|
+
if (term.hasSelection()) navigator.clipboard.writeText(term.getSelection()).catch(() => {});
|
|
1064
|
+
return false;
|
|
1065
|
+
}
|
|
1066
|
+
if (isPaste) {
|
|
1067
|
+
navigator.clipboard.readText().then(text => {
|
|
1068
|
+
if (xtermRef.current) xtermRef.current.paste(text);
|
|
1069
|
+
}).catch(() => {});
|
|
1070
|
+
return false;
|
|
1071
|
+
}
|
|
1072
|
+
return true;
|
|
1073
|
+
});
|
|
1074
|
+
const fitAddon = new FitAddon.FitAddon();
|
|
1075
|
+
term.loadAddon(fitAddon);
|
|
1076
|
+
try { term.loadAddon(new WebLinksAddon.WebLinksAddon()); } catch(e) {}
|
|
1077
|
+
term.open(containerRef.current);
|
|
1078
|
+
fitAddon.fit();
|
|
1079
|
+
xtermRef.current = term;
|
|
1080
|
+
// Mouse select auto-copy: onSelectionChange fires synchronously from mouseup (user gesture)
|
|
1081
|
+
term.onSelectionChange(() => {
|
|
1082
|
+
if (term.hasSelection()) navigator.clipboard.writeText(term.getSelection()).catch(() => {});
|
|
1083
|
+
});
|
|
1084
|
+
fitAddonRef.current = fitAddon;
|
|
1085
|
+
|
|
1086
|
+
term.onData((data) => {
|
|
1087
|
+
if (wsRef.current && wsRef.current.readyState === WebSocket.OPEN) {
|
|
1088
|
+
wsRef.current.send(new TextEncoder().encode('\x30' + data));
|
|
1089
|
+
}
|
|
1090
|
+
});
|
|
1091
|
+
|
|
1092
|
+
const ro = new ResizeObserver(() => {
|
|
1093
|
+
if (fitAddonRef.current && xtermRef.current) {
|
|
1094
|
+
fitAddonRef.current.fit();
|
|
1095
|
+
if (wsRef.current && wsRef.current.readyState === WebSocket.OPEN) {
|
|
1096
|
+
const dims = fitAddonRef.current.proposeDimensions();
|
|
1097
|
+
if (dims) {
|
|
1098
|
+
wsRef.current.send(new TextEncoder().encode('\x31' + JSON.stringify({ columns: dims.cols, rows: dims.rows })));
|
|
1099
|
+
}
|
|
1100
|
+
}
|
|
1101
|
+
}
|
|
1102
|
+
});
|
|
1103
|
+
ro.observe(containerRef.current);
|
|
1104
|
+
|
|
1105
|
+
connectWs(resumeId);
|
|
1106
|
+
|
|
1107
|
+
return () => {
|
|
1108
|
+
ro.disconnect();
|
|
1109
|
+
if (wsRef.current) { wsRef.current.close(); wsRef.current = null; }
|
|
1110
|
+
if (xtermRef.current) { xtermRef.current.dispose(); xtermRef.current = null; }
|
|
1111
|
+
};
|
|
1112
|
+
}, [ready]);
|
|
1113
|
+
|
|
1114
|
+
const handleToggleSelectMode = useCallback(() => {
|
|
1115
|
+
if (!xtermRef.current) return;
|
|
1116
|
+
setSelectMode(prev => {
|
|
1117
|
+
const enabling = !prev;
|
|
1118
|
+
if (enabling) {
|
|
1119
|
+
// Disable mouse event reporting — xterm stops forwarding mouse to terminal, drag selects text
|
|
1120
|
+
xtermRef.current.write('\x1b[?1000l\x1b[?1002l\x1b[?1003l\x1b[?1006l');
|
|
1121
|
+
} else {
|
|
1122
|
+
// Re-enable mouse event reporting
|
|
1123
|
+
xtermRef.current.write('\x1b[?1000h\x1b[?1002h\x1b[?1006h');
|
|
1124
|
+
}
|
|
1125
|
+
return enabling;
|
|
1126
|
+
});
|
|
1127
|
+
}, []);
|
|
1128
|
+
|
|
1129
|
+
const handleClear = useCallback(() => {
|
|
1130
|
+
if (wsRef.current && wsRef.current.readyState === WebSocket.OPEN) {
|
|
1131
|
+
// Send Ctrl+L
|
|
1132
|
+
wsRef.current.send(new TextEncoder().encode('\x30\x0c'));
|
|
1133
|
+
}
|
|
1134
|
+
}, []);
|
|
1135
|
+
|
|
1136
|
+
const handleKill = useCallback(async () => {
|
|
1137
|
+
// Send Ctrl+C
|
|
1138
|
+
if (wsRef.current && wsRef.current.readyState === WebSocket.OPEN) {
|
|
1139
|
+
wsRef.current.send(new TextEncoder().encode('\x30\x03'));
|
|
1140
|
+
}
|
|
1141
|
+
// Kill claude processes
|
|
1142
|
+
try {
|
|
1143
|
+
const resp = await fetch(`/terminal/${chatId}/processes?_t=${Date.now()}`);
|
|
1144
|
+
const data = await resp.json();
|
|
1145
|
+
for (const p of (data.processes || [])) {
|
|
1146
|
+
await fetch(`/terminal/${chatId}/processes/${p.pid}/kill`, { method: 'POST' });
|
|
1147
|
+
}
|
|
1148
|
+
} catch(e) {}
|
|
1149
|
+
// Kill ttyd + tmux so next "Open terminal" starts fresh (with .bashrc → Claude Code autostart)
|
|
1150
|
+
try {
|
|
1151
|
+
await fetch(`/terminal/${chatId}/stop-ttyd`, { method: 'POST' });
|
|
1152
|
+
} catch(e) {}
|
|
1153
|
+
if (wsRef.current) { wsRef.current.close(); wsRef.current = null; }
|
|
1154
|
+
if (xtermRef.current) { xtermRef.current.dispose(); xtermRef.current = null; }
|
|
1155
|
+
onBack();
|
|
1156
|
+
}, [chatId, onBack]);
|
|
1157
|
+
|
|
1158
|
+
if (startError) {
|
|
1159
|
+
const isStopped = startError === '__stopped__';
|
|
1160
|
+
const isMetaExists = startError === '__meta_exists__';
|
|
1161
|
+
const canRecover = isStopped || isMetaExists;
|
|
1162
|
+
const [restarting, setRestarting] = useState(false);
|
|
1163
|
+
|
|
1164
|
+
const handleRestart = async () => {
|
|
1165
|
+
setRestarting(true);
|
|
1166
|
+
try {
|
|
1167
|
+
const endpoint = isMetaExists
|
|
1168
|
+
? `/terminal/${chatId}/resurrect-container`
|
|
1169
|
+
: `/terminal/${chatId}/restart-container`;
|
|
1170
|
+
const resp = await fetch(endpoint, { method: 'POST' });
|
|
1171
|
+
if (resp.ok) {
|
|
1172
|
+
setStartError(null);
|
|
1173
|
+
setReady(false);
|
|
1174
|
+
// Re-trigger Phase 1: start ttyd
|
|
1175
|
+
const ttydResp = await fetch(`/terminal/${chatId}/start-ttyd`, {
|
|
1176
|
+
method: 'POST',
|
|
1177
|
+
headers: { 'Content-Type': 'application/json' },
|
|
1178
|
+
body: JSON.stringify({ dangerous_mode: dangerousModeRef.current }),
|
|
1179
|
+
});
|
|
1180
|
+
if (ttydResp.ok) {
|
|
1181
|
+
const data = await ttydResp.json();
|
|
1182
|
+
await new Promise(r => setTimeout(r, isMetaExists ? 2500 : (data.already_running ? 500 : 1500)));
|
|
1183
|
+
setReady(true);
|
|
1184
|
+
return;
|
|
1185
|
+
}
|
|
1186
|
+
}
|
|
1187
|
+
setStartError(t('restore_fail'));
|
|
1188
|
+
} catch { setStartError(t('restore_fail')); }
|
|
1189
|
+
setRestarting(false);
|
|
1190
|
+
};
|
|
1191
|
+
|
|
1192
|
+
return html`
|
|
1193
|
+
<div class="empty-state">
|
|
1194
|
+
<div class="empty-icon" dangerouslySetInnerHTML=${{ __html: icon('terminal', 48) }}></div>
|
|
1195
|
+
<div class="empty-title">${
|
|
1196
|
+
isStopped ? t('container_stopped')
|
|
1197
|
+
: isMetaExists ? t('container_removed')
|
|
1198
|
+
: startError
|
|
1199
|
+
}</div>
|
|
1200
|
+
<div class="empty-desc">${
|
|
1201
|
+
isStopped ? t('container_stopped_desc')
|
|
1202
|
+
: isMetaExists ? t('container_removed_desc')
|
|
1203
|
+
: t('container_generic_desc')
|
|
1204
|
+
}</div>
|
|
1205
|
+
${canRecover && html`
|
|
1206
|
+
<button class="dash-btn-primary" onClick=${handleRestart} disabled=${restarting}>
|
|
1207
|
+
<span class="icon-inline" dangerouslySetInnerHTML=${{ __html: icon('play') }}></span>
|
|
1208
|
+
${restarting
|
|
1209
|
+
? (isMetaExists ? t('restoring') : t('starting'))
|
|
1210
|
+
: (isMetaExists ? t('restore_container') : t('restart_container'))
|
|
1211
|
+
}
|
|
1212
|
+
</button>
|
|
1213
|
+
`}
|
|
1214
|
+
<button class="btn" onClick=${onBack}>
|
|
1215
|
+
<span class="icon-inline" dangerouslySetInnerHTML=${{ __html: icon('arrowLeft') }}></span> ${t('back')}
|
|
1216
|
+
</button>
|
|
1217
|
+
</div>
|
|
1218
|
+
`;
|
|
1219
|
+
}
|
|
1220
|
+
|
|
1221
|
+
if (!ready) {
|
|
1222
|
+
return html`
|
|
1223
|
+
<div class="empty-state">
|
|
1224
|
+
<div class="spinner"></div>
|
|
1225
|
+
<div class="empty-title">${t('starting_terminal')}</div>
|
|
1226
|
+
</div>
|
|
1227
|
+
`;
|
|
1228
|
+
}
|
|
1229
|
+
|
|
1230
|
+
return html`
|
|
1231
|
+
<div class="terminal-view">
|
|
1232
|
+
<div class="terminal-toolbar">
|
|
1233
|
+
<button class="terminal-btn" onClick=${onBack}>
|
|
1234
|
+
<span class="icon-inline" dangerouslySetInnerHTML=${{ __html: icon('arrowLeft') }}></span> ${t('back')}
|
|
1235
|
+
</button>
|
|
1236
|
+
<div style="flex:1"></div>
|
|
1237
|
+
<button class="terminal-btn ${selectMode ? 'terminal-btn-active' : ''}" onClick=${handleToggleSelectMode} title="${t('select_mode_title')}">
|
|
1238
|
+
${selectMode ? t('select_on') : t('select_off')}
|
|
1239
|
+
</button>
|
|
1240
|
+
<button class="terminal-btn terminal-btn-danger" onClick=${handleKill}>
|
|
1241
|
+
<span class="icon-inline" dangerouslySetInnerHTML=${{ __html: icon('stop') }}></span> ${t('terminate')}
|
|
1242
|
+
</button>
|
|
1243
|
+
</div>
|
|
1244
|
+
<div class="terminal-container" ref=${containerRef}></div>
|
|
1245
|
+
</div>
|
|
1246
|
+
`;
|
|
1247
|
+
}
|
|
1248
|
+
|
|
1249
|
+
function TerminalView({ chatId }) {
|
|
1250
|
+
const [mode, setMode] = useState('dashboard');
|
|
1251
|
+
const [resumeId, setResumeId] = useState(null);
|
|
1252
|
+
const [dangerousMode, setDangerousMode] = useState(() => {
|
|
1253
|
+
try { return localStorage.getItem('claudeDangerousMode') === '1'; } catch(e) { return false; }
|
|
1254
|
+
});
|
|
1255
|
+
const toggleDangerous = useCallback((val) => {
|
|
1256
|
+
setDangerousMode(val);
|
|
1257
|
+
try { localStorage.setItem('claudeDangerousMode', val ? '1' : '0'); } catch(e) {}
|
|
1258
|
+
}, []);
|
|
1259
|
+
|
|
1260
|
+
if (mode === 'terminal') {
|
|
1261
|
+
return html`<${TerminalSession}
|
|
1262
|
+
chatId=${chatId}
|
|
1263
|
+
resumeId=${resumeId}
|
|
1264
|
+
dangerousMode=${dangerousMode}
|
|
1265
|
+
onBack=${() => { setMode('dashboard'); setResumeId(null); }}
|
|
1266
|
+
/>`;
|
|
1267
|
+
}
|
|
1268
|
+
|
|
1269
|
+
return html`<${TerminalDashboard}
|
|
1270
|
+
chatId=${chatId}
|
|
1271
|
+
dangerousMode=${dangerousMode}
|
|
1272
|
+
onToggleDangerous=${toggleDangerous}
|
|
1273
|
+
onStartSession=${() => { setResumeId(null); setMode('terminal'); }}
|
|
1274
|
+
onResumeSession=${(id) => { setResumeId(id); setMode('terminal'); }}
|
|
1275
|
+
/>`;
|
|
1276
|
+
}
|
|
1277
|
+
|
|
1278
|
+
// =============================================================================
|
|
1279
|
+
// App
|
|
1280
|
+
// =============================================================================
|
|
1281
|
+
|
|
1282
|
+
// Phase 9.5 — render a USD cost as either "$X.XXXX" or the literal
|
|
1283
|
+
// "unavailable" so codex/opencode runs (where the CLI does not surface
|
|
1284
|
+
// cost) never display a misleading "$0.0000". Mirrors the server-side
|
|
1285
|
+
// branch in mcp_tools.sub_agent (see PITFALLS.md Pitfall 4).
|
|
1286
|
+
function renderCost(costUsd) {
|
|
1287
|
+
if (costUsd === null || costUsd === undefined) return 'unavailable';
|
|
1288
|
+
const n = Number(costUsd);
|
|
1289
|
+
if (!Number.isFinite(n)) return 'unavailable';
|
|
1290
|
+
return `$${n.toFixed(4)}`;
|
|
1291
|
+
}
|
|
1292
|
+
|
|
1293
|
+
// Phase 9.5 — small badge showing which sub-agent CLI the orchestrator
|
|
1294
|
+
// resolved at boot. Sourced from /api/runtime/cli (additive endpoint
|
|
1295
|
+
// added in app.py). When the endpoint 404s (older orchestrator) or the
|
|
1296
|
+
// fetch fails for any reason, the badge silently disappears — pure
|
|
1297
|
+
// progressive enhancement, no breakage of the existing layout.
|
|
1298
|
+
function ActiveCliBadge() {
|
|
1299
|
+
const [info, setInfo] = useState(null);
|
|
1300
|
+
useEffect(() => {
|
|
1301
|
+
let cancelled = false;
|
|
1302
|
+
fetch('/api/runtime/cli', { cache: 'no-store' })
|
|
1303
|
+
.then(r => r.ok ? r.json() : null)
|
|
1304
|
+
.then(data => { if (!cancelled && data) setInfo(data); })
|
|
1305
|
+
.catch(() => {}); // older orchestrator without the endpoint — stay silent
|
|
1306
|
+
return () => { cancelled = true; };
|
|
1307
|
+
}, []);
|
|
1308
|
+
if (!info || !info.cli) return null;
|
|
1309
|
+
const title = `Active sub-agent CLI: ${info.cli}`
|
|
1310
|
+
+ (info.default_model ? ` · default model: ${info.default_model}` : '')
|
|
1311
|
+
+ (info.supports_cost ? '' : ' · cost reporting: unavailable');
|
|
1312
|
+
return html`
|
|
1313
|
+
<span class="active-cli-badge" title=${title}
|
|
1314
|
+
style="display:inline-flex;align-items:center;gap:4px;padding:2px 8px;
|
|
1315
|
+
margin-right:6px;border:1px solid var(--border, #d4d4d4);
|
|
1316
|
+
border-radius:10px;font-size:11px;line-height:16px;
|
|
1317
|
+
color:var(--muted, #666);background:var(--badge-bg, #f5f5f5);">
|
|
1318
|
+
<span style="font-weight:600;text-transform:lowercase">${info.cli}</span>
|
|
1319
|
+
${!info.supports_cost && html`<span style="opacity:0.7">·</span><span style="opacity:0.7">cost n/a</span>`}
|
|
1320
|
+
</span>
|
|
1321
|
+
`;
|
|
1322
|
+
}
|
|
1323
|
+
|
|
1324
|
+
function App() {
|
|
1325
|
+
const [files, setFiles] = useState([]);
|
|
1326
|
+
const [selectedFile, setSelectedFile] = useState(null);
|
|
1327
|
+
const [currentView, setCurrentView] = useState('files');
|
|
1328
|
+
const [browserActive, setBrowserActive] = useState(false);
|
|
1329
|
+
const [terminalActive, setTerminalActive] = useState(false);
|
|
1330
|
+
const [seenFiles, setSeenFiles] = useState(new Set());
|
|
1331
|
+
const fileModTimesRef = useRef(new Map());
|
|
1332
|
+
const browserViewerRef = useRef(null);
|
|
1333
|
+
const lastSyncTimeRef = useRef(null);
|
|
1334
|
+
const syncDotRef = useRef(null);
|
|
1335
|
+
|
|
1336
|
+
// Fetch files
|
|
1337
|
+
const fetchFiles = useCallback(async () => {
|
|
1338
|
+
const dot = syncDotRef.current;
|
|
1339
|
+
if (dot) { dot.classList.add('syncing'); dot.title = t('checking'); }
|
|
1340
|
+
try {
|
|
1341
|
+
const resp = await fetch(`${API_URL}?_t=${Date.now()}`, { cache: 'no-store' });
|
|
1342
|
+
const data = await resp.json();
|
|
1343
|
+
const newFiles = data.files;
|
|
1344
|
+
|
|
1345
|
+
let autoSelectTarget = null;
|
|
1346
|
+
const modTimes = fileModTimesRef.current;
|
|
1347
|
+
for (const f of newFiles) {
|
|
1348
|
+
const prevMod = modTimes.get(f.path);
|
|
1349
|
+
const isRoot = !f.path.includes('/');
|
|
1350
|
+
if (prevMod === undefined && isRoot) { autoSelectTarget = f; break; }
|
|
1351
|
+
else if (f.modified && f.modified !== prevMod && isRoot) { autoSelectTarget = f; break; }
|
|
1352
|
+
}
|
|
1353
|
+
|
|
1354
|
+
modTimes.clear();
|
|
1355
|
+
for (const f of newFiles) { modTimes.set(f.path, f.modified || null); }
|
|
1356
|
+
|
|
1357
|
+
setFiles(newFiles);
|
|
1358
|
+
|
|
1359
|
+
if (autoSelectTarget) {
|
|
1360
|
+
setSelectedFile(autoSelectTarget);
|
|
1361
|
+
} else if (newFiles.length > 0) {
|
|
1362
|
+
setSelectedFile(prev => {
|
|
1363
|
+
if (!prev) return newFiles.find(f => !f.path.includes('/')) || newFiles[0];
|
|
1364
|
+
const updated = newFiles.find(f => f.path === prev.path);
|
|
1365
|
+
// Return SAME reference if path+modified unchanged — prevents re-render and scroll reset
|
|
1366
|
+
return updated && updated.path === prev.path && updated.modified === prev.modified ? prev : (updated || prev);
|
|
1367
|
+
});
|
|
1368
|
+
}
|
|
1369
|
+
|
|
1370
|
+
setSeenFiles(prev => {
|
|
1371
|
+
const next = new Set(prev);
|
|
1372
|
+
newFiles.forEach(f => next.add(f.path));
|
|
1373
|
+
return next;
|
|
1374
|
+
});
|
|
1375
|
+
|
|
1376
|
+
if (dot) { dot.classList.remove('syncing'); dot.title = t('synced'); }
|
|
1377
|
+
lastSyncTimeRef.current = Date.now();
|
|
1378
|
+
} catch (err) {
|
|
1379
|
+
if (dot) { dot.classList.remove('syncing'); dot.title = t('error'); }
|
|
1380
|
+
console.error('Poll error:', err);
|
|
1381
|
+
}
|
|
1382
|
+
}, []);
|
|
1383
|
+
|
|
1384
|
+
// Check browser status
|
|
1385
|
+
const checkBrowserStatus = useCallback(async () => {
|
|
1386
|
+
try {
|
|
1387
|
+
const resp = await fetch(`/browser/${CHAT_ID}/status?_t=${Date.now()}`, { cache: 'no-store' });
|
|
1388
|
+
const data = await resp.json();
|
|
1389
|
+
setBrowserActive(prev => {
|
|
1390
|
+
if (data.active && !prev) {
|
|
1391
|
+
// Auto-switch to browser when first active
|
|
1392
|
+
setCurrentView('browser');
|
|
1393
|
+
}
|
|
1394
|
+
if (!data.active && prev && browserViewerRef.current?.connected) {
|
|
1395
|
+
return true; // Transient blip, keep active
|
|
1396
|
+
}
|
|
1397
|
+
return data.active;
|
|
1398
|
+
});
|
|
1399
|
+
// Update URL bar
|
|
1400
|
+
if (data.active && data.pages && data.pages.length > 0) {
|
|
1401
|
+
const urlBar = document.getElementById('browserUrlBar');
|
|
1402
|
+
if (urlBar) urlBar.textContent = data.pages[0].url || '';
|
|
1403
|
+
}
|
|
1404
|
+
} catch (e) {}
|
|
1405
|
+
}, []);
|
|
1406
|
+
|
|
1407
|
+
// Check terminal status
|
|
1408
|
+
const checkTerminalStatus = useCallback(async () => {
|
|
1409
|
+
try {
|
|
1410
|
+
const resp = await fetch(`/terminal/${CHAT_ID}/processes?_t=${Date.now()}`, { cache: 'no-store' });
|
|
1411
|
+
const data = await resp.json();
|
|
1412
|
+
setTerminalActive(data.processes && data.processes.length > 0);
|
|
1413
|
+
} catch(e) {}
|
|
1414
|
+
}, []);
|
|
1415
|
+
|
|
1416
|
+
// Polling loop
|
|
1417
|
+
useEffect(() => {
|
|
1418
|
+
const poll = async () => {
|
|
1419
|
+
await fetchFiles();
|
|
1420
|
+
await checkBrowserStatus();
|
|
1421
|
+
await checkTerminalStatus();
|
|
1422
|
+
};
|
|
1423
|
+
poll();
|
|
1424
|
+
let timer = setInterval(poll, 3000);
|
|
1425
|
+
|
|
1426
|
+
const visHandler = () => {
|
|
1427
|
+
clearInterval(timer);
|
|
1428
|
+
if (document.hidden) {
|
|
1429
|
+
timer = setInterval(poll, 15000);
|
|
1430
|
+
} else {
|
|
1431
|
+
poll();
|
|
1432
|
+
timer = setInterval(poll, 3000);
|
|
1433
|
+
}
|
|
1434
|
+
};
|
|
1435
|
+
document.addEventListener('visibilitychange', visHandler);
|
|
1436
|
+
|
|
1437
|
+
return () => {
|
|
1438
|
+
clearInterval(timer);
|
|
1439
|
+
document.removeEventListener('visibilitychange', visHandler);
|
|
1440
|
+
};
|
|
1441
|
+
}, [fetchFiles, checkBrowserStatus, checkTerminalStatus]);
|
|
1442
|
+
|
|
1443
|
+
// Listen for iframe link clicks
|
|
1444
|
+
useEffect(() => {
|
|
1445
|
+
const handler = (event) => {
|
|
1446
|
+
if (!event.data || event.data.type !== 'iframe-link-click') return;
|
|
1447
|
+
handleLinkClick(event.data.href, event.data.resolvedUrl, files, selectedFile, setSelectedFile);
|
|
1448
|
+
};
|
|
1449
|
+
window.addEventListener('message', handler);
|
|
1450
|
+
return () => window.removeEventListener('message', handler);
|
|
1451
|
+
}, [files, selectedFile]);
|
|
1452
|
+
|
|
1453
|
+
const onSelectFile = useCallback((file) => {
|
|
1454
|
+
setSelectedFile(file);
|
|
1455
|
+
}, []);
|
|
1456
|
+
|
|
1457
|
+
return html`
|
|
1458
|
+
<div class="toolbar">
|
|
1459
|
+
<div class="toolbar-left">
|
|
1460
|
+
<${ViewTabs}
|
|
1461
|
+
currentView=${currentView}
|
|
1462
|
+
onSwitch=${setCurrentView}
|
|
1463
|
+
browserActive=${browserActive}
|
|
1464
|
+
terminalActive=${terminalActive}
|
|
1465
|
+
/>
|
|
1466
|
+
${currentView === 'files' && files.length > 0 && html`
|
|
1467
|
+
<${FileSelector}
|
|
1468
|
+
files=${files}
|
|
1469
|
+
selectedFile=${selectedFile}
|
|
1470
|
+
seenFiles=${seenFiles}
|
|
1471
|
+
onSelect=${onSelectFile}
|
|
1472
|
+
/>
|
|
1473
|
+
`}
|
|
1474
|
+
</div>
|
|
1475
|
+
<div class="toolbar-right">
|
|
1476
|
+
<${ActiveCliBadge} />
|
|
1477
|
+
<a class="btn btn-icon" href=${location.href} target="_blank" rel="noopener" title="${t('open_new_tab')}">
|
|
1478
|
+
<span class="icon-inline" dangerouslySetInnerHTML=${{ __html: icon('externalLink') }}></span>
|
|
1479
|
+
</a>
|
|
1480
|
+
<div class="status">
|
|
1481
|
+
<span class="dot" ref=${syncDotRef} title="${t('waiting_files')}"></span>
|
|
1482
|
+
</div>
|
|
1483
|
+
${currentView === 'files' && selectedFile && html`
|
|
1484
|
+
<a class="btn btn-icon" href="${selectedFile.url}?download=1" download title="${t('download_file')}">
|
|
1485
|
+
<span class="icon-inline" dangerouslySetInnerHTML=${{ __html: icon('download') }}></span>
|
|
1486
|
+
</a>
|
|
1487
|
+
`}
|
|
1488
|
+
${currentView === 'files' && files.length > 0 && html`
|
|
1489
|
+
<a class="btn btn-icon" href="${FILES_BASE}/archive" download title="${t('download_all')}">
|
|
1490
|
+
<span class="icon-inline" dangerouslySetInnerHTML=${{ __html: icon('archive') }}></span>
|
|
1491
|
+
</a>
|
|
1492
|
+
`}
|
|
1493
|
+
</div>
|
|
1494
|
+
</div>
|
|
1495
|
+
|
|
1496
|
+
<div style="display:${currentView === 'files' ? 'flex' : 'none'};flex:1;flex-direction:column;overflow:hidden">
|
|
1497
|
+
<${FilesView}
|
|
1498
|
+
files=${files}
|
|
1499
|
+
selectedFile=${selectedFile}
|
|
1500
|
+
onSelectFile=${onSelectFile}
|
|
1501
|
+
/>
|
|
1502
|
+
</div>
|
|
1503
|
+
|
|
1504
|
+
<div style="display:${currentView === 'browser' ? 'flex' : 'none'};flex:1;flex-direction:column;overflow:hidden">
|
|
1505
|
+
<${BrowserView}
|
|
1506
|
+
chatId=${CHAT_ID}
|
|
1507
|
+
browserActive=${browserActive}
|
|
1508
|
+
onBrowserViewerRef=${(v) => { browserViewerRef.current = v; }}
|
|
1509
|
+
/>
|
|
1510
|
+
</div>
|
|
1511
|
+
|
|
1512
|
+
<div class="terminal-panel" style="display:${currentView === 'terminal' ? 'flex' : 'none'}">
|
|
1513
|
+
<${TerminalView} chatId=${CHAT_ID} />
|
|
1514
|
+
</div>
|
|
1515
|
+
`;
|
|
1516
|
+
}
|
|
1517
|
+
|
|
1518
|
+
// =============================================================================
|
|
1519
|
+
// Mount
|
|
1520
|
+
// =============================================================================
|
|
1521
|
+
|
|
1522
|
+
render(html`<${App} />`, document.getElementById('app'));
|