tmuxes 0.1.9 → 0.1.11

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 (142) hide show
  1. package/LICENSE +21 -21
  2. package/README.md +283 -296
  3. package/{server/bin → bin}/tmuxes.js +36 -36
  4. package/dist/agentHooks.js +91 -0
  5. package/dist/agentHooks.js.map +1 -0
  6. package/dist/agentOutput.js +30 -0
  7. package/dist/agentOutput.js.map +1 -0
  8. package/dist/agentState.js +45 -0
  9. package/dist/agentState.js.map +1 -0
  10. package/dist/config.js +32 -0
  11. package/dist/config.js.map +1 -0
  12. package/dist/exe.js +37 -0
  13. package/dist/exe.js.map +1 -0
  14. package/dist/exec.js +43 -0
  15. package/dist/exec.js.map +1 -0
  16. package/dist/files.js +308 -0
  17. package/dist/files.js.map +1 -0
  18. package/dist/foldersStore.js +103 -0
  19. package/dist/foldersStore.js.map +1 -0
  20. package/dist/index.js +117 -0
  21. package/dist/index.js.map +1 -0
  22. package/dist/logger.js +16 -0
  23. package/dist/logger.js.map +1 -0
  24. package/{server/src/monitor.ts → dist/monitor.js} +9 -10
  25. package/dist/monitor.js.map +1 -0
  26. package/dist/openBrowser.js +31 -0
  27. package/dist/openBrowser.js.map +1 -0
  28. package/{server/src/platform.ts → dist/platform.js} +5 -4
  29. package/dist/platform.js.map +1 -0
  30. package/dist/rest/router.js +198 -0
  31. package/dist/rest/router.js.map +1 -0
  32. package/dist/targetCommand.js +60 -0
  33. package/dist/targetCommand.js.map +1 -0
  34. package/dist/targets.js +131 -0
  35. package/dist/targets.js.map +1 -0
  36. package/dist/tmux/builder.js +174 -0
  37. package/dist/tmux/builder.js.map +1 -0
  38. package/dist/tmux/formats.js +61 -0
  39. package/dist/tmux/formats.js.map +1 -0
  40. package/dist/tmux/sessions.js +157 -0
  41. package/dist/tmux/sessions.js.map +1 -0
  42. package/dist/validate.js +65 -0
  43. package/dist/validate.js.map +1 -0
  44. package/dist/windowsSsh.js +209 -0
  45. package/dist/windowsSsh.js.map +1 -0
  46. package/dist/winshell/manager.js +267 -0
  47. package/dist/winshell/manager.js.map +1 -0
  48. package/dist/ws/protocol.js +4 -0
  49. package/dist/ws/protocol.js.map +1 -0
  50. package/dist/ws/sshState.js +35 -0
  51. package/dist/ws/sshState.js.map +1 -0
  52. package/dist/ws/terminalSession.js +204 -0
  53. package/dist/ws/terminalSession.js.map +1 -0
  54. package/dist/ws/wsServer.js +151 -0
  55. package/dist/ws/wsServer.js.map +1 -0
  56. package/dist/wsl.js +35 -0
  57. package/dist/wsl.js.map +1 -0
  58. package/package.json +61 -28
  59. package/public/assets/index-D_X5SnGx.css +1 -0
  60. package/public/assets/index-Dl69CPyt.js +44 -0
  61. package/{client → public}/index.html +13 -12
  62. package/.node-version +0 -1
  63. package/.nvmrc +0 -1
  64. package/.tmp-npm-cache/_cacache/content-v2/sha512/43/27/5e000b8b9c56a6ccc66f709485499f4304e2cb1982582ba571321c07b3ef56fcabd2c671898cc8003365a0485b6fd8e73e7b17b073cec0f7d1628c1a99df +0 -0
  65. package/.tmp-npm-cache/_cacache/content-v2/sha512/51/cf/4301295d74559ed494bae160d54d8741077f89faebb311882ac065019246951e7b53f3dcb913793c42b331e14c7070c4810c3cdc27a427d103a7db4614e0 +0 -0
  66. package/.tmp-npm-cache/_cacache/content-v2/sha512/c3/4d/d68a454a916e74c2617f586fbf770981b33811d667c2547eb0e9fc21938f4ee7e98f1ceee4bde8ad8815b5f6efe21b60eee798837d68f51a3340d7e5bb7a +0 -0
  67. package/.tmp-npm-cache/_cacache/content-v2/sha512/fe/40/2abfbefc96299e8bf714aa91d62607190ae299e102cf5933db2e2904640d65d25d67dbbb6fa2ddc92a17f00b9dbfdf2e37487f67d96ec36c64a285b59a7d +0 -0
  68. package/.tmp-npm-cache/_cacache/index-v5/27/fe/81a3de6ce7ae3d1e41a3421de20c5629998c4ee5d0ffe2037630f03b03b2 +0 -4
  69. package/.tmp-npm-cache/_cacache/index-v5/65/22/dd66711f62681fce09aabb2357a2907b4a0c778ac5227c4baf9603fd86e8 +0 -4
  70. package/.tmp-npm-cache/_update-notifier-last-checked +0 -0
  71. package/AGENTS.md +0 -15
  72. package/CLAUDE.md +0 -3
  73. package/README.en.md +0 -304
  74. package/SECURITY.md +0 -31
  75. package/client/package.json +0 -29
  76. package/client/src/App.tsx +0 -123
  77. package/client/src/activity.ts +0 -5
  78. package/client/src/api.ts +0 -130
  79. package/client/src/attention.tsx +0 -157
  80. package/client/src/components/FileExplorer.tsx +0 -156
  81. package/client/src/components/FileViewer.tsx +0 -194
  82. package/client/src/components/SessionRow.tsx +0 -108
  83. package/client/src/components/SessionTree.tsx +0 -197
  84. package/client/src/components/SettingsButton.tsx +0 -122
  85. package/client/src/components/Sidebar.tsx +0 -96
  86. package/client/src/components/StatusBanner.tsx +0 -31
  87. package/client/src/components/TargetGroup.tsx +0 -275
  88. package/client/src/components/TerminalPanel.tsx +0 -192
  89. package/client/src/folders.ts +0 -245
  90. package/client/src/hooks/useTerminal.ts +0 -67
  91. package/client/src/hooks/useTmuxSocket.ts +0 -65
  92. package/client/src/i18n.ts +0 -213
  93. package/client/src/main.tsx +0 -17
  94. package/client/src/settings.tsx +0 -87
  95. package/client/src/styles.css +0 -723
  96. package/client/src/types.ts +0 -93
  97. package/client/src/util.ts +0 -65
  98. package/client/tsconfig.json +0 -13
  99. package/client/vite.config.ts +0 -15
  100. package/fig/fig1.png +0 -0
  101. package/scripts/prepack.mjs +0 -35
  102. package/server/package.json +0 -61
  103. package/server/src/agentHooks.ts +0 -120
  104. package/server/src/agentOutput.ts +0 -36
  105. package/server/src/agentState.ts +0 -70
  106. package/server/src/config.ts +0 -31
  107. package/server/src/exe.ts +0 -34
  108. package/server/src/exec.ts +0 -61
  109. package/server/src/files.ts +0 -330
  110. package/server/src/foldersStore.ts +0 -114
  111. package/server/src/index.ts +0 -114
  112. package/server/src/logger.ts +0 -16
  113. package/server/src/openBrowser.ts +0 -28
  114. package/server/src/rest/router.ts +0 -290
  115. package/server/src/targetCommand.ts +0 -79
  116. package/server/src/targets.ts +0 -152
  117. package/server/src/tmux/builder.ts +0 -198
  118. package/server/src/tmux/formats.ts +0 -95
  119. package/server/src/tmux/sessions.ts +0 -204
  120. package/server/src/validate.ts +0 -79
  121. package/server/src/windowsSsh.ts +0 -239
  122. package/server/src/winshell/manager.ts +0 -296
  123. package/server/src/ws/protocol.ts +0 -15
  124. package/server/src/ws/sshState.ts +0 -36
  125. package/server/src/ws/terminalSession.ts +0 -207
  126. package/server/src/ws/wsServer.ts +0 -153
  127. package/server/src/wsl.ts +0 -38
  128. package/server/test/agentHooks.test.ts +0 -66
  129. package/server/test/agentOutput.test.ts +0 -26
  130. package/server/test/agentState.test.ts +0 -24
  131. package/server/test/builder.test.ts +0 -162
  132. package/server/test/files.test.ts +0 -81
  133. package/server/test/formats.test.ts +0 -123
  134. package/server/test/monitor.test.ts +0 -25
  135. package/server/test/validate.test.ts +0 -71
  136. package/server/test/wsl.test.ts +0 -18
  137. package/server/tsconfig.json +0 -9
  138. package/server/vitest.config.ts +0 -12
  139. package/start.cmd +0 -30
  140. package/start.command +0 -20
  141. package/start.sh +0 -20
  142. package/tsconfig.base.json +0 -19
