atoo-studio 0.0.1 → 0.0.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -0
- package/README.github.md +322 -0
- package/README.md +112 -0
- package/README.npm.md +112 -0
- package/bin/atoo-studio.js +90 -0
- package/dist/src/agents/claude-code-terminal/adapter.d.ts +42 -0
- package/dist/src/agents/claude-code-terminal/adapter.js +166 -0
- package/dist/src/agents/claude-code-terminal/index.d.ts +13 -0
- package/dist/src/agents/claude-code-terminal/index.js +45 -0
- package/dist/src/agents/claude-code-terminal/spawner.d.ts +9 -0
- package/dist/src/agents/claude-code-terminal/spawner.js +37 -0
- package/dist/src/agents/claude-code-terminal-chatro/adapter.d.ts +51 -0
- package/dist/src/agents/claude-code-terminal-chatro/adapter.js +301 -0
- package/dist/src/agents/claude-code-terminal-chatro/index.d.ts +13 -0
- package/dist/src/agents/claude-code-terminal-chatro/index.js +45 -0
- package/dist/src/agents/claude-code-terminal-chatro/jsonl-watcher.d.ts +67 -0
- package/dist/src/agents/claude-code-terminal-chatro/jsonl-watcher.js +431 -0
- package/dist/src/agents/claude-code-terminal-chatro/spawner.d.ts +9 -0
- package/dist/src/agents/claude-code-terminal-chatro/spawner.js +37 -0
- package/dist/src/agents/codex-terminal/adapter.d.ts +40 -0
- package/dist/src/agents/codex-terminal/adapter.js +160 -0
- package/dist/src/agents/codex-terminal/index.d.ts +13 -0
- package/dist/src/agents/codex-terminal/index.js +47 -0
- package/dist/src/agents/codex-terminal/spawner.d.ts +9 -0
- package/dist/src/agents/codex-terminal/spawner.js +56 -0
- package/dist/src/agents/codex-terminal-chatro/adapter.d.ts +58 -0
- package/dist/src/agents/codex-terminal-chatro/adapter.js +266 -0
- package/dist/src/agents/codex-terminal-chatro/index.d.ts +13 -0
- package/dist/src/agents/codex-terminal-chatro/index.js +50 -0
- package/dist/src/agents/codex-terminal-chatro/jsonl-watcher.d.ts +36 -0
- package/dist/src/agents/codex-terminal-chatro/jsonl-watcher.js +205 -0
- package/dist/src/agents/codex-terminal-chatro/spawner.d.ts +9 -0
- package/dist/src/agents/codex-terminal-chatro/spawner.js +57 -0
- package/dist/src/agents/lib/chain-builder.d.ts +21 -0
- package/dist/src/agents/lib/chain-builder.js +139 -0
- package/dist/src/agents/lib/claude/fs-sessions.d.ts +31 -0
- package/dist/src/agents/lib/claude/fs-sessions.js +329 -0
- package/dist/src/agents/lib/claude/jsonl-writer.d.ts +32 -0
- package/dist/src/agents/lib/claude/jsonl-writer.js +342 -0
- package/dist/src/agents/lib/claude/workspace-trust.d.ts +1 -0
- package/dist/src/agents/lib/claude/workspace-trust.js +29 -0
- package/dist/src/agents/lib/codex/fs-sessions.d.ts +34 -0
- package/dist/src/agents/lib/codex/fs-sessions.js +255 -0
- package/dist/src/agents/lib/codex/jsonl-mapper.d.ts +11 -0
- package/dist/src/agents/lib/codex/jsonl-mapper.js +154 -0
- package/dist/src/agents/lib/codex/jsonl-writer.d.ts +8 -0
- package/dist/src/agents/lib/codex/jsonl-writer.js +440 -0
- package/dist/src/agents/lib/fs-tracking.d.ts +36 -0
- package/dist/src/agents/lib/fs-tracking.js +109 -0
- package/dist/src/agents/lib/pty-activity-tracker.d.ts +37 -0
- package/dist/src/agents/lib/pty-activity-tracker.js +105 -0
- package/dist/src/agents/lib/session-id-utils.d.ts +46 -0
- package/dist/src/agents/lib/session-id-utils.js +147 -0
- package/dist/src/agents/lib/session-precreate.d.ts +17 -0
- package/dist/src/agents/lib/session-precreate.js +177 -0
- package/dist/src/agents/registry.d.ts +72 -0
- package/dist/src/agents/registry.js +337 -0
- package/dist/src/agents/types.d.ts +135 -0
- package/dist/src/agents/types.js +1 -0
- package/dist/src/auth/crypto-key.d.ts +6 -0
- package/dist/src/auth/crypto-key.js +45 -0
- package/dist/src/auth/middleware.d.ts +18 -0
- package/dist/src/auth/middleware.js +54 -0
- package/dist/src/auth/password.d.ts +2 -0
- package/dist/src/auth/password.js +12 -0
- package/dist/src/auth/session.d.ts +10 -0
- package/dist/src/auth/session.js +33 -0
- package/dist/src/auth/totp.d.ts +12 -0
- package/dist/src/auth/totp.js +61 -0
- package/dist/src/auth/webauthn.d.ts +6 -0
- package/dist/src/auth/webauthn.js +117 -0
- package/dist/src/config.d.ts +10 -0
- package/dist/src/config.js +16 -0
- package/dist/src/database/connection-manager.d.ts +25 -0
- package/dist/src/database/connection-manager.js +211 -0
- package/dist/src/database/discovery/container.d.ts +6 -0
- package/dist/src/database/discovery/container.js +226 -0
- package/dist/src/database/discovery/env-parser.d.ts +9 -0
- package/dist/src/database/discovery/env-parser.js +525 -0
- package/dist/src/database/discovery/local-files.d.ts +6 -0
- package/dist/src/database/discovery/local-files.js +58 -0
- package/dist/src/database/discovery/port-scan.d.ts +7 -0
- package/dist/src/database/discovery/port-scan.js +61 -0
- package/dist/src/database/drivers/cassandra.d.ts +12 -0
- package/dist/src/database/drivers/cassandra.js +91 -0
- package/dist/src/database/drivers/clickhouse.d.ts +11 -0
- package/dist/src/database/drivers/clickhouse.js +127 -0
- package/dist/src/database/drivers/elasticsearch.d.ts +12 -0
- package/dist/src/database/drivers/elasticsearch.js +169 -0
- package/dist/src/database/drivers/influxdb.d.ts +14 -0
- package/dist/src/database/drivers/influxdb.js +194 -0
- package/dist/src/database/drivers/memcached.d.ts +11 -0
- package/dist/src/database/drivers/memcached.js +117 -0
- package/dist/src/database/drivers/mongodb.d.ts +12 -0
- package/dist/src/database/drivers/mongodb.js +128 -0
- package/dist/src/database/drivers/mysql.d.ts +11 -0
- package/dist/src/database/drivers/mysql.js +112 -0
- package/dist/src/database/drivers/neo4j.d.ts +11 -0
- package/dist/src/database/drivers/neo4j.js +158 -0
- package/dist/src/database/drivers/postgresql.d.ts +11 -0
- package/dist/src/database/drivers/postgresql.js +133 -0
- package/dist/src/database/drivers/redis.d.ts +11 -0
- package/dist/src/database/drivers/redis.js +91 -0
- package/dist/src/database/drivers/sqlite.d.ts +10 -0
- package/dist/src/database/drivers/sqlite.js +100 -0
- package/dist/src/database/query-stream.d.ts +5 -0
- package/dist/src/database/query-stream.js +75 -0
- package/dist/src/database/types.d.ts +71 -0
- package/dist/src/database/types.js +1 -0
- package/dist/src/events/index.d.ts +3 -0
- package/dist/src/events/index.js +3 -0
- package/dist/src/events/types.d.ts +214 -0
- package/dist/src/events/types.js +22 -0
- package/dist/src/events/wire.d.ts +114 -0
- package/dist/src/events/wire.js +296 -0
- package/dist/src/fs-monitor-types.d.ts +24 -0
- package/dist/src/fs-monitor-types.js +1 -0
- package/dist/src/fs-monitor.d.ts +80 -0
- package/dist/src/fs-monitor.js +637 -0
- package/dist/src/handlers/auth.d.ts +1 -0
- package/dist/src/handlers/auth.js +170 -0
- package/dist/src/handlers/changes.d.ts +1 -0
- package/dist/src/handlers/changes.js +203 -0
- package/dist/src/handlers/containers.d.ts +12 -0
- package/dist/src/handlers/containers.js +379 -0
- package/dist/src/handlers/databases.d.ts +3 -0
- package/dist/src/handlers/databases.js +327 -0
- package/dist/src/handlers/environments.d.ts +3 -0
- package/dist/src/handlers/environments.js +286 -0
- package/dist/src/handlers/github.d.ts +1 -0
- package/dist/src/handlers/github.js +153 -0
- package/dist/src/handlers/projects.d.ts +1 -0
- package/dist/src/handlers/projects.js +895 -0
- package/dist/src/handlers/ssh.d.ts +1 -0
- package/dist/src/handlers/ssh.js +162 -0
- package/dist/src/handlers/users.d.ts +1 -0
- package/dist/src/handlers/users.js +195 -0
- package/dist/src/index.d.ts +1 -0
- package/dist/src/index.js +228 -0
- package/dist/src/mcp/config.d.ts +32 -0
- package/dist/src/mcp/config.js +227 -0
- package/dist/src/mcp/server.d.ts +1 -0
- package/dist/src/mcp/server.js +574 -0
- package/dist/src/serial/cuse-device.d.ts +19 -0
- package/dist/src/serial/cuse-device.js +260 -0
- package/dist/src/serial/manager.d.ts +63 -0
- package/dist/src/serial/manager.js +206 -0
- package/dist/src/serial/pty-pair.d.ts +16 -0
- package/dist/src/serial/pty-pair.js +68 -0
- package/dist/src/services/fs-browser.d.ts +14 -0
- package/dist/src/services/fs-browser.js +98 -0
- package/dist/src/services/git-ops.d.ts +78 -0
- package/dist/src/services/git-ops.js +288 -0
- package/dist/src/services/github-ops.d.ts +104 -0
- package/dist/src/services/github-ops.js +192 -0
- package/dist/src/services/obfuscation.d.ts +2 -0
- package/dist/src/services/obfuscation.js +16 -0
- package/dist/src/services/preview/headless-backend.d.ts +62 -0
- package/dist/src/services/preview/headless-backend.js +698 -0
- package/dist/src/services/preview/injected-scripts.d.ts +9 -0
- package/dist/src/services/preview/injected-scripts.js +232 -0
- package/dist/src/services/preview/preview-backend.d.ts +92 -0
- package/dist/src/services/preview/preview-backend.js +15 -0
- package/dist/src/services/preview/universal-setter.d.ts +7 -0
- package/dist/src/services/preview/universal-setter.js +46 -0
- package/dist/src/services/preview-manager.d.ts +50 -0
- package/dist/src/services/preview-manager.js +216 -0
- package/dist/src/services/project-watcher.d.ts +6 -0
- package/dist/src/services/project-watcher.js +307 -0
- package/dist/src/services/remote-fs-browser.d.ts +11 -0
- package/dist/src/services/remote-fs-browser.js +50 -0
- package/dist/src/services/remote-git-ops.d.ts +71 -0
- package/dist/src/services/remote-git-ops.js +215 -0
- package/dist/src/services/session-search.d.ts +56 -0
- package/dist/src/services/session-search.js +303 -0
- package/dist/src/services/ssh-manager.d.ts +44 -0
- package/dist/src/services/ssh-manager.js +359 -0
- package/dist/src/session-writer.d.ts +9 -0
- package/dist/src/session-writer.js +66 -0
- package/dist/src/spawner.d.ts +56 -0
- package/dist/src/spawner.js +135 -0
- package/dist/src/state/db.d.ts +214 -0
- package/dist/src/state/db.js +897 -0
- package/dist/src/state/store.d.ts +37 -0
- package/dist/src/state/store.js +108 -0
- package/dist/src/state/types.d.ts +13 -0
- package/dist/src/state/types.js +1 -0
- package/dist/src/web/devtools-proxy.d.ts +7 -0
- package/dist/src/web/devtools-proxy.js +176 -0
- package/dist/src/web/port-proxy.d.ts +15 -0
- package/dist/src/web/port-proxy.js +124 -0
- package/dist/src/web/preview-ws.d.ts +5 -0
- package/dist/src/web/preview-ws.js +207 -0
- package/dist/src/web/server.d.ts +6 -0
- package/dist/src/web/server.js +1694 -0
- package/dist/src/ws/agent-ws.d.ts +5 -0
- package/dist/src/ws/agent-ws.js +93 -0
- package/frontend/dist/assets/_basePickBy-B-LibQ4-.js +1 -0
- package/frontend/dist/assets/_baseUniq-CprifHap.js +1 -0
- package/frontend/dist/assets/_createAssigner-ByDUqGii.js +1 -0
- package/frontend/dist/assets/abap-DuT-3z4x.js +1 -0
- package/frontend/dist/assets/addon-fit-CxQet2ja.js +1 -0
- package/frontend/dist/assets/addon-web-links-D_jRkPIl.js +1 -0
- package/frontend/dist/assets/apex-B-em86xX.js +1 -0
- package/frontend/dist/assets/api-SUPuHhSY.js +2 -0
- package/frontend/dist/assets/arc-Z0_eVteO.js +1 -0
- package/frontend/dist/assets/architecture-PBZL5I3N-hvVXGhqd.js +1 -0
- package/frontend/dist/assets/architectureDiagram-2XIMDMQ5-DiHPxX4j.js +36 -0
- package/frontend/dist/assets/array-CwG8vNfn.js +1 -0
- package/frontend/dist/assets/auth-store-R7eW5SVu.js +1 -0
- package/frontend/dist/assets/azcli-Bg9wQloi.js +1 -0
- package/frontend/dist/assets/bat-BM46z99L.js +1 -0
- package/frontend/dist/assets/bicep-DcBsJUfh.js +2 -0
- package/frontend/dist/assets/blockDiagram-WCTKOSBZ-C40u_hLo.js +132 -0
- package/frontend/dist/assets/c4Diagram-IC4MRINW-Ct7LjWFQ.js +10 -0
- package/frontend/dist/assets/cameligo-zw7JTtim.js +1 -0
- package/frontend/dist/assets/channel-ClCsE6HN.js +1 -0
- package/frontend/dist/assets/chunk-4BX2VUAB-zZ6P90VO.js +1 -0
- package/frontend/dist/assets/chunk-55IACEB6-DXllTDQl.js +1 -0
- package/frontend/dist/assets/chunk-7E7YKBS2-7zRaOLjj.js +1 -0
- package/frontend/dist/assets/chunk-7R4GIKGN-Csst1274.js +80 -0
- package/frontend/dist/assets/chunk-C72U2L5F-_JbQPbLN.js +1 -0
- package/frontend/dist/assets/chunk-CFjPhJqf.js +1 -0
- package/frontend/dist/assets/chunk-EGIJ26TM-B--aFyPw.js +1 -0
- package/frontend/dist/assets/chunk-FMBD7UC4-DVR34RNb.js +15 -0
- package/frontend/dist/assets/chunk-GEFDOKGD-CnmN6cC8.js +2 -0
- package/frontend/dist/assets/chunk-JSJVCQXG-CWxHBzeJ.js +1 -0
- package/frontend/dist/assets/chunk-KX2RTZJC-DkRk56s7.js +1 -0
- package/frontend/dist/assets/chunk-KYZI473N-DCCsG2dK.js +53 -0
- package/frontend/dist/assets/chunk-L3YUKLVL-C-DkZTMr.js +1 -0
- package/frontend/dist/assets/chunk-MX3YWQON-OUdzv5sZ.js +1 -0
- package/frontend/dist/assets/chunk-NQ4KR5QH-Bpu9FsM7.js +220 -0
- package/frontend/dist/assets/chunk-O4XLMI2P-BMLK6_ib.js +7 -0
- package/frontend/dist/assets/chunk-OZEHJAEY-CNNiJtG0.js +1 -0
- package/frontend/dist/assets/chunk-PQ6SQG4A-evVHD3KM.js +1 -0
- package/frontend/dist/assets/chunk-PU5JKC2W-DPFTYuvl.js +70 -0
- package/frontend/dist/assets/chunk-QZHKN3VN-JRdddPvu.js +1 -0
- package/frontend/dist/assets/chunk-R5LLSJPH-CHQzVVOV.js +1 -0
- package/frontend/dist/assets/chunk-WL4C6EOR-BNFU6IIi.js +189 -0
- package/frontend/dist/assets/chunk-XIRO2GV7-98T93G85.js +1 -0
- package/frontend/dist/assets/chunk-XZSTWKYB-BcW3cyNp.js +94 -0
- package/frontend/dist/assets/chunk-YBOYWFTD-BgKO1qAJ.js +1 -0
- package/frontend/dist/assets/classDiagram-VBA2DB6C-DikXzgcD.js +1 -0
- package/frontend/dist/assets/classDiagram-v2-RAHNMMFH-D7E3tQUK.js +1 -0
- package/frontend/dist/assets/clojure-FspFoNNQ.js +1 -0
- package/frontend/dist/assets/clone-mOXuZa7C.js +1 -0
- package/frontend/dist/assets/codicon-ngg6Pgfi.ttf +0 -0
- package/frontend/dist/assets/coffee-13n8Bk2W.js +1 -0
- package/frontend/dist/assets/cose-bilkent-S5V4N54A-zUOWQqLe.js +1 -0
- package/frontend/dist/assets/cpp-BVm2xGEs.js +1 -0
- package/frontend/dist/assets/csharp-D2kAWmUm.js +1 -0
- package/frontend/dist/assets/csp-Ezvgpf0e.js +1 -0
- package/frontend/dist/assets/css-CYxRwcFy.js +3 -0
- package/frontend/dist/assets/css.worker-Cd5h-ZOL.js +89 -0
- package/frontend/dist/assets/cssMode-CrXej49V.js +1 -0
- package/frontend/dist/assets/cypher-jg3SGErc.js +1 -0
- package/frontend/dist/assets/cytoscape.esm-kyyvzxNV.js +321 -0
- package/frontend/dist/assets/dagre-DH4bgZO7.js +1 -0
- package/frontend/dist/assets/dagre-KLK3FWXG-DNSqDkwT.js +4 -0
- package/frontend/dist/assets/dart-179jqhK4.js +1 -0
- package/frontend/dist/assets/defaultLocale-Dda4OpKy.js +1 -0
- package/frontend/dist/assets/diagram-E7M64L7V-RqPNT5Vs.js +24 -0
- package/frontend/dist/assets/diagram-IFDJBPK2-B-5NRyaE.js +43 -0
- package/frontend/dist/assets/diagram-P4PSJMXO-BrP69Hk0.js +24 -0
- package/frontend/dist/assets/dist-CU_Nb1G5.js +1 -0
- package/frontend/dist/assets/dockerfile-CIAtSGxS.js +1 -0
- package/frontend/dist/assets/ecl-CGVKfDxD.js +1 -0
- package/frontend/dist/assets/editor-Br_kD0ds.css +1 -0
- package/frontend/dist/assets/editor.api2-YXkDn0Gm.js +872 -0
- package/frontend/dist/assets/editor.main-fBaXZjJ0.js +6 -0
- package/frontend/dist/assets/elixir-BZ-6w0y3.js +1 -0
- package/frontend/dist/assets/erDiagram-INFDFZHY-BYiB9NYg.js +70 -0
- package/frontend/dist/assets/flow9-CVuOjTMv.js +1 -0
- package/frontend/dist/assets/flowDiagram-PKNHOUZH-Cwq47rsR.js +162 -0
- package/frontend/dist/assets/freemarker2-DM-pztJU.js +3 -0
- package/frontend/dist/assets/fsharp-q0pGJYr6.js +1 -0
- package/frontend/dist/assets/ganttDiagram-A5KZAMGK-Dnx3szD9.js +292 -0
- package/frontend/dist/assets/gitGraph-HDMCJU4V-COlTQ7bA.js +1 -0
- package/frontend/dist/assets/gitGraphDiagram-K3NZZRJ6-BaUxboNc.js +65 -0
- package/frontend/dist/assets/go-dzSPfdEO.js +1 -0
- package/frontend/dist/assets/graphlib-kEFlkt3U.js +1 -0
- package/frontend/dist/assets/graphql-CG4OUoEV.js +1 -0
- package/frontend/dist/assets/handlebars-BbK53Vec.js +1 -0
- package/frontend/dist/assets/hcl-Cy14JPk3.js +1 -0
- package/frontend/dist/assets/html-DYtTQNOG.js +1 -0
- package/frontend/dist/assets/html.worker-BjVEKLoU.js +502 -0
- package/frontend/dist/assets/htmlMode-C6GTouth.js +1 -0
- package/frontend/dist/assets/index-DMLxes_u.js +157 -0
- package/frontend/dist/assets/index-DmzeqkB1.css +1 -0
- package/frontend/dist/assets/info-3K5VOQVL-DBtHyA4C.js +1 -0
- package/frontend/dist/assets/infoDiagram-LFFYTUFH-yBXLgMPI.js +2 -0
- package/frontend/dist/assets/ini-Pbg8HGVD.js +1 -0
- package/frontend/dist/assets/init-D6KNwrax.js +1 -0
- package/frontend/dist/assets/ishikawaDiagram-PHBUUO56-Bld4two_.js +70 -0
- package/frontend/dist/assets/java-BmVu6Qrl.js +1 -0
- package/frontend/dist/assets/javascript-PbfQEdcJ.js +1 -0
- package/frontend/dist/assets/journeyDiagram-4ABVD52K-4HyMd4R2.js +139 -0
- package/frontend/dist/assets/json.worker-DqU5Wxnl.js +58 -0
- package/frontend/dist/assets/jsonMode-CASsGppE.js +7 -0
- package/frontend/dist/assets/julia-3cGnieBq.js +1 -0
- package/frontend/dist/assets/kanban-definition-K7BYSVSG-DpgsZmpG.js +89 -0
- package/frontend/dist/assets/katex-CEw3x5bf.js +261 -0
- package/frontend/dist/assets/kotlin-BuWkVcfV.js +1 -0
- package/frontend/dist/assets/less-CJ_VPy2C.js +2 -0
- package/frontend/dist/assets/lexon-BygAuZPu.js +1 -0
- package/frontend/dist/assets/line-CA_wh_TY.js +1 -0
- package/frontend/dist/assets/linear-BAcLW45z.js +1 -0
- package/frontend/dist/assets/liquid-kz84dle6.js +1 -0
- package/frontend/dist/assets/lspLanguageFeatures-C7hAHFn1.js +4 -0
- package/frontend/dist/assets/lua-C8Xs3dCx.js +1 -0
- package/frontend/dist/assets/m3-DTJeKBk4.js +1 -0
- package/frontend/dist/assets/markdown-QCgx8JqZ.js +1 -0
- package/frontend/dist/assets/math-D0YcMJAn.js +1 -0
- package/frontend/dist/assets/mdx-yRw0ap-E.js +1 -0
- package/frontend/dist/assets/mermaid-parser.core-DAeTodBQ.js +4 -0
- package/frontend/dist/assets/mindmap-definition-YRQLILUH-CoNlFyVl.js +68 -0
- package/frontend/dist/assets/mips-DopWaYgE.js +1 -0
- package/frontend/dist/assets/monaco.contribution-DeY0Qei-.js +2 -0
- package/frontend/dist/assets/msdax-BDis4ARV.js +1 -0
- package/frontend/dist/assets/mysql-BV6MLsOI.js +1 -0
- package/frontend/dist/assets/objective-c-B1UuzKs6.js +1 -0
- package/frontend/dist/assets/ordinal-jM7S0YHN.js +1 -0
- package/frontend/dist/assets/packet-RMMSAZCW-FF6-Tmai.js +1 -0
- package/frontend/dist/assets/pascal-BkvESCrc.js +1 -0
- package/frontend/dist/assets/pascaligo-lTy0kZYr.js +1 -0
- package/frontend/dist/assets/path-DNPd7Py7.js +1 -0
- package/frontend/dist/assets/perl-CrtUPXLV.js +1 -0
- package/frontend/dist/assets/pgsql-B9IbNWx2.js +1 -0
- package/frontend/dist/assets/php-CXvQBY2p.js +1 -0
- package/frontend/dist/assets/pie-UPGHQEXC-CFvXY2o-.js +1 -0
- package/frontend/dist/assets/pieDiagram-SKSYHLDU-CM_hbCcn.js +30 -0
- package/frontend/dist/assets/pla-DxBxuqWu.js +1 -0
- package/frontend/dist/assets/postiats-OkEuT5YF.js +1 -0
- package/frontend/dist/assets/powerquery-CMx5Tq4K.js +1 -0
- package/frontend/dist/assets/powershell-CstRxrEc.js +1 -0
- package/frontend/dist/assets/preload-helper-D4M6sveU.js +1 -0
- package/frontend/dist/assets/protobuf-Bx0Z-uRj.js +2 -0
- package/frontend/dist/assets/pug--W8vanWl.js +1 -0
- package/frontend/dist/assets/python-DA0rnlw3.js +1 -0
- package/frontend/dist/assets/qsharp-CRtr0YbN.js +1 -0
- package/frontend/dist/assets/quadrantDiagram-337W2JSQ-B3n3IUhC.js +7 -0
- package/frontend/dist/assets/r-C6E1d6iv.js +1 -0
- package/frontend/dist/assets/radar-KQ55EAFF-MPZu7SdX.js +1 -0
- package/frontend/dist/assets/razor-yd73uata.js +1 -0
- package/frontend/dist/assets/redis-Dx13voP3.js +1 -0
- package/frontend/dist/assets/redshift-D66HwlyV.js +1 -0
- package/frontend/dist/assets/requirementDiagram-Z7DCOOCP-CorP7L7F.js +73 -0
- package/frontend/dist/assets/restructuredtext-DQT2NKJ2.js +1 -0
- package/frontend/dist/assets/rough.esm-DxAX5Vpo.js +1 -0
- package/frontend/dist/assets/ruby-iFXI8hwH.js +1 -0
- package/frontend/dist/assets/rust-CSKiei34.js +1 -0
- package/frontend/dist/assets/sankeyDiagram-WA2Y5GQK-RDx6Bd-B.js +10 -0
- package/frontend/dist/assets/sb-Bo3ttdP2.js +1 -0
- package/frontend/dist/assets/scala-BC1D-Nxp.js +1 -0
- package/frontend/dist/assets/scheme-Z4OAo4Lv.js +1 -0
- package/frontend/dist/assets/scss-BvrdPs6B.js +3 -0
- package/frontend/dist/assets/sequenceDiagram-2WXFIKYE-JMqJSFq6.js +145 -0
- package/frontend/dist/assets/shell-Bh_aCyF-.js +1 -0
- package/frontend/dist/assets/solidity-CWHj6tSe.js +1 -0
- package/frontend/dist/assets/sophia-raoNtKtm.js +1 -0
- package/frontend/dist/assets/sparql-XzmoGnue.js +1 -0
- package/frontend/dist/assets/sql-BD0i9Gvg.js +1 -0
- package/frontend/dist/assets/src-Bn-kKzs7.js +1 -0
- package/frontend/dist/assets/st-DtVKyms6.js +1 -0
- package/frontend/dist/assets/stateDiagram-RAJIS63D-CgFfENdy.js +1 -0
- package/frontend/dist/assets/stateDiagram-v2-FVOUBMTO-C4Hh2P-U.js +1 -0
- package/frontend/dist/assets/swift--UZs77wT.js +1 -0
- package/frontend/dist/assets/systemverilog-CDnBSWUd.js +1 -0
- package/frontend/dist/assets/tcl-DdCEuTHZ.js +1 -0
- package/frontend/dist/assets/timeline-definition-YZTLITO2-BnatPBR5.js +61 -0
- package/frontend/dist/assets/treemap-KZPCXAKY-qb1Pl9la.js +1 -0
- package/frontend/dist/assets/ts.worker-DyPAEIuH.js +67719 -0
- package/frontend/dist/assets/tsMode-iuvyEpyO.js +11 -0
- package/frontend/dist/assets/twig-SSL-Altf.js +1 -0
- package/frontend/dist/assets/typescript-17918Hud.js +1 -0
- package/frontend/dist/assets/typespec-BT7S0ETg.js +1 -0
- package/frontend/dist/assets/vb-CrIgucua.js +1 -0
- package/frontend/dist/assets/vennDiagram-LZ73GAT5-DygS4Zzd.js +34 -0
- package/frontend/dist/assets/wgsl-BeKc3oEp.js +298 -0
- package/frontend/dist/assets/workers-DTfwKVoM.js +1 -0
- package/frontend/dist/assets/xml-CBMr_Wbw.js +1 -0
- package/frontend/dist/assets/xterm-BrP-ENHg.css +1 -0
- package/frontend/dist/assets/xterm-CBX2m0YM.js +36 -0
- package/frontend/dist/assets/xychartDiagram-JWTSCODW-D6wY1Jwd.js +7 -0
- package/frontend/dist/assets/yaml-CTjCH7Bv.js +1 -0
- package/frontend/dist/fonts/inter-300.ttf +0 -0
- package/frontend/dist/fonts/inter-400.ttf +0 -0
- package/frontend/dist/fonts/inter-500.ttf +0 -0
- package/frontend/dist/fonts/inter-600.ttf +0 -0
- package/frontend/dist/fonts/inter-700.ttf +0 -0
- package/frontend/dist/index.html +49 -0
- package/frontend/dist/logo_192x192.png +0 -0
- package/frontend/dist/logo_32x32.png +0 -0
- package/frontend/dist/logo_512x512.png +0 -0
- package/frontend/dist/logo_64x64.png +0 -0
- package/frontend/dist/logobg_192x192.png +0 -0
- package/frontend/dist/logobg_512x512.png +0 -0
- package/frontend/dist/logobg_64x64.png +0 -0
- package/frontend/dist/manifest.json +25 -0
- package/frontend/dist/sw.js +22 -0
- package/package.json +74 -7
- package/preload/Makefile +12 -0
- package/preload/atoo-studio-preload.c +647 -0
- package/preload/atoo-studio-preload.so +0 -0
- package/setup-cuse.sh +260 -0
- package/setup.sh +81 -0
- package/src/serial/native/binding.gyp +10 -0
- package/src/serial/native/pty_pair.c +222 -0
|
@@ -0,0 +1,170 @@
|
|
|
1
|
+
import { Router } from 'express';
|
|
2
|
+
import { z } from 'zod';
|
|
3
|
+
import { db } from '../state/db.js';
|
|
4
|
+
import { hashPassword, verifyPassword } from '../auth/password.js';
|
|
5
|
+
import { createSession, setSessionCookie, clearSessionCookie, destroySession, SESSION_COOKIE_NAME } from '../auth/session.js';
|
|
6
|
+
import { getSessionUser } from '../auth/session.js';
|
|
7
|
+
import { hasTotpEnabled, getTotpSecret, verifyTotpToken } from '../auth/totp.js';
|
|
8
|
+
import { getAuthOptions, verifyAuth } from '../auth/webauthn.js';
|
|
9
|
+
import { requireAuth } from '../auth/middleware.js';
|
|
10
|
+
export const authRouter = Router();
|
|
11
|
+
// GET /api/auth/status — public, returns setup state
|
|
12
|
+
authRouter.get('/api/auth/status', (_req, res) => {
|
|
13
|
+
const userCount = db.getUserCount();
|
|
14
|
+
res.json({ setupRequired: userCount === 0, userCount });
|
|
15
|
+
});
|
|
16
|
+
// POST /api/auth/setup — create initial admin (only when no users exist)
|
|
17
|
+
const setupSchema = z.object({
|
|
18
|
+
username: z.string().min(1).max(64),
|
|
19
|
+
display_name: z.string().min(1).max(128),
|
|
20
|
+
password: z.string().min(8).max(256),
|
|
21
|
+
});
|
|
22
|
+
authRouter.post('/api/auth/setup', async (req, res) => {
|
|
23
|
+
if (db.getUserCount() > 0) {
|
|
24
|
+
return res.status(400).json({ error: 'Setup already completed' });
|
|
25
|
+
}
|
|
26
|
+
const parsed = setupSchema.safeParse(req.body);
|
|
27
|
+
if (!parsed.success) {
|
|
28
|
+
return res.status(400).json({ error: 'Invalid input', details: parsed.error.flatten() });
|
|
29
|
+
}
|
|
30
|
+
const { username, display_name, password } = parsed.data;
|
|
31
|
+
const passwordHash = await hashPassword(password);
|
|
32
|
+
const user = db.createUser(username, display_name, 'admin', passwordHash);
|
|
33
|
+
// Assign all existing unowned environments to this admin
|
|
34
|
+
db.assignUnownedEnvironments(user.id);
|
|
35
|
+
const sessionId = createSession(user.id, req);
|
|
36
|
+
setSessionCookie(res, sessionId);
|
|
37
|
+
res.json({
|
|
38
|
+
user: { id: user.id, username: user.username, display_name: user.display_name, role: user.role },
|
|
39
|
+
});
|
|
40
|
+
});
|
|
41
|
+
// POST /api/auth/login — password login
|
|
42
|
+
const loginSchema = z.object({
|
|
43
|
+
username: z.string().min(1),
|
|
44
|
+
password: z.string().min(1),
|
|
45
|
+
});
|
|
46
|
+
authRouter.post('/api/auth/login', async (req, res) => {
|
|
47
|
+
const parsed = loginSchema.safeParse(req.body);
|
|
48
|
+
if (!parsed.success) {
|
|
49
|
+
return res.status(400).json({ error: 'Invalid input' });
|
|
50
|
+
}
|
|
51
|
+
const { username, password } = parsed.data;
|
|
52
|
+
const user = db.getUserByUsername(username);
|
|
53
|
+
if (!user) {
|
|
54
|
+
return res.status(401).json({ error: 'Invalid credentials' });
|
|
55
|
+
}
|
|
56
|
+
const valid = await verifyPassword(password, user.password_hash);
|
|
57
|
+
if (!valid) {
|
|
58
|
+
return res.status(401).json({ error: 'Invalid credentials' });
|
|
59
|
+
}
|
|
60
|
+
// Check if TOTP is enabled — require second factor
|
|
61
|
+
if (hasTotpEnabled(user.id)) {
|
|
62
|
+
// Create a short-lived pending session to track the TOTP step
|
|
63
|
+
const pendingSessionId = createSession(user.id, req);
|
|
64
|
+
return res.json({
|
|
65
|
+
totpRequired: true,
|
|
66
|
+
pendingSessionId,
|
|
67
|
+
});
|
|
68
|
+
}
|
|
69
|
+
const sessionId = createSession(user.id, req);
|
|
70
|
+
setSessionCookie(res, sessionId);
|
|
71
|
+
res.json({
|
|
72
|
+
user: { id: user.id, username: user.username, display_name: user.display_name, role: user.role },
|
|
73
|
+
});
|
|
74
|
+
});
|
|
75
|
+
// POST /api/auth/login/totp — verify TOTP after password
|
|
76
|
+
const totpLoginSchema = z.object({
|
|
77
|
+
pendingSessionId: z.string().min(1),
|
|
78
|
+
token: z.string().length(6),
|
|
79
|
+
});
|
|
80
|
+
authRouter.post('/api/auth/login/totp', (req, res) => {
|
|
81
|
+
const parsed = totpLoginSchema.safeParse(req.body);
|
|
82
|
+
if (!parsed.success) {
|
|
83
|
+
return res.status(400).json({ error: 'Invalid input' });
|
|
84
|
+
}
|
|
85
|
+
const { pendingSessionId, token } = parsed.data;
|
|
86
|
+
const user = getSessionUser(pendingSessionId);
|
|
87
|
+
if (!user) {
|
|
88
|
+
return res.status(401).json({ error: 'Invalid or expired pending session' });
|
|
89
|
+
}
|
|
90
|
+
const secret = getTotpSecret(user.id);
|
|
91
|
+
if (!secret) {
|
|
92
|
+
return res.status(400).json({ error: 'TOTP not configured' });
|
|
93
|
+
}
|
|
94
|
+
if (!verifyTotpToken(secret, token)) {
|
|
95
|
+
return res.status(401).json({ error: 'Invalid TOTP code' });
|
|
96
|
+
}
|
|
97
|
+
// TOTP verified — promote to full session (reuse the pending session)
|
|
98
|
+
setSessionCookie(res, pendingSessionId);
|
|
99
|
+
res.json({
|
|
100
|
+
user: { id: user.id, username: user.username, display_name: user.display_name, role: user.role },
|
|
101
|
+
});
|
|
102
|
+
});
|
|
103
|
+
// POST /api/auth/login/passkey/options — get passkey authentication options
|
|
104
|
+
const passkeyOptionsSchema = z.object({
|
|
105
|
+
username: z.string().min(1),
|
|
106
|
+
});
|
|
107
|
+
authRouter.post('/api/auth/login/passkey/options', async (req, res) => {
|
|
108
|
+
const parsed = passkeyOptionsSchema.safeParse(req.body);
|
|
109
|
+
if (!parsed.success) {
|
|
110
|
+
return res.status(400).json({ error: 'Invalid input' });
|
|
111
|
+
}
|
|
112
|
+
const user = db.getUserByUsername(parsed.data.username);
|
|
113
|
+
if (!user) {
|
|
114
|
+
return res.status(401).json({ error: 'Invalid credentials' });
|
|
115
|
+
}
|
|
116
|
+
const passkeys = db.listPasskeys(user.id);
|
|
117
|
+
if (passkeys.length === 0) {
|
|
118
|
+
return res.status(400).json({ error: 'No passkeys registered' });
|
|
119
|
+
}
|
|
120
|
+
const rpID = req.hostname;
|
|
121
|
+
const options = await getAuthOptions(user.id, rpID);
|
|
122
|
+
res.json({ options, userId: user.id });
|
|
123
|
+
});
|
|
124
|
+
// POST /api/auth/login/passkey — verify passkey authentication
|
|
125
|
+
const passkeyLoginSchema = z.object({
|
|
126
|
+
userId: z.string().min(1),
|
|
127
|
+
response: z.any(),
|
|
128
|
+
});
|
|
129
|
+
authRouter.post('/api/auth/login/passkey', async (req, res) => {
|
|
130
|
+
const parsed = passkeyLoginSchema.safeParse(req.body);
|
|
131
|
+
if (!parsed.success) {
|
|
132
|
+
return res.status(400).json({ error: 'Invalid input' });
|
|
133
|
+
}
|
|
134
|
+
const { userId, response } = parsed.data;
|
|
135
|
+
const user = db.getUser(userId);
|
|
136
|
+
if (!user) {
|
|
137
|
+
return res.status(401).json({ error: 'Invalid credentials' });
|
|
138
|
+
}
|
|
139
|
+
const rpID = req.hostname;
|
|
140
|
+
const origin = `${req.protocol}://${req.headers.host}`;
|
|
141
|
+
const verified = await verifyAuth(userId, rpID, origin, response);
|
|
142
|
+
if (!verified) {
|
|
143
|
+
return res.status(401).json({ error: 'Passkey verification failed' });
|
|
144
|
+
}
|
|
145
|
+
const sessionId = createSession(user.id, req);
|
|
146
|
+
setSessionCookie(res, sessionId);
|
|
147
|
+
res.json({
|
|
148
|
+
user: { id: user.id, username: user.username, display_name: user.display_name, role: user.role },
|
|
149
|
+
});
|
|
150
|
+
});
|
|
151
|
+
// POST /api/auth/logout — destroy session
|
|
152
|
+
authRouter.post('/api/auth/logout', (req, res) => {
|
|
153
|
+
const sessionId = req.cookies?.[SESSION_COOKIE_NAME];
|
|
154
|
+
if (sessionId) {
|
|
155
|
+
destroySession(sessionId);
|
|
156
|
+
}
|
|
157
|
+
clearSessionCookie(res);
|
|
158
|
+
res.json({ ok: true });
|
|
159
|
+
});
|
|
160
|
+
// GET /api/auth/me — get current user info
|
|
161
|
+
authRouter.get('/api/auth/me', requireAuth, (req, res) => {
|
|
162
|
+
const user = req.user;
|
|
163
|
+
const hasTOTP = hasTotpEnabled(user.id);
|
|
164
|
+
const passkeys = db.listPasskeys(user.id);
|
|
165
|
+
res.json({
|
|
166
|
+
user: { id: user.id, username: user.username, display_name: user.display_name, role: user.role },
|
|
167
|
+
hasTOTP,
|
|
168
|
+
passkeyCount: passkeys.length,
|
|
169
|
+
});
|
|
170
|
+
});
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export declare const changesRouter: import("express-serve-static-core").Router;
|
|
@@ -0,0 +1,203 @@
|
|
|
1
|
+
import { Router } from 'express';
|
|
2
|
+
import { fsMonitor } from '../fs-monitor.js';
|
|
3
|
+
import { store } from '../state/store.js';
|
|
4
|
+
export const changesRouter = Router();
|
|
5
|
+
/**
|
|
6
|
+
* Consolidate raw change events to remove artifacts from atomic write patterns.
|
|
7
|
+
*
|
|
8
|
+
* Claude Code does: write(tmp) → rename(tmp → final). With LD_PRELOAD we see:
|
|
9
|
+
* 1. Write(tmp) — open with O_WRONLY|O_CREAT on tmp file
|
|
10
|
+
* 2. Rename(final, old=tmp) — rename from tmp to final
|
|
11
|
+
*
|
|
12
|
+
* Consolidation rules:
|
|
13
|
+
* - Rename absorbs: if a Rename has old_path matching a Write/Create, remove the Write/Create
|
|
14
|
+
* (the tmp file was just an intermediate artifact) and promote the Rename.
|
|
15
|
+
* - Create + Delete cancel: if a path is Created then Deleted in the same batch, remove both.
|
|
16
|
+
* - Deduplicate: multiple writes to the same path keep only the last one.
|
|
17
|
+
*/
|
|
18
|
+
function consolidateChanges(changes) {
|
|
19
|
+
if (changes.length <= 1)
|
|
20
|
+
return changes;
|
|
21
|
+
const result = [...changes];
|
|
22
|
+
const toRemove = new Set();
|
|
23
|
+
// Index changes by path
|
|
24
|
+
const createsByPath = new Map();
|
|
25
|
+
const deletesByPath = new Map();
|
|
26
|
+
for (let i = 0; i < result.length; i++) {
|
|
27
|
+
const c = result[i];
|
|
28
|
+
if (c.operation === 'create') {
|
|
29
|
+
if (!createsByPath.has(c.path))
|
|
30
|
+
createsByPath.set(c.path, []);
|
|
31
|
+
createsByPath.get(c.path).push(i);
|
|
32
|
+
}
|
|
33
|
+
else if (c.operation === 'delete') {
|
|
34
|
+
if (!deletesByPath.has(c.path))
|
|
35
|
+
deletesByPath.set(c.path, []);
|
|
36
|
+
deletesByPath.get(c.path).push(i);
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
// Rule 1: Rename absorbs its old_path Create/Write (remove tmp file artifacts)
|
|
40
|
+
// Promote Rename → Create/Modify
|
|
41
|
+
for (let i = 0; i < result.length; i++) {
|
|
42
|
+
const c = result[i];
|
|
43
|
+
if (c.operation === 'rename' && c.oldPath && !toRemove.has(i)) {
|
|
44
|
+
const oldCreates = createsByPath.get(c.oldPath);
|
|
45
|
+
if (oldCreates) {
|
|
46
|
+
for (const ci of oldCreates) {
|
|
47
|
+
toRemove.add(ci);
|
|
48
|
+
}
|
|
49
|
+
// If before_hash exists, the target file existed before → modify
|
|
50
|
+
// Otherwise it's a new file → create
|
|
51
|
+
const op = c.beforeHash ? 'modify' : 'create';
|
|
52
|
+
result[i] = { ...c, operation: op, oldPath: undefined };
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
// Rule 2: Create + Delete cancel out
|
|
57
|
+
for (const [, createIndices] of createsByPath) {
|
|
58
|
+
const path = result[createIndices[0]]?.path;
|
|
59
|
+
if (!path)
|
|
60
|
+
continue;
|
|
61
|
+
const delIndices = deletesByPath.get(path);
|
|
62
|
+
if (delIndices && delIndices.length > 0) {
|
|
63
|
+
let ci = 0, di = 0;
|
|
64
|
+
while (ci < createIndices.length && di < delIndices.length) {
|
|
65
|
+
const createIdx = createIndices[ci];
|
|
66
|
+
const deleteIdx = delIndices[di];
|
|
67
|
+
if (toRemove.has(createIdx)) {
|
|
68
|
+
ci++;
|
|
69
|
+
continue;
|
|
70
|
+
}
|
|
71
|
+
if (toRemove.has(deleteIdx)) {
|
|
72
|
+
di++;
|
|
73
|
+
continue;
|
|
74
|
+
}
|
|
75
|
+
if (deleteIdx > createIdx) {
|
|
76
|
+
toRemove.add(createIdx);
|
|
77
|
+
toRemove.add(deleteIdx);
|
|
78
|
+
ci++;
|
|
79
|
+
di++;
|
|
80
|
+
}
|
|
81
|
+
else {
|
|
82
|
+
di++;
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
// Rule 3: Deduplicate writes/creates for the same path — keep last
|
|
88
|
+
const dedupByPath = new Map();
|
|
89
|
+
for (let i = 0; i < result.length; i++) {
|
|
90
|
+
if (toRemove.has(i))
|
|
91
|
+
continue;
|
|
92
|
+
const c = result[i];
|
|
93
|
+
if (c.operation === 'create' || c.operation === 'modify') {
|
|
94
|
+
if (!dedupByPath.has(c.path))
|
|
95
|
+
dedupByPath.set(c.path, []);
|
|
96
|
+
dedupByPath.get(c.path).push(i);
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
for (const [, indices] of dedupByPath) {
|
|
100
|
+
if (indices.length <= 1)
|
|
101
|
+
continue;
|
|
102
|
+
const keep = indices[indices.length - 1];
|
|
103
|
+
for (const i of indices) {
|
|
104
|
+
if (i !== keep)
|
|
105
|
+
toRemove.add(i);
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
return result.filter((_, i) => !toRemove.has(i));
|
|
109
|
+
}
|
|
110
|
+
// GET /api/sessions/:id/changes — list file changes for a session
|
|
111
|
+
changesRouter.get('/api/sessions/:id/changes', async (req, res) => {
|
|
112
|
+
const sessionId = req.params.id;
|
|
113
|
+
const session = store.sessions.get(sessionId);
|
|
114
|
+
if (!session)
|
|
115
|
+
return res.status(404).json({ error: 'Session not found' });
|
|
116
|
+
const from = req.query.from ? parseFloat(req.query.from) : undefined;
|
|
117
|
+
const to = req.query.to ? parseFloat(req.query.to) : undefined;
|
|
118
|
+
const changes = consolidateChanges(fsMonitor.getChangesInRange(sessionId, from, to));
|
|
119
|
+
// Materialize after_hash lazily for non-delete changes
|
|
120
|
+
for (const c of changes) {
|
|
121
|
+
if (c.afterHash === null && c.operation !== 'delete') {
|
|
122
|
+
await fsMonitor.materializeAfterHash(c);
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
res.json({
|
|
126
|
+
available: fsMonitor.isAvailable(),
|
|
127
|
+
changes: changes.map(c => ({
|
|
128
|
+
change_id: c.changeId,
|
|
129
|
+
session_id: c.sessionId,
|
|
130
|
+
timestamp: c.timestamp,
|
|
131
|
+
pid: c.pid,
|
|
132
|
+
operation: c.operation,
|
|
133
|
+
path: c.path,
|
|
134
|
+
old_path: c.oldPath || null,
|
|
135
|
+
before_hash: c.beforeHash,
|
|
136
|
+
after_hash: c.afterHash,
|
|
137
|
+
file_size: c.fileSize,
|
|
138
|
+
is_binary: c.isBinary,
|
|
139
|
+
})),
|
|
140
|
+
});
|
|
141
|
+
});
|
|
142
|
+
// GET /api/sessions/:id/changes/:changeId/diff — get diff data for a change
|
|
143
|
+
changesRouter.get('/api/sessions/:id/changes/:changeId/diff', async (req, res) => {
|
|
144
|
+
const change = fsMonitor.findChange(req.params.changeId);
|
|
145
|
+
if (!change)
|
|
146
|
+
return res.status(404).json({ error: 'Change not found' });
|
|
147
|
+
// Materialize after_hash if needed
|
|
148
|
+
if (change.afterHash === null && change.operation !== 'delete') {
|
|
149
|
+
await fsMonitor.materializeAfterHash(change);
|
|
150
|
+
}
|
|
151
|
+
let before = null;
|
|
152
|
+
let after = null;
|
|
153
|
+
if (change.beforeHash) {
|
|
154
|
+
before = await fsMonitor.getFileContent(change.beforeHash);
|
|
155
|
+
}
|
|
156
|
+
if (change.afterHash) {
|
|
157
|
+
after = await fsMonitor.getFileContent(change.afterHash);
|
|
158
|
+
}
|
|
159
|
+
res.json({
|
|
160
|
+
operation: change.operation,
|
|
161
|
+
path: change.path,
|
|
162
|
+
old_path: change.oldPath || null,
|
|
163
|
+
before, // base64-encoded or null
|
|
164
|
+
after, // base64-encoded or null
|
|
165
|
+
is_binary: change.isBinary,
|
|
166
|
+
file_size: change.fileSize,
|
|
167
|
+
before_hash: change.beforeHash,
|
|
168
|
+
after_hash: change.afterHash,
|
|
169
|
+
timestamp: change.timestamp,
|
|
170
|
+
});
|
|
171
|
+
});
|
|
172
|
+
// POST /api/sessions/:id/changes/:changeId/revert — revert a single change
|
|
173
|
+
changesRouter.post('/api/sessions/:id/changes/:changeId/revert', async (req, res) => {
|
|
174
|
+
const change = fsMonitor.findChange(req.params.changeId);
|
|
175
|
+
if (!change)
|
|
176
|
+
return res.status(404).json({ error: 'Change not found' });
|
|
177
|
+
const result = await fsMonitor.revertChange(change);
|
|
178
|
+
res.json(result);
|
|
179
|
+
});
|
|
180
|
+
// POST /api/sessions/:id/changes/revert-all — revert all changes for a session
|
|
181
|
+
changesRouter.post('/api/sessions/:id/changes/revert-all', async (req, res) => {
|
|
182
|
+
const sessionId = req.params.id;
|
|
183
|
+
const session = store.sessions.get(sessionId);
|
|
184
|
+
if (!session)
|
|
185
|
+
return res.status(404).json({ error: 'Session not found' });
|
|
186
|
+
const result = await fsMonitor.revertAll(sessionId);
|
|
187
|
+
res.json(result);
|
|
188
|
+
});
|
|
189
|
+
// GET /api/objects/:hash — download raw file content from object store
|
|
190
|
+
changesRouter.get('/api/objects/{*hash}', async (req, res) => {
|
|
191
|
+
const hash = Array.isArray(req.params.hash) ? req.params.hash.join('/') : (req.params.hash ?? '');
|
|
192
|
+
if (!hash)
|
|
193
|
+
return res.status(400).json({ error: 'Hash required' });
|
|
194
|
+
const content = await fsMonitor.getFileContent(hash);
|
|
195
|
+
if (!content) {
|
|
196
|
+
return res.status(404).json({ error: 'Object not found' });
|
|
197
|
+
}
|
|
198
|
+
const buf = Buffer.from(content, 'base64');
|
|
199
|
+
res.setHeader('Content-Type', 'application/octet-stream');
|
|
200
|
+
res.setHeader('Content-Disposition', 'attachment');
|
|
201
|
+
res.setHeader('Content-Length', buf.length);
|
|
202
|
+
res.send(buf);
|
|
203
|
+
});
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
export declare const containersRouter: import("express-serve-static-core").Router;
|
|
2
|
+
interface RuntimeStatus {
|
|
3
|
+
installed: boolean;
|
|
4
|
+
accessible: boolean;
|
|
5
|
+
error?: string;
|
|
6
|
+
}
|
|
7
|
+
export declare function getContainerRuntimes(): {
|
|
8
|
+
docker: RuntimeStatus;
|
|
9
|
+
podman: RuntimeStatus;
|
|
10
|
+
lxc: RuntimeStatus;
|
|
11
|
+
};
|
|
12
|
+
export {};
|