@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.
Files changed (769) hide show
  1. package/.coderabbit.yaml +25 -0
  2. package/.dockerignore +95 -0
  3. package/.env.example +137 -0
  4. package/.githooks/pre-commit +68 -0
  5. package/.github/CODEOWNERS +125 -0
  6. package/.github/ISSUE_TEMPLATE/adr-proposal.md +41 -0
  7. package/.github/ISSUE_TEMPLATE/bug-report.md +49 -0
  8. package/.github/ISSUE_TEMPLATE/component-proposal.md +38 -0
  9. package/.github/ISSUE_TEMPLATE/config.yml +15 -0
  10. package/.github/ISSUE_TEMPLATE/dependency-proposal.md +59 -0
  11. package/.github/ISSUE_TEMPLATE/feature_request.md +15 -0
  12. package/.github/ISSUE_TEMPLATE/nfr-proposal.md +44 -0
  13. package/.github/PULL_REQUEST_TEMPLATE.md +15 -0
  14. package/.github/codeql/codeql-config.yml +11 -0
  15. package/.github/codeql/extensions/security-models/python-sanitizers.model.yml +17 -0
  16. package/.github/codeql/extensions/security-models/qlpack.yml +7 -0
  17. package/.github/dependabot.yml +23 -0
  18. package/.github/security-exceptions.yml +23 -0
  19. package/.github/workflows/build.yml +420 -0
  20. package/.github/workflows/codeql.yml +33 -0
  21. package/.github/workflows/contracts-lint.yml +90 -0
  22. package/.github/workflows/docs-lint.yml +151 -0
  23. package/.github/workflows/helm.yml +131 -0
  24. package/.github/workflows/identity-lint.yml +30 -0
  25. package/.github/workflows/release-chart.yml +177 -0
  26. package/.github/workflows/release.yml +95 -0
  27. package/.github/workflows/security.yml +332 -0
  28. package/.github/workflows/stale.yml +31 -0
  29. package/.github/workflows/supply-chain.yml +242 -0
  30. package/.gitleaks.toml +53 -0
  31. package/.markdownlint.yaml +51 -0
  32. package/.semgrepignore +85 -0
  33. package/.vale/styles/Architecture/ap13-data-class-substrate.yml +12 -0
  34. package/.vale/styles/Architecture/banned-phrases.yml +23 -0
  35. package/.vale/styles/Architecture/banned-vocab.yml +23 -0
  36. package/.vale/styles/Architecture/marketing-tone.yml +19 -0
  37. package/.vale.ini +18 -0
  38. package/CHANGELOG.md +411 -0
  39. package/CLAUDE.md +218 -0
  40. package/CONTRIBUTING.md +82 -0
  41. package/Dockerfile +676 -0
  42. package/LICENSE +98 -0
  43. package/LICENSE-APACHE +202 -0
  44. package/LICENSE-MIT +21 -0
  45. package/NOTICE +36 -0
  46. package/README.md +516 -0
  47. package/SECURITY.md +45 -0
  48. package/THIRD-PARTY-LICENSES.md +14 -0
  49. package/apt-packages.txt +108 -0
  50. package/computer-use-server/.dockerignore +13 -0
  51. package/computer-use-server/Dockerfile +44 -0
  52. package/computer-use-server/README.md +84 -0
  53. package/computer-use-server/app.py +1544 -0
  54. package/computer-use-server/bin/list-subagent-models +449 -0
  55. package/computer-use-server/cli-defaults/README.md +31 -0
  56. package/computer-use-server/cli-defaults/codex.json +7 -0
  57. package/computer-use-server/cli-defaults/opencode.json +18 -0
  58. package/computer-use-server/cli_adapters/__init__.py +46 -0
  59. package/computer-use-server/cli_adapters/claude.py +163 -0
  60. package/computer-use-server/cli_adapters/codex.py +163 -0
  61. package/computer-use-server/cli_adapters/opencode.py +169 -0
  62. package/computer-use-server/cli_adapters/result.py +34 -0
  63. package/computer-use-server/cli_runtime.py +316 -0
  64. package/computer-use-server/context_vars.py +24 -0
  65. package/computer-use-server/docker_manager.py +1100 -0
  66. package/computer-use-server/docs_html.py +12 -0
  67. package/computer-use-server/mcp_resources.py +170 -0
  68. package/computer-use-server/mcp_tools.py +1430 -0
  69. package/computer-use-server/requirements.txt +17 -0
  70. package/computer-use-server/security.py +50 -0
  71. package/computer-use-server/skill_manager.py +664 -0
  72. package/computer-use-server/static/browser-viewer.js +445 -0
  73. package/computer-use-server/static/chart.umd.js +14 -0
  74. package/computer-use-server/static/docs.html +203 -0
  75. package/computer-use-server/static/github-dark.min.css +10 -0
  76. package/computer-use-server/static/github.min.css +10 -0
  77. package/computer-use-server/static/highlight.min.js +1213 -0
  78. package/computer-use-server/static/highlightjs-line-numbers.min.js +1 -0
  79. package/computer-use-server/static/icons.js +74 -0
  80. package/computer-use-server/static/jszip.min.js +13 -0
  81. package/computer-use-server/static/katex/auto-render.min.js +1 -0
  82. package/computer-use-server/static/katex/fonts/KaTeX_AMS-Regular.ttf +0 -0
  83. package/computer-use-server/static/katex/fonts/KaTeX_AMS-Regular.woff +0 -0
  84. package/computer-use-server/static/katex/fonts/KaTeX_AMS-Regular.woff2 +0 -0
  85. package/computer-use-server/static/katex/fonts/KaTeX_Caligraphic-Bold.ttf +0 -0
  86. package/computer-use-server/static/katex/fonts/KaTeX_Caligraphic-Bold.woff +0 -0
  87. package/computer-use-server/static/katex/fonts/KaTeX_Caligraphic-Bold.woff2 +0 -0
  88. package/computer-use-server/static/katex/fonts/KaTeX_Caligraphic-Regular.ttf +0 -0
  89. package/computer-use-server/static/katex/fonts/KaTeX_Caligraphic-Regular.woff +0 -0
  90. package/computer-use-server/static/katex/fonts/KaTeX_Caligraphic-Regular.woff2 +0 -0
  91. package/computer-use-server/static/katex/fonts/KaTeX_Fraktur-Bold.ttf +0 -0
  92. package/computer-use-server/static/katex/fonts/KaTeX_Fraktur-Bold.woff +0 -0
  93. package/computer-use-server/static/katex/fonts/KaTeX_Fraktur-Bold.woff2 +0 -0
  94. package/computer-use-server/static/katex/fonts/KaTeX_Fraktur-Regular.ttf +0 -0
  95. package/computer-use-server/static/katex/fonts/KaTeX_Fraktur-Regular.woff +0 -0
  96. package/computer-use-server/static/katex/fonts/KaTeX_Fraktur-Regular.woff2 +0 -0
  97. package/computer-use-server/static/katex/fonts/KaTeX_Main-Bold.ttf +0 -0
  98. package/computer-use-server/static/katex/fonts/KaTeX_Main-Bold.woff +0 -0
  99. package/computer-use-server/static/katex/fonts/KaTeX_Main-Bold.woff2 +0 -0
  100. package/computer-use-server/static/katex/fonts/KaTeX_Main-BoldItalic.ttf +0 -0
  101. package/computer-use-server/static/katex/fonts/KaTeX_Main-BoldItalic.woff +0 -0
  102. package/computer-use-server/static/katex/fonts/KaTeX_Main-BoldItalic.woff2 +0 -0
  103. package/computer-use-server/static/katex/fonts/KaTeX_Main-Italic.ttf +0 -0
  104. package/computer-use-server/static/katex/fonts/KaTeX_Main-Italic.woff +0 -0
  105. package/computer-use-server/static/katex/fonts/KaTeX_Main-Italic.woff2 +0 -0
  106. package/computer-use-server/static/katex/fonts/KaTeX_Main-Regular.ttf +0 -0
  107. package/computer-use-server/static/katex/fonts/KaTeX_Main-Regular.woff +0 -0
  108. package/computer-use-server/static/katex/fonts/KaTeX_Main-Regular.woff2 +0 -0
  109. package/computer-use-server/static/katex/fonts/KaTeX_Math-BoldItalic.ttf +0 -0
  110. package/computer-use-server/static/katex/fonts/KaTeX_Math-BoldItalic.woff +0 -0
  111. package/computer-use-server/static/katex/fonts/KaTeX_Math-BoldItalic.woff2 +0 -0
  112. package/computer-use-server/static/katex/fonts/KaTeX_Math-Italic.ttf +0 -0
  113. package/computer-use-server/static/katex/fonts/KaTeX_Math-Italic.woff +0 -0
  114. package/computer-use-server/static/katex/fonts/KaTeX_Math-Italic.woff2 +0 -0
  115. package/computer-use-server/static/katex/fonts/KaTeX_SansSerif-Bold.ttf +0 -0
  116. package/computer-use-server/static/katex/fonts/KaTeX_SansSerif-Bold.woff +0 -0
  117. package/computer-use-server/static/katex/fonts/KaTeX_SansSerif-Bold.woff2 +0 -0
  118. package/computer-use-server/static/katex/fonts/KaTeX_SansSerif-Italic.ttf +0 -0
  119. package/computer-use-server/static/katex/fonts/KaTeX_SansSerif-Italic.woff +0 -0
  120. package/computer-use-server/static/katex/fonts/KaTeX_SansSerif-Italic.woff2 +0 -0
  121. package/computer-use-server/static/katex/fonts/KaTeX_SansSerif-Regular.ttf +0 -0
  122. package/computer-use-server/static/katex/fonts/KaTeX_SansSerif-Regular.woff +0 -0
  123. package/computer-use-server/static/katex/fonts/KaTeX_SansSerif-Regular.woff2 +0 -0
  124. package/computer-use-server/static/katex/fonts/KaTeX_Script-Regular.ttf +0 -0
  125. package/computer-use-server/static/katex/fonts/KaTeX_Script-Regular.woff +0 -0
  126. package/computer-use-server/static/katex/fonts/KaTeX_Script-Regular.woff2 +0 -0
  127. package/computer-use-server/static/katex/fonts/KaTeX_Size1-Regular.ttf +0 -0
  128. package/computer-use-server/static/katex/fonts/KaTeX_Size1-Regular.woff +0 -0
  129. package/computer-use-server/static/katex/fonts/KaTeX_Size1-Regular.woff2 +0 -0
  130. package/computer-use-server/static/katex/fonts/KaTeX_Size2-Regular.ttf +0 -0
  131. package/computer-use-server/static/katex/fonts/KaTeX_Size2-Regular.woff +0 -0
  132. package/computer-use-server/static/katex/fonts/KaTeX_Size2-Regular.woff2 +0 -0
  133. package/computer-use-server/static/katex/fonts/KaTeX_Size3-Regular.ttf +0 -0
  134. package/computer-use-server/static/katex/fonts/KaTeX_Size3-Regular.woff +0 -0
  135. package/computer-use-server/static/katex/fonts/KaTeX_Size3-Regular.woff2 +0 -0
  136. package/computer-use-server/static/katex/fonts/KaTeX_Size4-Regular.ttf +0 -0
  137. package/computer-use-server/static/katex/fonts/KaTeX_Size4-Regular.woff +0 -0
  138. package/computer-use-server/static/katex/fonts/KaTeX_Size4-Regular.woff2 +0 -0
  139. package/computer-use-server/static/katex/fonts/KaTeX_Typewriter-Regular.ttf +0 -0
  140. package/computer-use-server/static/katex/fonts/KaTeX_Typewriter-Regular.woff +0 -0
  141. package/computer-use-server/static/katex/fonts/KaTeX_Typewriter-Regular.woff2 +0 -0
  142. package/computer-use-server/static/katex/katex.min.css +1 -0
  143. package/computer-use-server/static/katex/katex.min.js +1 -0
  144. package/computer-use-server/static/locale.js +242 -0
  145. package/computer-use-server/static/mammoth.browser.min.js +21 -0
  146. package/computer-use-server/static/marked.min.js +6 -0
  147. package/computer-use-server/static/mermaid.min.js +2811 -0
  148. package/computer-use-server/static/pdf.min.js +22 -0
  149. package/computer-use-server/static/pdf.worker.min.js +22 -0
  150. package/computer-use-server/static/pptxviewjs.min.js +1 -0
  151. package/computer-use-server/static/preact-htm.min.js +1 -0
  152. package/computer-use-server/static/preview.css +1030 -0
  153. package/computer-use-server/static/preview.js +1522 -0
  154. package/computer-use-server/static/xlsx.full.min.js +22 -0
  155. package/computer-use-server/static/xterm-addon-fit.min.js +2 -0
  156. package/computer-use-server/static/xterm-addon-web-links.min.js +2 -0
  157. package/computer-use-server/static/xterm.css +218 -0
  158. package/computer-use-server/static/xterm.min.js +2 -0
  159. package/computer-use-server/system_prompt.py +761 -0
  160. package/computer-use-server/uploads.py +82 -0
  161. package/contracts/README.md +53 -0
  162. package/contracts/audit/audit-fanin.asyncapi.yaml +407 -0
  163. package/contracts/exec/exec-channel.schema.json +240 -0
  164. package/contracts/mcp/2025-06-18/ocu-constraints.schema.json +178 -0
  165. package/contracts/storage/file-artifact-api.schema.json +390 -0
  166. package/contracts/storage/file-ops.schema.json +217 -0
  167. package/contracts/storage/mount-config.schema.json +197 -0
  168. package/cron/Dockerfile +15 -0
  169. package/cron/cleanup-quick.sh +21 -0
  170. package/cron/cleanup.sh +127 -0
  171. package/data/outputs/.gitkeep +0 -0
  172. package/data/uploads/.gitkeep +0 -0
  173. package/docker-compose.test.yml +54 -0
  174. package/docker-compose.webui.yml +77 -0
  175. package/docker-compose.yml +96 -0
  176. package/docs/CLOUD.md +29 -0
  177. package/docs/COMPARISON.md +128 -0
  178. package/docs/DOCKER.md +469 -0
  179. package/docs/DYNAMIC-SKILLS.md +77 -0
  180. package/docs/FEATURES.md +100 -0
  181. package/docs/INSTALL.md +111 -0
  182. package/docs/KNOWN-BUGS.md +86 -0
  183. package/docs/MCP.md +320 -0
  184. package/docs/SCREENSHOTS.md +39 -0
  185. package/docs/SKILLS-USER-GUIDE.md +86 -0
  186. package/docs/SKILLS.md +483 -0
  187. package/docs/TERMINAL-TAB.md +56 -0
  188. package/docs/architecture/02-trust-boundaries.md +224 -0
  189. package/docs/architecture/03-c4-context.md +61 -0
  190. package/docs/architecture/04-bounded-contexts.md +119 -0
  191. package/docs/architecture/05-c4-container.md +88 -0
  192. package/docs/architecture/06-threat-model.md +172 -0
  193. package/docs/architecture/08-contracts.md +105 -0
  194. package/docs/architecture/MANIFESTO.md +38 -0
  195. package/docs/architecture/PROCESS.md +64 -0
  196. package/docs/architecture/README.md +37 -0
  197. package/docs/architecture/adr/0000-template.md +65 -0
  198. package/docs/architecture/adr/0001-layer-0-gate-legacy-exclusion.md +75 -0
  199. package/docs/architecture/adr/0002-session-view-descriptor.md +57 -0
  200. package/docs/architecture/adr/0003-sandbox-runtime-tier-ladder.md +63 -0
  201. package/docs/architecture/adr/0004-operator-authentication-substrate.md +63 -0
  202. package/docs/architecture/adr/0005-egress-credential-delivery-envoy-sds.md +62 -0
  203. package/docs/architecture/adr/0006-egress-forward-proxy-substrate.md +65 -0
  204. package/docs/architecture/adr/0007-egress-auth-mechanism.md +72 -0
  205. package/docs/architecture/adr/0008-session-egress-attribution.md +59 -0
  206. package/docs/architecture/adr/0009-audit-pipeline-pluggable-by-contract.md +76 -0
  207. package/docs/architecture/adr/0010-storage-backend-pluggable-adapter.md +60 -0
  208. package/docs/architecture/adr/0011-storage-egress-lane.md +67 -0
  209. package/docs/architecture/adr/0012-implementation-language.md +67 -0
  210. package/docs/architecture/adr/0020-sandbox-image-provisioning.md +82 -0
  211. package/docs/architecture/adr/README.md +53 -0
  212. package/docs/architecture/compliance/.gitkeep +0 -0
  213. package/docs/architecture/components/00-overview.md +42 -0
  214. package/docs/architecture/components/0000-template.md +50 -0
  215. package/docs/architecture/components/01-mcp-gateway.md +80 -0
  216. package/docs/architecture/components/02-control-operator-api.md +80 -0
  217. package/docs/architecture/components/04-storage-broker.md +104 -0
  218. package/docs/architecture/components/05-session-sandbox.md +93 -0
  219. package/docs/architecture/components/06-egress-trust-edge.md +95 -0
  220. package/docs/architecture/components/07-audit-pipeline.md +110 -0
  221. package/docs/architecture/diagrams/.gitkeep +0 -0
  222. package/docs/architecture/diagrams/02-trust-boundaries.mmd +111 -0
  223. package/docs/architecture/diagrams/06-threat-model.mmd +41 -0
  224. package/docs/architecture/diagrams/08-contracts.mmd +47 -0
  225. package/docs/architecture/diagrams/c4-container.mmd +59 -0
  226. package/docs/architecture/diagrams/c4-context.mmd +46 -0
  227. package/docs/architecture/glossary.md +172 -0
  228. package/docs/architecture/manifesto/.gitkeep +0 -0
  229. package/docs/architecture/manifesto/01-audience-and-buyer.md +57 -0
  230. package/docs/architecture/manifesto/02-nfrs.md +325 -0
  231. package/docs/architecture/manifesto/03-non-negotiables.md +35 -0
  232. package/docs/architecture/manifesto/04-non-goals.md +23 -0
  233. package/docs/architecture/manifesto/05-licensing-posture.md +61 -0
  234. package/docs/architecture/manifesto/06-starter-mode-policy.md +49 -0
  235. package/docs/architecture/manifesto/07-governance.md +60 -0
  236. package/docs/architecture/primitives-backlog.md +51 -0
  237. package/docs/architecture.svg +117 -0
  238. package/docs/claude-code-gateway.md +173 -0
  239. package/docs/cli-config-templates.md +240 -0
  240. package/docs/data-flow.svg +72 -0
  241. package/docs/demo-landing-page.gif +0 -0
  242. package/docs/demo-qwen-trending.gif +0 -0
  243. package/docs/dynamic-skills.svg +77 -0
  244. package/docs/file-flow.svg +126 -0
  245. package/docs/future-architecture/README.md +152 -0
  246. package/docs/future-architecture/adr/0001-control-plane-language-go.md +80 -0
  247. package/docs/future-architecture/adr/0002-guest-agent-language-go.md +84 -0
  248. package/docs/future-architecture/adr/0003-docker-poc-first-then-k8s.md +37 -0
  249. package/docs/future-architecture/adr/0004-pluggable-runtime-via-runtimeclass.md +34 -0
  250. package/docs/future-architecture/adr/0005-mcp-as-control-plane-gateway.md +34 -0
  251. package/docs/future-architecture/adr/0006-no-agpl-no-bsl-dependencies.md +41 -0
  252. package/docs/future-architecture/adr/0007-superseded-by-future-architecture.md +37 -0
  253. package/docs/future-architecture/adr/0008-internal-grpc-external-rest-mcp.md +106 -0
  254. package/docs/future-architecture/adr/0009-external-protocol-dialects.md +94 -0
  255. package/docs/future-architecture/adr/0010-lambda-as-inspiration-not-runtime.md +86 -0
  256. package/docs/future-architecture/adr/0011-kata-as-first-class-dind-runtime.md +84 -0
  257. package/docs/future-architecture/antipatterns.md +552 -0
  258. package/docs/future-architecture/architecture/01-layers.md +109 -0
  259. package/docs/future-architecture/architecture/02-layer4-control-plane.md +122 -0
  260. package/docs/future-architecture/architecture/03-layer3-providers.md +174 -0
  261. package/docs/future-architecture/architecture/04-layer2-runtimes.md +114 -0
  262. package/docs/future-architecture/architecture/04b-credential-broker.md +153 -0
  263. package/docs/future-architecture/architecture/05-layer1-guest-agent.md +138 -0
  264. package/docs/future-architecture/architecture/06-storage.md +134 -0
  265. package/docs/future-architecture/architecture/07-security.md +194 -0
  266. package/docs/future-architecture/architecture/08-networking.md +149 -0
  267. package/docs/future-architecture/architecture/09-templates.md +122 -0
  268. package/docs/future-architecture/architecture/10-observability.md +121 -0
  269. package/docs/future-architecture/design-notes.md +72 -0
  270. package/docs/future-architecture/gaps.md +281 -0
  271. package/docs/future-architecture/phase-template.md +123 -0
  272. package/docs/future-architecture/references.md +225 -0
  273. package/docs/future-architecture/research/01-kata-containers.md +100 -0
  274. package/docs/future-architecture/research/02-e2b-infra.md +133 -0
  275. package/docs/future-architecture/research/03-coder.md +115 -0
  276. package/docs/future-architecture/research/04-cloud-hypervisor.md +99 -0
  277. package/docs/future-architecture/research/05-firecracker.md +114 -0
  278. package/docs/future-architecture/research/06-agent-sandbox.md +142 -0
  279. package/docs/future-architecture/research/07-chromedp.md +78 -0
  280. package/docs/future-architecture/research/08-microsandbox.md +78 -0
  281. package/docs/future-architecture/research/09-agentbox.md +135 -0
  282. package/docs/future-architecture/research/10-sysbox.md +100 -0
  283. package/docs/future-architecture/research/11-firecracker-containerd.md +93 -0
  284. package/docs/future-architecture/research/12-docker-socket-proxy.md +59 -0
  285. package/docs/future-architecture/research/14-e2b-desktop-and-surf.md +107 -0
  286. package/docs/future-architecture/research/18-open-webui-terminals-observed.md +135 -0
  287. package/docs/future-architecture/research/bank-buyer.md +96 -0
  288. package/docs/future-architecture/research/enthusiast-audience.md +106 -0
  289. package/docs/future-architecture/research/proof-uipath-anthropic-2026-05.md +76 -0
  290. package/docs/future-architecture/research/widemoat-thesis-advisor.md +124 -0
  291. package/docs/future-architecture/roadmap.md +438 -0
  292. package/docs/kata-runtime.md +267 -0
  293. package/docs/kubernetes.md +86 -0
  294. package/docs/logo.png +0 -0
  295. package/docs/multi-cli.md +161 -0
  296. package/docs/openwebui-filter.md +134 -0
  297. package/docs/roadmap/implementation-roadmap.md +104 -0
  298. package/docs/sandbox-contents.svg +229 -0
  299. package/docs/screenshots/01-create-document.png +0 -0
  300. package/docs/screenshots/02-file-preview.png +0 -0
  301. package/docs/screenshots/03-browser-viewer.png +0 -0
  302. package/docs/screenshots/04-sub-agent-terminal.png +0 -0
  303. package/docs/screenshots/05-chat-overview.png +0 -0
  304. package/docs/screenshots/06-sub-agent-dashboard.png +0 -0
  305. package/docs/screenshots/07-frontend-design-skill.png +0 -0
  306. package/docs/screenshots/08-pptx-skill.png +0 -0
  307. package/docs/screenshots/09-skill-creator.png +0 -0
  308. package/docs/screenshots/10-data-chart.png +0 -0
  309. package/docs/shared-browser.svg +102 -0
  310. package/docs/system-prompt.md +113 -0
  311. package/docs/terminal-flow.svg +69 -0
  312. package/examples/helm/README.md +20 -0
  313. package/examples/helm/standalone/values.yaml +49 -0
  314. package/examples/helm/with-open-webui/README.md +99 -0
  315. package/examples/helm/with-open-webui/values-computer-use.yaml +32 -0
  316. package/examples/helm/with-open-webui/values-open-webui.yaml +67 -0
  317. package/fonts/NotoEmoji-Regular.ttf +0 -0
  318. package/helm/computer-use-server/.helmignore +17 -0
  319. package/helm/computer-use-server/Chart.yaml +32 -0
  320. package/helm/computer-use-server/README.md +211 -0
  321. package/helm/computer-use-server/templates/NOTES.txt +66 -0
  322. package/helm/computer-use-server/templates/_helpers.tpl +115 -0
  323. package/helm/computer-use-server/templates/configmap-dind-init.yaml +82 -0
  324. package/helm/computer-use-server/templates/configmap.yaml +18 -0
  325. package/helm/computer-use-server/templates/deployment.yaml +248 -0
  326. package/helm/computer-use-server/templates/ingress.yaml +38 -0
  327. package/helm/computer-use-server/templates/networkpolicy.yaml +50 -0
  328. package/helm/computer-use-server/templates/pdb.yaml +16 -0
  329. package/helm/computer-use-server/templates/pvc-data.yaml +20 -0
  330. package/helm/computer-use-server/templates/pvc-skills-cache.yaml +20 -0
  331. package/helm/computer-use-server/templates/pvc-user-data.yaml +20 -0
  332. package/helm/computer-use-server/templates/pvc-var-lib-docker.yaml +27 -0
  333. package/helm/computer-use-server/templates/secret.yaml +23 -0
  334. package/helm/computer-use-server/templates/service.yaml +22 -0
  335. package/helm/computer-use-server/templates/serviceaccount.yaml +15 -0
  336. package/helm/computer-use-server/templates/tests/test-health.yaml +23 -0
  337. package/helm/computer-use-server/values.schema.json +183 -0
  338. package/helm/computer-use-server/values.yaml +297 -0
  339. package/lychee.toml +36 -0
  340. package/openwebui/Dockerfile +52 -0
  341. package/openwebui/README.md +38 -0
  342. package/openwebui/functions/README.md +48 -0
  343. package/openwebui/functions/computer_link_filter.py +487 -0
  344. package/openwebui/init.sh +305 -0
  345. package/openwebui/patches/README.md +44 -0
  346. package/openwebui/patches/fix_artifacts_auto_show.py +441 -0
  347. package/openwebui/patches/fix_attached_files_position.py +87 -0
  348. package/openwebui/patches/fix_large_tool_args.py +156 -0
  349. package/openwebui/patches/fix_large_tool_results.py +289 -0
  350. package/openwebui/patches/fix_preview_url_detection.py +230 -0
  351. package/openwebui/patches/fix_skip_embedding_chat_files.py +229 -0
  352. package/openwebui/patches/fix_skip_rag_files_native_fc.py +100 -0
  353. package/openwebui/patches/fix_tool_loop_errors.py +510 -0
  354. package/package.json +39 -0
  355. package/requirements.txt +112 -0
  356. package/scripts/check-config.sh +141 -0
  357. package/scripts/docs-lint/ai-slop-detector.sh +202 -0
  358. package/scripts/docs-lint/architecture-tree-whitelist.sh +131 -0
  359. package/scripts/docs-lint/ascii-diagram-detector.sh +58 -0
  360. package/scripts/docs-lint/front-matter-validator.sh +97 -0
  361. package/scripts/docs-lint/gitignored-ref-detector.sh +122 -0
  362. package/scripts/docs-lint/identity-email-detector.sh +48 -0
  363. package/scripts/docs-lint/test-linters.sh +354 -0
  364. package/scripts/docs-lint/wc-budget.sh +61 -0
  365. package/scripts/githooks/pre-push +75 -0
  366. package/server.json +13 -0
  367. package/settings-wrapper/Dockerfile +9 -0
  368. package/settings-wrapper/README.md +119 -0
  369. package/settings-wrapper/app.py +113 -0
  370. package/settings-wrapper/requirements.txt +2 -0
  371. package/settings-wrapper/skills.json +25 -0
  372. package/skills/README.md +46 -0
  373. package/skills/examples/algorithmic-art/SKILL.md +405 -0
  374. package/skills/examples/algorithmic-art/templates/generator_template.js +223 -0
  375. package/skills/examples/algorithmic-art/templates/viewer.html +601 -0
  376. package/skills/examples/artifacts-builder/SKILL.md +74 -0
  377. package/skills/examples/artifacts-builder/scripts/bundle-artifact.sh +54 -0
  378. package/skills/examples/artifacts-builder/scripts/init-artifact.sh +322 -0
  379. package/skills/examples/artifacts-builder/scripts/shadcn-components.tar.gz +0 -0
  380. package/skills/examples/canvas-design/LICENSE.txt +202 -0
  381. package/skills/examples/canvas-design/SKILL.md +130 -0
  382. package/skills/examples/canvas-design/canvas-fonts/ArsenalSC-OFL.txt +93 -0
  383. package/skills/examples/canvas-design/canvas-fonts/ArsenalSC-Regular.ttf +0 -0
  384. package/skills/examples/canvas-design/canvas-fonts/BigShoulders-Bold.ttf +0 -0
  385. package/skills/examples/canvas-design/canvas-fonts/BigShoulders-OFL.txt +93 -0
  386. package/skills/examples/canvas-design/canvas-fonts/BigShoulders-Regular.ttf +0 -0
  387. package/skills/examples/canvas-design/canvas-fonts/Boldonse-OFL.txt +93 -0
  388. package/skills/examples/canvas-design/canvas-fonts/Boldonse-Regular.ttf +0 -0
  389. package/skills/examples/canvas-design/canvas-fonts/BricolageGrotesque-Bold.ttf +0 -0
  390. package/skills/examples/canvas-design/canvas-fonts/BricolageGrotesque-OFL.txt +93 -0
  391. package/skills/examples/canvas-design/canvas-fonts/BricolageGrotesque-Regular.ttf +0 -0
  392. package/skills/examples/canvas-design/canvas-fonts/CrimsonPro-Bold.ttf +0 -0
  393. package/skills/examples/canvas-design/canvas-fonts/CrimsonPro-Italic.ttf +0 -0
  394. package/skills/examples/canvas-design/canvas-fonts/CrimsonPro-OFL.txt +93 -0
  395. package/skills/examples/canvas-design/canvas-fonts/CrimsonPro-Regular.ttf +0 -0
  396. package/skills/examples/canvas-design/canvas-fonts/DMMono-OFL.txt +93 -0
  397. package/skills/examples/canvas-design/canvas-fonts/DMMono-Regular.ttf +0 -0
  398. package/skills/examples/canvas-design/canvas-fonts/EricaOne-OFL.txt +94 -0
  399. package/skills/examples/canvas-design/canvas-fonts/EricaOne-Regular.ttf +0 -0
  400. package/skills/examples/canvas-design/canvas-fonts/GeistMono-Bold.ttf +0 -0
  401. package/skills/examples/canvas-design/canvas-fonts/GeistMono-OFL.txt +93 -0
  402. package/skills/examples/canvas-design/canvas-fonts/GeistMono-Regular.ttf +0 -0
  403. package/skills/examples/canvas-design/canvas-fonts/Gloock-OFL.txt +93 -0
  404. package/skills/examples/canvas-design/canvas-fonts/Gloock-Regular.ttf +0 -0
  405. package/skills/examples/canvas-design/canvas-fonts/IBMPlexMono-Bold.ttf +0 -0
  406. package/skills/examples/canvas-design/canvas-fonts/IBMPlexMono-OFL.txt +93 -0
  407. package/skills/examples/canvas-design/canvas-fonts/IBMPlexMono-Regular.ttf +0 -0
  408. package/skills/examples/canvas-design/canvas-fonts/IBMPlexSerif-Bold.ttf +0 -0
  409. package/skills/examples/canvas-design/canvas-fonts/IBMPlexSerif-BoldItalic.ttf +0 -0
  410. package/skills/examples/canvas-design/canvas-fonts/IBMPlexSerif-Italic.ttf +0 -0
  411. package/skills/examples/canvas-design/canvas-fonts/IBMPlexSerif-Regular.ttf +0 -0
  412. package/skills/examples/canvas-design/canvas-fonts/InstrumentSans-Bold.ttf +0 -0
  413. package/skills/examples/canvas-design/canvas-fonts/InstrumentSans-BoldItalic.ttf +0 -0
  414. package/skills/examples/canvas-design/canvas-fonts/InstrumentSans-Italic.ttf +0 -0
  415. package/skills/examples/canvas-design/canvas-fonts/InstrumentSans-OFL.txt +93 -0
  416. package/skills/examples/canvas-design/canvas-fonts/InstrumentSans-Regular.ttf +0 -0
  417. package/skills/examples/canvas-design/canvas-fonts/InstrumentSerif-Italic.ttf +0 -0
  418. package/skills/examples/canvas-design/canvas-fonts/InstrumentSerif-Regular.ttf +0 -0
  419. package/skills/examples/canvas-design/canvas-fonts/Italiana-OFL.txt +93 -0
  420. package/skills/examples/canvas-design/canvas-fonts/Italiana-Regular.ttf +0 -0
  421. package/skills/examples/canvas-design/canvas-fonts/JetBrainsMono-Bold.ttf +0 -0
  422. package/skills/examples/canvas-design/canvas-fonts/JetBrainsMono-OFL.txt +93 -0
  423. package/skills/examples/canvas-design/canvas-fonts/JetBrainsMono-Regular.ttf +0 -0
  424. package/skills/examples/canvas-design/canvas-fonts/Jura-Light.ttf +0 -0
  425. package/skills/examples/canvas-design/canvas-fonts/Jura-Medium.ttf +0 -0
  426. package/skills/examples/canvas-design/canvas-fonts/Jura-OFL.txt +93 -0
  427. package/skills/examples/canvas-design/canvas-fonts/LibreBaskerville-OFL.txt +93 -0
  428. package/skills/examples/canvas-design/canvas-fonts/LibreBaskerville-Regular.ttf +0 -0
  429. package/skills/examples/canvas-design/canvas-fonts/Lora-Bold.ttf +0 -0
  430. package/skills/examples/canvas-design/canvas-fonts/Lora-BoldItalic.ttf +0 -0
  431. package/skills/examples/canvas-design/canvas-fonts/Lora-Italic.ttf +0 -0
  432. package/skills/examples/canvas-design/canvas-fonts/Lora-OFL.txt +93 -0
  433. package/skills/examples/canvas-design/canvas-fonts/Lora-Regular.ttf +0 -0
  434. package/skills/examples/canvas-design/canvas-fonts/NationalPark-Bold.ttf +0 -0
  435. package/skills/examples/canvas-design/canvas-fonts/NationalPark-OFL.txt +93 -0
  436. package/skills/examples/canvas-design/canvas-fonts/NationalPark-Regular.ttf +0 -0
  437. package/skills/examples/canvas-design/canvas-fonts/NothingYouCouldDo-OFL.txt +93 -0
  438. package/skills/examples/canvas-design/canvas-fonts/NothingYouCouldDo-Regular.ttf +0 -0
  439. package/skills/examples/canvas-design/canvas-fonts/Outfit-Bold.ttf +0 -0
  440. package/skills/examples/canvas-design/canvas-fonts/Outfit-OFL.txt +93 -0
  441. package/skills/examples/canvas-design/canvas-fonts/Outfit-Regular.ttf +0 -0
  442. package/skills/examples/canvas-design/canvas-fonts/PixelifySans-Medium.ttf +0 -0
  443. package/skills/examples/canvas-design/canvas-fonts/PixelifySans-OFL.txt +93 -0
  444. package/skills/examples/canvas-design/canvas-fonts/PoiretOne-OFL.txt +93 -0
  445. package/skills/examples/canvas-design/canvas-fonts/PoiretOne-Regular.ttf +0 -0
  446. package/skills/examples/canvas-design/canvas-fonts/RedHatMono-Bold.ttf +0 -0
  447. package/skills/examples/canvas-design/canvas-fonts/RedHatMono-OFL.txt +93 -0
  448. package/skills/examples/canvas-design/canvas-fonts/RedHatMono-Regular.ttf +0 -0
  449. package/skills/examples/canvas-design/canvas-fonts/Silkscreen-OFL.txt +93 -0
  450. package/skills/examples/canvas-design/canvas-fonts/Silkscreen-Regular.ttf +0 -0
  451. package/skills/examples/canvas-design/canvas-fonts/SmoochSans-Medium.ttf +0 -0
  452. package/skills/examples/canvas-design/canvas-fonts/SmoochSans-OFL.txt +93 -0
  453. package/skills/examples/canvas-design/canvas-fonts/Tektur-Medium.ttf +0 -0
  454. package/skills/examples/canvas-design/canvas-fonts/Tektur-OFL.txt +93 -0
  455. package/skills/examples/canvas-design/canvas-fonts/Tektur-Regular.ttf +0 -0
  456. package/skills/examples/canvas-design/canvas-fonts/WorkSans-Bold.ttf +0 -0
  457. package/skills/examples/canvas-design/canvas-fonts/WorkSans-BoldItalic.ttf +0 -0
  458. package/skills/examples/canvas-design/canvas-fonts/WorkSans-Italic.ttf +0 -0
  459. package/skills/examples/canvas-design/canvas-fonts/WorkSans-OFL.txt +93 -0
  460. package/skills/examples/canvas-design/canvas-fonts/WorkSans-Regular.ttf +0 -0
  461. package/skills/examples/canvas-design/canvas-fonts/YoungSerif-OFL.txt +93 -0
  462. package/skills/examples/canvas-design/canvas-fonts/YoungSerif-Regular.ttf +0 -0
  463. package/skills/examples/copy-editing/SKILL.md +447 -0
  464. package/skills/examples/copy-editing/evals/evals.json +89 -0
  465. package/skills/examples/copy-editing/references/plain-english-alternatives.md +394 -0
  466. package/skills/examples/internal-comms/LICENSE.txt +202 -0
  467. package/skills/examples/internal-comms/SKILL.md +32 -0
  468. package/skills/examples/internal-comms/examples/3p-updates.md +47 -0
  469. package/skills/examples/internal-comms/examples/company-newsletter.md +65 -0
  470. package/skills/examples/internal-comms/examples/faq-answers.md +30 -0
  471. package/skills/examples/internal-comms/examples/general-comms.md +16 -0
  472. package/skills/examples/mcp-builder/SKILL.md +328 -0
  473. package/skills/examples/mcp-builder/reference/evaluation.md +602 -0
  474. package/skills/examples/mcp-builder/reference/mcp_best_practices.md +915 -0
  475. package/skills/examples/mcp-builder/reference/node_mcp_server.md +916 -0
  476. package/skills/examples/mcp-builder/reference/python_mcp_server.md +752 -0
  477. package/skills/examples/mcp-builder/scripts/connections.py +151 -0
  478. package/skills/examples/mcp-builder/scripts/evaluation.py +373 -0
  479. package/skills/examples/mcp-builder/scripts/example_evaluation.xml +22 -0
  480. package/skills/examples/mcp-builder/scripts/requirements.txt +2 -0
  481. package/skills/examples/product-marketing-context/SKILL.md +241 -0
  482. package/skills/examples/product-marketing-context/evals/evals.json +85 -0
  483. package/skills/examples/single-cell-rna-qc/SKILL.md +175 -0
  484. package/skills/examples/single-cell-rna-qc/references/scverse_qc_guidelines.md +186 -0
  485. package/skills/examples/single-cell-rna-qc/scripts/qc_analysis.py +232 -0
  486. package/skills/examples/single-cell-rna-qc/scripts/qc_core.py +233 -0
  487. package/skills/examples/single-cell-rna-qc/scripts/qc_plotting.py +235 -0
  488. package/skills/examples/skill-creator/SKILL.md +355 -0
  489. package/skills/examples/skill-creator/references/output-patterns.md +82 -0
  490. package/skills/examples/skill-creator/references/workflows.md +28 -0
  491. package/skills/examples/skill-creator/scripts/init_skill.py +303 -0
  492. package/skills/examples/skill-creator/scripts/package_skill.py +110 -0
  493. package/skills/examples/skill-creator/scripts/quick_validate.py +95 -0
  494. package/skills/examples/slack-gif-creator/SKILL.md +254 -0
  495. package/skills/examples/slack-gif-creator/core/easing.py +234 -0
  496. package/skills/examples/slack-gif-creator/core/frame_composer.py +176 -0
  497. package/skills/examples/slack-gif-creator/core/gif_builder.py +269 -0
  498. package/skills/examples/slack-gif-creator/core/validators.py +136 -0
  499. package/skills/examples/slack-gif-creator/requirements.txt +4 -0
  500. package/skills/examples/social-content/SKILL.md +278 -0
  501. package/skills/examples/social-content/evals/evals.json +92 -0
  502. package/skills/examples/social-content/references/platforms.md +170 -0
  503. package/skills/examples/social-content/references/post-templates.md +177 -0
  504. package/skills/examples/social-content/references/reverse-engineering.md +195 -0
  505. package/skills/examples/theme-factory/SKILL.md +59 -0
  506. package/skills/examples/theme-factory/theme-showcase.pdf +0 -0
  507. package/skills/examples/theme-factory/themes/arctic-frost.md +19 -0
  508. package/skills/examples/theme-factory/themes/botanical-garden.md +19 -0
  509. package/skills/examples/theme-factory/themes/desert-rose.md +19 -0
  510. package/skills/examples/theme-factory/themes/forest-canopy.md +19 -0
  511. package/skills/examples/theme-factory/themes/golden-hour.md +19 -0
  512. package/skills/examples/theme-factory/themes/midnight-galaxy.md +19 -0
  513. package/skills/examples/theme-factory/themes/modern-minimalist.md +19 -0
  514. package/skills/examples/theme-factory/themes/ocean-depths.md +19 -0
  515. package/skills/examples/theme-factory/themes/sunset-boulevard.md +19 -0
  516. package/skills/examples/theme-factory/themes/tech-innovation.md +19 -0
  517. package/skills/examples/web-artifacts-builder/LICENSE.txt +202 -0
  518. package/skills/examples/web-artifacts-builder/SKILL.md +74 -0
  519. package/skills/examples/web-artifacts-builder/scripts/bundle-artifact.sh +54 -0
  520. package/skills/examples/web-artifacts-builder/scripts/init-artifact.sh +322 -0
  521. package/skills/examples/web-artifacts-builder/scripts/shadcn-components.tar.gz +0 -0
  522. package/skills/examples/writing-skills/SKILL.md +655 -0
  523. package/skills/examples/writing-skills/anthropic-best-practices.md +1150 -0
  524. package/skills/examples/writing-skills/examples/CLAUDE_MD_TESTING.md +189 -0
  525. package/skills/examples/writing-skills/graphviz-conventions.dot +172 -0
  526. package/skills/examples/writing-skills/persuasion-principles.md +187 -0
  527. package/skills/examples/writing-skills/render-graphs.js +168 -0
  528. package/skills/examples/writing-skills/testing-skills-with-subagents.md +384 -0
  529. package/skills/public/describe-image/SKILL.md +105 -0
  530. package/skills/public/describe-image/scripts/describe.py +389 -0
  531. package/skills/public/doc-coauthoring/SKILL.md +375 -0
  532. package/skills/public/docx/LICENSE.txt +30 -0
  533. package/skills/public/docx/SKILL.md +199 -0
  534. package/skills/public/docx/docx-js.md +350 -0
  535. package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-chart.xsd +1499 -0
  536. package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-chartDrawing.xsd +146 -0
  537. package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-diagram.xsd +1085 -0
  538. package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-lockedCanvas.xsd +11 -0
  539. package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-main.xsd +3081 -0
  540. package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-picture.xsd +23 -0
  541. package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-spreadsheetDrawing.xsd +185 -0
  542. package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-wordprocessingDrawing.xsd +287 -0
  543. package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/pml.xsd +1676 -0
  544. package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-additionalCharacteristics.xsd +28 -0
  545. package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-bibliography.xsd +144 -0
  546. package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-commonSimpleTypes.xsd +174 -0
  547. package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-customXmlDataProperties.xsd +25 -0
  548. package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-customXmlSchemaProperties.xsd +18 -0
  549. package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesCustom.xsd +59 -0
  550. package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesExtended.xsd +56 -0
  551. package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesVariantTypes.xsd +195 -0
  552. package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-math.xsd +582 -0
  553. package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-relationshipReference.xsd +25 -0
  554. package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/sml.xsd +4439 -0
  555. package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/vml-main.xsd +570 -0
  556. package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/vml-officeDrawing.xsd +509 -0
  557. package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/vml-presentationDrawing.xsd +12 -0
  558. package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/vml-spreadsheetDrawing.xsd +108 -0
  559. package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/vml-wordprocessingDrawing.xsd +96 -0
  560. package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/wml.xsd +3646 -0
  561. package/skills/public/docx/ooxml/schemas/ISO-IEC29500-4_2016/xml.xsd +116 -0
  562. package/skills/public/docx/ooxml/schemas/ecma/fouth-edition/opc-contentTypes.xsd +42 -0
  563. package/skills/public/docx/ooxml/schemas/ecma/fouth-edition/opc-coreProperties.xsd +50 -0
  564. package/skills/public/docx/ooxml/schemas/ecma/fouth-edition/opc-digSig.xsd +49 -0
  565. package/skills/public/docx/ooxml/schemas/ecma/fouth-edition/opc-relationships.xsd +33 -0
  566. package/skills/public/docx/ooxml/schemas/mce/mc.xsd +75 -0
  567. package/skills/public/docx/ooxml/schemas/microsoft/wml-2010.xsd +560 -0
  568. package/skills/public/docx/ooxml/schemas/microsoft/wml-2012.xsd +67 -0
  569. package/skills/public/docx/ooxml/schemas/microsoft/wml-2018.xsd +14 -0
  570. package/skills/public/docx/ooxml/schemas/microsoft/wml-cex-2018.xsd +20 -0
  571. package/skills/public/docx/ooxml/schemas/microsoft/wml-cid-2016.xsd +13 -0
  572. package/skills/public/docx/ooxml/schemas/microsoft/wml-sdtdatahash-2020.xsd +4 -0
  573. package/skills/public/docx/ooxml/schemas/microsoft/wml-symex-2015.xsd +8 -0
  574. package/skills/public/docx/ooxml/scripts/pack.py +159 -0
  575. package/skills/public/docx/ooxml/scripts/unpack.py +29 -0
  576. package/skills/public/docx/ooxml/scripts/validate.py +69 -0
  577. package/skills/public/docx/ooxml/scripts/validation/__init__.py +15 -0
  578. package/skills/public/docx/ooxml/scripts/validation/base.py +951 -0
  579. package/skills/public/docx/ooxml/scripts/validation/docx.py +274 -0
  580. package/skills/public/docx/ooxml/scripts/validation/pptx.py +315 -0
  581. package/skills/public/docx/ooxml/scripts/validation/redlining.py +279 -0
  582. package/skills/public/docx/ooxml.md +632 -0
  583. package/skills/public/docx/scripts/__init__.py +1 -0
  584. package/skills/public/docx/scripts/document.py +1292 -0
  585. package/skills/public/docx/scripts/templates/comments.xml +3 -0
  586. package/skills/public/docx/scripts/templates/commentsExtended.xml +3 -0
  587. package/skills/public/docx/scripts/templates/commentsExtensible.xml +3 -0
  588. package/skills/public/docx/scripts/templates/commentsIds.xml +3 -0
  589. package/skills/public/docx/scripts/templates/people.xml +3 -0
  590. package/skills/public/docx/scripts/utilities.py +374 -0
  591. package/skills/public/file-reading/LICENSE.txt +30 -0
  592. package/skills/public/file-reading/SKILL.md +350 -0
  593. package/skills/public/frontend-design/LICENSE.txt +177 -0
  594. package/skills/public/frontend-design/SKILL.md +42 -0
  595. package/skills/public/gitlab-explorer/SKILL.md +174 -0
  596. package/skills/public/gitlab-explorer/references/git-commands.md +323 -0
  597. package/skills/public/gitlab-explorer/references/glab-commands.md +282 -0
  598. package/skills/public/gitlab-explorer/scripts/check_gitlab_auth.sh +109 -0
  599. package/skills/public/pdf/FORMS.md +205 -0
  600. package/skills/public/pdf/REFERENCE.md +612 -0
  601. package/skills/public/pdf/SKILL.md +364 -0
  602. package/skills/public/pdf/scripts/check_bounding_boxes.py +70 -0
  603. package/skills/public/pdf/scripts/check_bounding_boxes_test.py +226 -0
  604. package/skills/public/pdf/scripts/check_fillable_fields.py +12 -0
  605. package/skills/public/pdf/scripts/convert_pdf_to_images.py +35 -0
  606. package/skills/public/pdf/scripts/create_validation_image.py +41 -0
  607. package/skills/public/pdf/scripts/extract_form_field_info.py +152 -0
  608. package/skills/public/pdf/scripts/fill_fillable_fields.py +114 -0
  609. package/skills/public/pdf/scripts/fill_pdf_form_with_annotations.py +108 -0
  610. package/skills/public/pdf-reading/LICENSE.txt +30 -0
  611. package/skills/public/pdf-reading/REFERENCE.md +196 -0
  612. package/skills/public/pdf-reading/SKILL.md +305 -0
  613. package/skills/public/playwright-cli/SKILL.md +278 -0
  614. package/skills/public/playwright-cli/references/request-mocking.md +87 -0
  615. package/skills/public/playwright-cli/references/running-code.md +232 -0
  616. package/skills/public/playwright-cli/references/session-management.md +169 -0
  617. package/skills/public/playwright-cli/references/storage-state.md +275 -0
  618. package/skills/public/playwright-cli/references/test-generation.md +88 -0
  619. package/skills/public/playwright-cli/references/tracing.md +139 -0
  620. package/skills/public/playwright-cli/references/video-recording.md +43 -0
  621. package/skills/public/pptx/LICENSE.txt +30 -0
  622. package/skills/public/pptx/SKILL.md +484 -0
  623. package/skills/public/pptx/css.md +335 -0
  624. package/skills/public/pptx/html2pptx.md +893 -0
  625. package/skills/public/pptx/html2pptx.tgz +0 -0
  626. package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-chart.xsd +1499 -0
  627. package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-chartDrawing.xsd +146 -0
  628. package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-diagram.xsd +1085 -0
  629. package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-lockedCanvas.xsd +11 -0
  630. package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-main.xsd +3081 -0
  631. package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-picture.xsd +23 -0
  632. package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-spreadsheetDrawing.xsd +185 -0
  633. package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-wordprocessingDrawing.xsd +287 -0
  634. package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/pml.xsd +1676 -0
  635. package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-additionalCharacteristics.xsd +28 -0
  636. package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-bibliography.xsd +144 -0
  637. package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-commonSimpleTypes.xsd +174 -0
  638. package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-customXmlDataProperties.xsd +25 -0
  639. package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-customXmlSchemaProperties.xsd +18 -0
  640. package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesCustom.xsd +59 -0
  641. package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesExtended.xsd +56 -0
  642. package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesVariantTypes.xsd +195 -0
  643. package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-math.xsd +582 -0
  644. package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-relationshipReference.xsd +25 -0
  645. package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/sml.xsd +4439 -0
  646. package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/vml-main.xsd +570 -0
  647. package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/vml-officeDrawing.xsd +509 -0
  648. package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/vml-presentationDrawing.xsd +12 -0
  649. package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/vml-spreadsheetDrawing.xsd +108 -0
  650. package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/vml-wordprocessingDrawing.xsd +96 -0
  651. package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/wml.xsd +3646 -0
  652. package/skills/public/pptx/ooxml/schemas/ISO-IEC29500-4_2016/xml.xsd +116 -0
  653. package/skills/public/pptx/ooxml/schemas/ecma/fouth-edition/opc-contentTypes.xsd +42 -0
  654. package/skills/public/pptx/ooxml/schemas/ecma/fouth-edition/opc-coreProperties.xsd +50 -0
  655. package/skills/public/pptx/ooxml/schemas/ecma/fouth-edition/opc-digSig.xsd +49 -0
  656. package/skills/public/pptx/ooxml/schemas/ecma/fouth-edition/opc-relationships.xsd +33 -0
  657. package/skills/public/pptx/ooxml/schemas/mce/mc.xsd +75 -0
  658. package/skills/public/pptx/ooxml/schemas/microsoft/wml-2010.xsd +560 -0
  659. package/skills/public/pptx/ooxml/schemas/microsoft/wml-2012.xsd +67 -0
  660. package/skills/public/pptx/ooxml/schemas/microsoft/wml-2018.xsd +14 -0
  661. package/skills/public/pptx/ooxml/schemas/microsoft/wml-cex-2018.xsd +20 -0
  662. package/skills/public/pptx/ooxml/schemas/microsoft/wml-cid-2016.xsd +13 -0
  663. package/skills/public/pptx/ooxml/schemas/microsoft/wml-sdtdatahash-2020.xsd +4 -0
  664. package/skills/public/pptx/ooxml/schemas/microsoft/wml-symex-2015.xsd +8 -0
  665. package/skills/public/pptx/ooxml/scripts/pack.py +159 -0
  666. package/skills/public/pptx/ooxml/scripts/unpack.py +29 -0
  667. package/skills/public/pptx/ooxml/scripts/validate.py +69 -0
  668. package/skills/public/pptx/ooxml/scripts/validation/__init__.py +15 -0
  669. package/skills/public/pptx/ooxml/scripts/validation/base.py +951 -0
  670. package/skills/public/pptx/ooxml/scripts/validation/docx.py +274 -0
  671. package/skills/public/pptx/ooxml/scripts/validation/pptx.py +315 -0
  672. package/skills/public/pptx/ooxml/scripts/validation/redlining.py +279 -0
  673. package/skills/public/pptx/ooxml.md +427 -0
  674. package/skills/public/pptx/scripts/inventory.py +1020 -0
  675. package/skills/public/pptx/scripts/rearrange.py +231 -0
  676. package/skills/public/pptx/scripts/replace.py +385 -0
  677. package/skills/public/pptx/scripts/thumbnail.py +450 -0
  678. package/skills/public/skill-creator/SKILL.md +356 -0
  679. package/skills/public/skill-creator/references/output-patterns.md +82 -0
  680. package/skills/public/skill-creator/references/workflows.md +28 -0
  681. package/skills/public/skill-creator/scripts/init_skill.py +303 -0
  682. package/skills/public/skill-creator/scripts/package_skill.py +110 -0
  683. package/skills/public/skill-creator/scripts/quick_validate.py +95 -0
  684. package/skills/public/sub-agent/SKILL.md +186 -0
  685. package/skills/public/sub-agent/references/security-review.md +153 -0
  686. package/skills/public/sub-agent/references/usage.md +207 -0
  687. package/skills/public/sub-agent/scripts/list_subagent_models.sh +22 -0
  688. package/skills/public/test-driven-development/SKILL.md +371 -0
  689. package/skills/public/test-driven-development/testing-anti-patterns.md +299 -0
  690. package/skills/public/webapp-testing/LICENSE.txt +202 -0
  691. package/skills/public/webapp-testing/SKILL.md +96 -0
  692. package/skills/public/webapp-testing/examples/console_logging.py +35 -0
  693. package/skills/public/webapp-testing/examples/element_discovery.py +40 -0
  694. package/skills/public/webapp-testing/examples/static_html_automation.py +33 -0
  695. package/skills/public/webapp-testing/scripts/with_server.py +106 -0
  696. package/skills/public/xlsx/LICENSE.txt +30 -0
  697. package/skills/public/xlsx/SKILL.md +316 -0
  698. package/skills/public/xlsx/preview_data.py +93 -0
  699. package/skills/public/xlsx/recalc.py +178 -0
  700. package/tests/README.md +42 -0
  701. package/tests/fixtures/cli/claude_v0.9.2.0_argv.json +46 -0
  702. package/tests/fixtures/cli/claude_v0.9.2.0_stdout.json +32 -0
  703. package/tests/fixtures/cli/codex_run.jsonl +4 -0
  704. package/tests/fixtures/cli/opencode_run.jsonl +6 -0
  705. package/tests/integration/README.md +56 -0
  706. package/tests/integration/conftest.py +280 -0
  707. package/tests/integration/pytest.ini +13 -0
  708. package/tests/integration/test_mcp_auth.py +85 -0
  709. package/tests/integration/test_mcp_tools.py +101 -0
  710. package/tests/integration/test_workspace_lifecycle.py +125 -0
  711. package/tests/orchestrator/mock_llm_server.py +343 -0
  712. package/tests/orchestrator/test_cli_adapters.py +566 -0
  713. package/tests/orchestrator/test_cli_adapters_live.py +527 -0
  714. package/tests/orchestrator/test_cli_runtime.py +451 -0
  715. package/tests/orchestrator/test_docker_manager.py +302 -0
  716. package/tests/orchestrator/test_dynamic_instructions.py +69 -0
  717. package/tests/orchestrator/test_mcp_resources.py +140 -0
  718. package/tests/orchestrator/test_mcp_tools.py +224 -0
  719. package/tests/orchestrator/test_passthrough_isolation.py +201 -0
  720. package/tests/orchestrator/test_readme_in_container.py +76 -0
  721. package/tests/orchestrator/test_render_cache.py +84 -0
  722. package/tests/orchestrator/test_runtime_cli_endpoint.py +108 -0
  723. package/tests/orchestrator/test_single_user_mode.py +212 -0
  724. package/tests/orchestrator/test_startup_warnings.py +123 -0
  725. package/tests/orchestrator/test_sub_agent_dispatch.py +327 -0
  726. package/tests/orchestrator/test_subagent_claude_compat.py +367 -0
  727. package/tests/orchestrator/test_system_prompt_endpoint.py +191 -0
  728. package/tests/orchestrator/test_tool_descriptions.py +52 -0
  729. package/tests/orchestrator/test_view_image.py +201 -0
  730. package/tests/patches/conftest.py +30 -0
  731. package/tests/patches/fixtures/__init__.py +10 -0
  732. package/tests/patches/fixtures/middleware_v0.9.1.py +5057 -0
  733. package/tests/patches/fixtures/middleware_v0.9.2.py +5120 -0
  734. package/tests/patches/fixtures/retrieval_v0.9.1.py +2684 -0
  735. package/tests/patches/fixtures/retrieval_v0.9.2.py +2700 -0
  736. package/tests/patches/test_fix_attached_files_position.py +118 -0
  737. package/tests/patches/test_fix_large_tool_args.py +130 -0
  738. package/tests/patches/test_fix_large_tool_results.py +531 -0
  739. package/tests/patches/test_fix_skip_embedding_chat_files.py +160 -0
  740. package/tests/patches/test_fix_skip_rag_files_native_fc.py +120 -0
  741. package/tests/patches/test_fix_tool_loop_errors.py +128 -0
  742. package/tests/security/test_path_traversal_app.py +132 -0
  743. package/tests/security/test_path_traversal_docker.py +36 -0
  744. package/tests/security/test_path_traversal_settings.py +87 -0
  745. package/tests/security/test_safe_path_util.py +166 -0
  746. package/tests/security/test_xss_preview.py +46 -0
  747. package/tests/test-default-model-resolution.py +136 -0
  748. package/tests/test-docker-image.sh +358 -0
  749. package/tests/test-list-subagent-models.sh +421 -0
  750. package/tests/test-mcp-endpoint-live.sh +92 -0
  751. package/tests/test-mcp-native-surface.sh +213 -0
  752. package/tests/test-no-cyrillic.sh +135 -0
  753. package/tests/test-opencode-error-mapping.py +130 -0
  754. package/tests/test-pr88-skills.sh +305 -0
  755. package/tests/test-project-structure.sh +202 -0
  756. package/tests/test-single-user-mode.sh +269 -0
  757. package/tests/test-skill-no-hardcoded-models.sh +65 -0
  758. package/tests/test-subagent-cli-surface.py +137 -0
  759. package/tests/test-subagent-runtime.sh +109 -0
  760. package/tests/test_codex_toml_converter.py +204 -0
  761. package/tests/test_default_resolver_no_legacy_global.py +159 -0
  762. package/tests/test_filter.py +648 -0
  763. package/tests/test_init_sh_unchanged.sh +49 -0
  764. package/tests/test_opencode_alias_map_drop.py +144 -0
  765. package/tests/test_requirements.py +91 -0
  766. package/tests/test_subagent_docstring.py +193 -0
  767. package/tests/test_tools.py +34 -0
  768. package/vendor/extract-text/README.md +46 -0
  769. 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, '&lt;') + '</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'));