@@ -0,0 +1,151 @@
1
+ import { WebSocketServer } from 'ws';
2
+ import { config } from '../config.js';
3
+ import { getTarget, isValidTargetId } from '../targets.js';
4
+ import { isValidSessionName, isValidDimension } from '../validate.js';
5
+ import { TerminalSession, track } from './terminalSession.js';
6
+ import { winShell } from '../winshell/manager.js';
7
+ import { log } from '../logger.js';
8
+ const HEARTBEAT_MS = 30_000;
9
+ function reject(socket, status, message) {
10
+ socket.write(`HTTP/1.1 ${status} ${message}\r\nConnection: close\r\n\r\n`);
11
+ socket.destroy();
12
+ }
13
+ function rawToString(data) {
14
+ if (typeof data === 'string')
15
+ return data;
16
+ if (Buffer.isBuffer(data))
17
+ return data.toString('utf8');
18
+ if (Array.isArray(data))
19
+ return Buffer.concat(data).toString('utf8');
20
+ if (data instanceof ArrayBuffer)
21
+ return Buffer.from(data).toString('utf8');
22
+ return String(data);
23
+ }
24
+ let nextClientId = 1;
25
+ /** Attach a WS to a native shell session (one persistent pty, many clients). */
26
+ function attachWinShell(ws, target, session, cols, rows) {
27
+ const client = {
28
+ id: nextClientId++,
29
+ sendBinary: (buf) => {
30
+ if (ws.readyState === ws.OPEN)
31
+ ws.send(buf, { binary: true });
32
+ },
33
+ sendControl: (msg) => {
34
+ if (ws.readyState === ws.OPEN)
35
+ ws.send(JSON.stringify(msg), { binary: false });
36
+ },
37
+ isOpen: () => ws.readyState === ws.OPEN,
38
+ close: (code, reason) => {
39
+ try {
40
+ ws.close(code, reason);
41
+ }
42
+ catch {
43
+ /* ignore */
44
+ }
45
+ },
46
+ };
47
+ let shellSession;
48
+ try {
49
+ shellSession = winShell.attachOrCreate(session, cols, rows, client);
50
+ }
51
+ catch (e) {
52
+ client.sendControl({ type: 'error', message: e instanceof Error ? e.message : 'failed to start shell' });
53
+ client.close(1011, 'shell error');
54
+ return;
55
+ }
56
+ client.sendControl({ type: 'ready', target: target.id, session });
57
+ ws.on('message', (data, isBinary) => {
58
+ if (isBinary) {
59
+ shellSession.write(rawToString(data));
60
+ return;
61
+ }
62
+ let msg;
63
+ try {
64
+ msg = JSON.parse(rawToString(data));
65
+ }
66
+ catch {
67
+ return;
68
+ }
69
+ if (msg.type === 'resize' && isValidDimension(msg.cols) && isValidDimension(msg.rows)) {
70
+ shellSession.resize(msg.cols, msg.rows);
71
+ }
72
+ else if (msg.type === 'ping') {
73
+ client.sendControl({ type: 'pong' });
74
+ }
75
+ });
76
+ let alive = true;
77
+ ws.on('pong', () => {
78
+ alive = true;
79
+ });
80
+ const hb = setInterval(() => {
81
+ if (!alive) {
82
+ try {
83
+ ws.terminate();
84
+ }
85
+ catch {
86
+ /* ignore */
87
+ }
88
+ return;
89
+ }
90
+ alive = false;
91
+ try {
92
+ ws.ping();
93
+ }
94
+ catch {
95
+ /* ignore */
96
+ }
97
+ }, HEARTBEAT_MS);
98
+ const cleanup = () => {
99
+ clearInterval(hb);
100
+ shellSession.detach(client); // pty stays alive — persistence across reconnects
101
+ };
102
+ ws.on('close', cleanup);
103
+ ws.on('error', cleanup);
104
+ }
105
+ /** Attach the single /ws interactive-attach endpoint to the HTTP server. */
106
+ export function attachWebSocket(server) {
107
+ const wss = new WebSocketServer({ noServer: true });
108
+ server.on('upgrade', (req, socket, head) => {
109
+ let url;
110
+ try {
111
+ url = new URL(req.url ?? '', `http://${req.headers.host ?? 'localhost'}`);
112
+ }
113
+ catch {
114
+ reject(socket, 400, 'Bad Request');
115
+ return;
116
+ }
117
+ if (url.pathname !== '/ws') {
118
+ reject(socket, 404, 'Not Found');
119
+ return;
120
+ }
121
+ // The WS upgrade bypasses Express middleware — enforce Origin here.
122
+ if (!config.isAllowedOrigin(req.headers.origin)) {
123
+ log.warn(`rejected WS upgrade from disallowed origin: ${req.headers.origin}`);
124
+ reject(socket, 403, 'Forbidden');
125
+ return;
126
+ }
127
+ const targetId = url.searchParams.get('target') ?? '';
128
+ const session = url.searchParams.get('session') ?? '';
129
+ if (!isValidTargetId(targetId))
130
+ return reject(socket, 400, 'Bad Request');
131
+ const target = getTarget(targetId);
132
+ if (!target)
133
+ return reject(socket, 404, 'Not Found');
134
+ if (!isValidSessionName(session))
135
+ return reject(socket, 400, 'Bad Request');
136
+ const colsRaw = Number(url.searchParams.get('cols'));
137
+ const rowsRaw = Number(url.searchParams.get('rows'));
138
+ const cols = isValidDimension(colsRaw) ? colsRaw : 80;
139
+ const rows = isValidDimension(rowsRaw) ? rowsRaw : 24;
140
+ wss.handleUpgrade(req, socket, head, (ws) => {
141
+ log.info(`attach ${target.id}/${session} (${cols}x${rows})`);
142
+ if (target.kind === 'winlocal') {
143
+ attachWinShell(ws, target, session, cols, rows);
144
+ return;
145
+ }
146
+ const ts = new TerminalSession(ws, target, session, cols, rows);
147
+ track(ts);
148
+ });
149
+ });
150
+ }
151
+ //# sourceMappingURL=wsServer.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"wsServer.js","sourceRoot":"","sources":["../../src/ws/wsServer.ts"],"names":[],"mappings":"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"}
package/dist/wsl.js ADDED
@@ -0,0 +1,35 @@
1
+ import { runCommand } from './exec.js';
2
+ import { log } from './logger.js';
3
+ /** Distros that exist for the container runtime, never for interactive use. */
4
+ const SYSTEM_DISTROS = /^docker-desktop(-data)?$/i;
5
+ const NUL = 0;
6
+ const BOM = 0xfeff;
7
+ /**
8
+ * Enumerate installed WSL distros (Windows only). `wsl.exe -l -q` prints names
9
+ * one per line in UTF-16LE (with a BOM), so we decode accordingly.
10
+ */
11
+ export async function listWslDistros() {
12
+ const r = await runCommand('wsl.exe', ['-l', '-q'], { encoding: 'utf16le', timeoutMs: 8000 });
13
+ if (r.code !== 0) {
14
+ if (r.stdout.trim() || r.stderr.trim()) {
15
+ log.warn(`wsl.exe -l -q failed: ${(r.stderr || r.stdout).trim().split('\n')[0]}`);
16
+ }
17
+ return [];
18
+ }
19
+ return parseWslList(r.stdout);
20
+ }
21
+ /** Parse `wsl.exe -l -q` decoded output into clean distro names. */
22
+ export function parseWslList(stdout) {
23
+ // Drop NULs / BOM via code point so the source stays pure ASCII.
24
+ const cleaned = Array.from(stdout)
25
+ .filter((ch) => {
26
+ const c = ch.charCodeAt(0);
27
+ return c !== NUL && c !== BOM;
28
+ })
29
+ .join('');
30
+ return cleaned
31
+ .split(/\r?\n/)
32
+ .map((line) => line.trim())
33
+ .filter((name) => name.length > 0 && !SYSTEM_DISTROS.test(name));
34
+ }
35
+ //# sourceMappingURL=wsl.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"wsl.js","sourceRoot":"","sources":["../src/wsl.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,WAAW,CAAC;AACvC,OAAO,EAAE,GAAG,EAAE,MAAM,aAAa,CAAC;AAElC,+EAA+E;AAC/E,MAAM,cAAc,GAAG,2BAA2B,CAAC;AAEnD,MAAM,GAAG,GAAG,CAAC,CAAC;AACd,MAAM,GAAG,GAAG,MAAM,CAAC;AAEnB;;;GAGG;AACH,MAAM,CAAC,KAAK,UAAU,cAAc;IAClC,MAAM,CAAC,GAAG,MAAM,UAAU,CAAC,SAAS,EAAE,CAAC,IAAI,EAAE,IAAI,CAAC,EAAE,EAAE,QAAQ,EAAE,SAAS,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;IAC9F,IAAI,CAAC,CAAC,IAAI,KAAK,CAAC,EAAE,CAAC;QACjB,IAAI,CAAC,CAAC,MAAM,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC,MAAM,CAAC,IAAI,EAAE,EAAE,CAAC;YACvC,GAAG,CAAC,IAAI,CAAC,yBAAyB,CAAC,CAAC,CAAC,MAAM,IAAI,CAAC,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;QACpF,CAAC;QACD,OAAO,EAAE,CAAC;IACZ,CAAC;IACD,OAAO,YAAY,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC;AAChC,CAAC;AAED,oEAAoE;AACpE,MAAM,UAAU,YAAY,CAAC,MAAc;IACzC,iEAAiE;IACjE,MAAM,OAAO,GAAG,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC;SAC/B,MAAM,CAAC,CAAC,EAAE,EAAE,EAAE;QACb,MAAM,CAAC,GAAG,EAAE,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC;QAC3B,OAAO,CAAC,KAAK,GAAG,IAAI,CAAC,KAAK,GAAG,CAAC;IAChC,CAAC,CAAC;SACD,IAAI,CAAC,EAAE,CAAC,CAAC;IACZ,OAAO,OAAO;SACX,KAAK,CAAC,OAAO,CAAC;SACd,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC;SAC1B,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,MAAM,GAAG,CAAC,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC;AACrE,CAAC"}
package/package.json CHANGED
@@ -1,28 +1,61 @@
1
- {
2
- "name": "tmuxes",
3
- "version": "0.1.9",
4
- "private": false,
5
- "license": "MIT",
6
- "description": "Dev monorepo for tmuxes — the publishable package lives in server/ (name: tmuxes).",
7
- "packageManager": "npm@10.9.7",
8
- "engines": {
9
- "node": ">=22.12.0 <23"
10
- },
11
- "workspaces": [
12
- "server",
13
- "client"
14
- ],
15
- "scripts": {
16
- "dev": "concurrently -n server,client -c blue,green \"npm:dev:server\" \"npm:dev:client\"",
17
- "dev:server": "npm --workspace server run dev",
18
- "dev:client": "npm --workspace client run dev",
19
- "build": "npm --workspace client run build && npm --workspace server run build",
20
- "start": "npm --workspace server run start",
21
- "test": "npm --workspace server run test",
22
- "pack:check": "npm pack --workspace server --dry-run",
23
- "release": "npm publish --workspace server"
24
- },
25
- "devDependencies": {
26
- "concurrently": "^9.1.0"
27
- }
28
- }
1
+ {
2
+ "name": "tmuxes",
3
+ "version": "0.1.11",
4
+ "description": "Web UI to run and supervise many CLI coding agents in tmux — local, SSH, and WSL.",
5
+ "keywords": [
6
+ "tmux",
7
+ "terminal",
8
+ "xterm",
9
+ "ssh",
10
+ "wsl",
11
+ "web-terminal",
12
+ "node-pty",
13
+ "cli-agents"
14
+ ],
15
+ "license": "MIT",
16
+ "author": "f1974939505 (https://github.com/f1974939505)",
17
+ "homepage": "https://github.com/f1974939505/tmuxes#readme",
18
+ "repository": {
19
+ "type": "git",
20
+ "url": "git+https://github.com/f1974939505/tmuxes.git",
21
+ "directory": "server"
22
+ },
23
+ "bugs": {
24
+ "url": "https://github.com/f1974939505/tmuxes/issues"
25
+ },
26
+ "type": "module",
27
+ "main": "dist/index.js",
28
+ "bin": {
29
+ "tmuxes": "bin/tmuxes.js"
30
+ },
31
+ "engines": {
32
+ "node": ">=22.12.0 <23"
33
+ },
34
+ "files": [
35
+ "dist",
36
+ "public",
37
+ "bin",
38
+ "README.md",
39
+ "LICENSE"
40
+ ],
41
+ "scripts": {
42
+ "dev": "tsx watch src/index.ts",
43
+ "build": "tsc -p tsconfig.json",
44
+ "start": "node dist/index.js",
45
+ "test": "vitest run",
46
+ "prepack": "node ../scripts/prepack.mjs"
47
+ },
48
+ "dependencies": {
49
+ "express": "^5.2.1",
50
+ "node-pty": "1.1.0",
51
+ "ws": "^8.21.0"
52
+ },
53
+ "devDependencies": {
54
+ "@types/express": "^5.0.0",
55
+ "@types/node": "^22.10.0",
56
+ "@types/ws": "^8.5.13",
57
+ "tsx": "^4.19.2",
58
+ "typescript": "^5.7.2",
59
+ "vitest": "^2.1.8"
60
+ }
61
+ }
@@ -0,0 +1 @@
1
+ .xterm{cursor:text;-webkit-user-select:none;user-select:none;position:relative}.xterm.focus,.xterm:focus{outline:none}.xterm .xterm-helpers{z-index:5;position:absolute;top:0}.xterm .xterm-helper-textarea{opacity:0;z-index:-5;white-space:nowrap;resize:none;border:0;width:0;height:0;margin:0;padding:0;position:absolute;top:0;left:-9999em;overflow:hidden}.xterm .composition-view{color:#fff;white-space:nowrap;z-index:1;background:#000;display:none;position:absolute}.xterm .composition-view.active{display:block}.xterm .xterm-viewport{cursor:default;background-color:#000;position:absolute;inset:0;overflow-y:scroll}.xterm .xterm-screen{position:relative}.xterm .xterm-screen canvas{position:absolute;top:0;left:0}.xterm-char-measure-element{visibility:hidden;line-height:normal;display:inline-block;position:absolute;top:0;left:-9999em}.xterm.enable-mouse-events{cursor:default}.xterm.xterm-cursor-pointer,.xterm .xterm-cursor-pointer{cursor:pointer}.xterm.column-select.focus{cursor:crosshair}.xterm .xterm-accessibility:not(.debug),.xterm .xterm-message{z-index:10;color:#0000;pointer-events:none;position:absolute;inset:0}.xterm .xterm-accessibility-tree:not(.debug) ::selection{color:#0000}.xterm .xterm-accessibility-tree{-webkit-user-select:text;user-select:text;white-space:pre;font-family:monospace}.xterm .xterm-accessibility-tree>div{transform-origin:0;width:fit-content}.xterm .live-region{width:1px;height:1px;position:absolute;left:-9999px;overflow:hidden}.xterm-dim{opacity:1!important}.xterm-underline-1{text-decoration:underline}.xterm-underline-2{-webkit-text-decoration:underline double;text-decoration:underline double}.xterm-underline-3{-webkit-text-decoration:underline wavy;text-decoration:underline wavy}.xterm-underline-4{-webkit-text-decoration:underline dotted;text-decoration:underline dotted}.xterm-underline-5{-webkit-text-decoration:underline dashed;text-decoration:underline dashed}.xterm-overline{text-decoration:overline}.xterm-overline.xterm-underline-1{text-decoration:underline overline}.xterm-overline.xterm-underline-2{-webkit-text-decoration:overline double underline;text-decoration:overline double underline}.xterm-overline.xterm-underline-3{-webkit-text-decoration:overline wavy underline;text-decoration:overline wavy underline}.xterm-overline.xterm-underline-4{-webkit-text-decoration:overline dotted underline;text-decoration:overline dotted underline}.xterm-overline.xterm-underline-5{-webkit-text-decoration:overline dashed underline;text-decoration:overline dashed underline}.xterm-strikethrough{text-decoration:line-through}.xterm-screen .xterm-decoration-container .xterm-decoration{z-index:6;position:absolute}.xterm-screen .xterm-decoration-container .xterm-decoration.xterm-decoration-top-layer{z-index:7}.xterm-decoration-overview-ruler{z-index:8;pointer-events:none;position:absolute;top:0;right:0}.xterm-decoration-top{z-index:2;position:relative}.xterm .xterm-scrollable-element>.scrollbar{cursor:default}.xterm .xterm-scrollable-element>.scrollbar>.scra{cursor:pointer;font-size:11px!important}.xterm .xterm-scrollable-element>.visible{opacity:1;z-index:11;background:0 0;transition:opacity .1s linear}.xterm .xterm-scrollable-element>.invisible{opacity:0;pointer-events:none}.xterm .xterm-scrollable-element>.invisible.fade{transition:opacity .8s linear}.xterm .xterm-scrollable-element>.shadow{display:none;position:absolute}.xterm .xterm-scrollable-element>.shadow.top{width:100%;height:3px;box-shadow:var(--vscode-scrollbar-shadow,#000) 0 6px 6px -6px inset;display:block;top:0;left:3px}.xterm .xterm-scrollable-element>.shadow.left{width:3px;height:100%;box-shadow:var(--vscode-scrollbar-shadow,#000) 6px 0 6px -6px inset;display:block;top:3px;left:0}.xterm .xterm-scrollable-element>.shadow.top-left-corner{width:3px;height:3px;display:block;top:0;left:0}.xterm .xterm-scrollable-element>.shadow.top.left{box-shadow:var(--vscode-scrollbar-shadow,#000) 6px 0 6px -6px inset}:root{--bg:#1a1b26;--bg-alt:#16161e;--panel:#1f2233;--border:#2a2e42;--fg:#c0caf5;--fg-dim:#7a82a8;--accent:#7aa2f7;--green:#9ece6a;--red:#f7768e;--yellow:#e0af68;font-family:-apple-system,BlinkMacSystemFont,Segoe UI,Roboto,sans-serif}*{box-sizing:border-box}html,body,#root{height:100%;margin:0;overflow:hidden}body{background:var(--bg);color:var(--fg);font-size:14px}button{font:inherit;color:var(--fg);background:var(--panel);border:1px solid var(--border);cursor:pointer;border-radius:5px;padding:3px 8px}button:hover:not(:disabled){border-color:var(--accent)}button:disabled{opacity:.5;cursor:default}button.primary{background:var(--accent);color:#11121a;border-color:var(--accent);font-weight:600}button.danger:hover:not(:disabled){border-color:var(--red);color:var(--red)}input{font:inherit;color:var(--fg);background:var(--bg-alt);border:1px solid var(--border);border-radius:5px;padding:4px 7px}input:focus{border-color:var(--accent);outline:none}.app{height:100%;display:flex;overflow:hidden}.sidebar{background:var(--bg-alt);border-right:1px solid var(--border);flex-direction:column;width:320px;min-width:320px;display:flex;overflow:hidden}.sidebar-header{border-bottom:1px solid var(--border);justify-content:space-between;align-items:center;padding:12px 14px;display:flex}.sidebar-header h1{letter-spacing:.5px;margin:0;font-size:15px}.sidebar-top{flex:1;min-height:80px;padding:6px;overflow-y:auto}.sidebar-vdivider{cursor:row-resize;background:var(--border);flex:none;height:6px}.sidebar-vdivider:hover{background:var(--accent)}.sidebar-bottom{border-top:1px solid var(--border);flex-direction:column;flex:none;display:flex;overflow:hidden}.sidebar-footer{border-top:1px solid var(--border);flex:none;padding:6px 8px}.section-label{text-transform:uppercase;letter-spacing:.5px;color:var(--fg-dim);flex:none;padding:5px 8px 3px;font-size:11px}.workspace{flex-direction:column;flex:1;min-width:0;display:flex;overflow:hidden}.term-region{flex:1;min-height:0;display:flex;position:relative}.hdivider{cursor:row-resize;background:var(--border);flex:none;height:6px}.hdivider:hover{background:var(--accent)}.viewer-region{background:var(--bg-alt);flex-direction:column;flex:none;display:flex;overflow:hidden}.panel{background:#000;flex:1;min-width:0;position:relative;overflow:hidden}.target-group{margin-bottom:6px}.target-head{cursor:pointer;-webkit-user-select:none;user-select:none;border-radius:6px;align-items:center;gap:6px;padding:6px 8px;display:flex}.target-head:hover{background:var(--panel)}.target-head .label{text-overflow:ellipsis;white-space:nowrap;flex:1;font-weight:600;overflow:hidden}.target-head .kind{color:var(--fg-dim);border:1px solid var(--border);border-radius:4px;padding:0 5px;font-size:11px}.caret{width:12px;color:var(--fg-dim)}.badge{border-radius:4px;padding:0 5px;font-size:11px}.badge.err{color:var(--red)}.badge.loading{color:var(--yellow)}.session-list{padding:2px 0 6px 14px}.empty{color:var(--fg-dim);padding:4px 8px;font-size:12px}.error-line{color:var(--red);align-items:center;gap:8px;padding:4px 8px;font-size:12px;display:flex}.error-line span{text-overflow:ellipsis;flex:1;min-width:0;overflow:hidden}.error-line button{flex:none;padding:2px 6px;font-size:12px}.session-row{cursor:pointer;border-radius:6px;align-items:center;gap:6px;padding:5px 8px;display:flex}.session-row:hover{background:var(--panel)}.session-row.selected{background:var(--panel);box-shadow:inset 2px 0 0 var(--accent)}.session-row .name{text-overflow:ellipsis;white-space:nowrap;flex:1;overflow:hidden}.dot{border-radius:50%;flex:none;width:8px;height:8px}.dot.inactive{background:var(--green)}.dot.active{background:var(--red);box-shadow:0 0 0 2px #f7768e3d}.session-row.attention .name{font-weight:600}.attn-badge{border:1px solid var(--border);white-space:nowrap;border-radius:5px;flex:none;padding:1px 5px;font-size:11px;line-height:1.35}.attn-badge.decision{border-color:var(--red);color:var(--red)}.attn-badge.done{border-color:var(--green);color:var(--green)}.attn-badge.error{border-color:var(--red);color:var(--red)}.meta{color:var(--fg-dim);font-size:11px}.row-actions{opacity:0;gap:4px;display:flex}.session-row:hover .row-actions{opacity:1}.row-actions button{padding:1px 6px;font-size:12px}.create-form{flex-direction:column;gap:5px;padding:6px 8px;display:flex}.create-form .row{gap:5px;display:flex}.create-form input{flex:1;min-width:0}.create-form select{font:inherit;color:var(--fg);background:var(--bg-alt);border:1px solid var(--border);border-radius:5px;padding:4px 7px}.field-error{color:var(--red);font-size:11px}.agent-toolbar{z-index:4;border:1px solid var(--border);-webkit-backdrop-filter:blur(4px);backdrop-filter:blur(4px);background:#111827e0;border-radius:6px;align-items:center;gap:5px;padding:3px;display:flex;position:absolute;top:8px;right:10px}.agent-toolbar button{min-width:0;height:22px;padding:2px 7px;font-size:12px;line-height:1}.agent-error{color:var(--red);padding:0 4px;font-size:12px}.term-host{padding:4px;position:absolute;inset:0;overflow:hidden}.term-host .xterm{width:100%;height:100%}.panel-placeholder{color:var(--fg-dim);flex-direction:column;justify-content:center;align-items:center;gap:6px;display:flex;position:absolute;inset:0}.status-banner{background:var(--panel);border:1px solid var(--border);z-index:5;border-radius:8px;align-items:center;gap:12px;max-width:80%;padding:8px 14px;display:flex;position:absolute;top:12px;left:50%;transform:translate(-50%);box-shadow:0 6px 24px #0006}.status-banner.error{border-color:var(--red)}.status-banner .msg{font-size:13px}.spinner{border:2px solid var(--border);border-top-color:var(--accent);border-radius:50%;width:14px;height:14px;animation:.8s linear infinite spin}@keyframes spin{to{transform:rotate(360deg)}}.tree.drop-root{outline:1px dashed var(--accent);outline-offset:-2px;border-radius:6px}.folder.drop{background:#7aa2f71f;border-radius:6px}.folder-head{cursor:pointer;-webkit-user-select:none;user-select:none;border-radius:6px;align-items:center;gap:4px;padding:4px 8px;display:flex}.folder-head:hover{background:var(--panel)}.folder-name{text-overflow:ellipsis;white-space:nowrap;flex:1;overflow:hidden}.folder-actions{opacity:0;gap:3px;display:flex}.folder-head:hover .folder-actions{opacity:1}.folder-actions button{padding:0 5px;font-size:12px}.folder-head input{flex:1;min-width:0}.list-toolbar{gap:6px;padding:6px 8px 2px;display:flex}.list-toolbar button{padding:2px 8px;font-size:12px}.explorer{flex-direction:column;flex:1;display:flex;overflow:hidden}.explorer-empty{color:var(--fg-dim);padding:8px;font-size:12px}.explorer-head{flex:none;align-items:center;gap:6px;padding:2px 8px 4px;display:flex}.explorer-path{text-overflow:ellipsis;white-space:nowrap;color:var(--accent);flex:1;font-weight:600;overflow:hidden}.explorer-actions{gap:4px;display:flex}.explorer-actions button{padding:1px 7px}.explorer-list{flex:1;overflow-y:auto}.file-row{cursor:pointer;border-radius:5px;align-items:center;gap:6px;padding:3px 8px;display:flex}.file-row:hover{background:var(--panel)}.file-row.disabled{cursor:default;color:var(--fg-dim)}.file-row.disabled:hover{background:0 0}.file-row.open{background:var(--panel);box-shadow:inset 2px 0 0 var(--accent)}.file-icon{flex:none}.file-name{text-overflow:ellipsis;white-space:nowrap;overflow:hidden}.settings{position:relative}.settings-gear{text-align:left;width:100%}.settings-backdrop{z-index:19;position:fixed;inset:0}.settings-panel{background:var(--panel);border:1px solid var(--border);z-index:20;border-radius:8px;width:250px;padding:10px;position:absolute;bottom:40px;left:0;box-shadow:0 8px 28px #00000080}.settings-title{color:var(--fg-dim);margin-bottom:8px;font-size:12px}.stepper{justify-content:space-between;align-items:center;margin-bottom:6px;display:flex}.stepper-label{font-size:13px}.stepper-controls{align-items:center;gap:6px;display:flex}.stepper-value{text-align:center;font-variant-numeric:tabular-nums;min-width:36px;font-size:12px}.toggle{cursor:pointer;align-items:center;gap:8px;margin-bottom:6px;display:flex}.toggle input{cursor:pointer}.toggle-label{font-size:13px}.settings-panel .settings-title+.stepper,.settings-panel .settings-title:not(:first-child){margin-top:2px}.settings-panel .settings-title:not(:first-of-type){margin-top:12px}.settings-actions{justify-content:flex-end;gap:6px;margin-top:10px;display:flex}.viewer{flex-direction:column;height:100%;display:flex;overflow:hidden}.viewer-head{border-bottom:1px solid var(--border);flex:none;align-items:center;gap:8px;padding:5px 10px;display:flex}.viewer-name{font-weight:600}.viewer-path{color:var(--fg-dim);text-overflow:ellipsis;white-space:nowrap;font-size:12px;overflow:hidden}.viewer-head-spacer{flex:1}.viewer-note{color:var(--yellow);font-size:12px}.viewer-body{flex:1;overflow:auto}.viewer-pre{white-space:pre;tab-size:4;margin:0;padding:8px 10px;font-family:Menlo,Consolas,DejaVu Sans Mono,monospace;line-height:1.45}.viewer-editor{box-sizing:border-box;resize:none;width:100%;height:100%;color:var(--fg);tab-size:2;white-space:pre;background:0 0;border:0;outline:none;margin:0;padding:8px 10px;font-family:Menlo,Consolas,DejaVu Sans Mono,monospace;line-height:1.45;display:block;overflow:auto}.viewer-head button{padding:2px 8px;font-size:12px}.dirty-dot{color:var(--yellow);vertical-align:middle;margin-right:5px;font-size:11px}.viewer-msg{color:var(--fg-dim);padding:12px}.viewer-msg.error{color:var(--red)}