tmuxes 0.1.9 → 0.1.10
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/agentHooks.js +91 -0
- package/dist/agentHooks.js.map +1 -0
- package/dist/agentOutput.js +30 -0
- package/dist/agentOutput.js.map +1 -0
- package/dist/agentState.js +45 -0
- package/dist/agentState.js.map +1 -0
- package/dist/config.js +32 -0
- package/dist/config.js.map +1 -0
- package/dist/exe.js +37 -0
- package/dist/exe.js.map +1 -0
- package/dist/exec.js +43 -0
- package/dist/exec.js.map +1 -0
- package/dist/files.js +308 -0
- package/dist/files.js.map +1 -0
- package/dist/foldersStore.js +103 -0
- package/dist/foldersStore.js.map +1 -0
- package/dist/index.js +117 -0
- package/dist/index.js.map +1 -0
- package/dist/logger.js +16 -0
- package/dist/logger.js.map +1 -0
- package/{server/src/monitor.ts → dist/monitor.js} +9 -10
- package/dist/monitor.js.map +1 -0
- package/dist/openBrowser.js +31 -0
- package/dist/openBrowser.js.map +1 -0
- package/{server/src/platform.ts → dist/platform.js} +5 -4
- package/dist/platform.js.map +1 -0
- package/dist/rest/router.js +198 -0
- package/dist/rest/router.js.map +1 -0
- package/dist/targetCommand.js +60 -0
- package/dist/targetCommand.js.map +1 -0
- package/dist/targets.js +131 -0
- package/dist/targets.js.map +1 -0
- package/dist/tmux/builder.js +174 -0
- package/dist/tmux/builder.js.map +1 -0
- package/dist/tmux/formats.js +61 -0
- package/dist/tmux/formats.js.map +1 -0
- package/dist/tmux/sessions.js +157 -0
- package/dist/tmux/sessions.js.map +1 -0
- package/dist/validate.js +65 -0
- package/dist/validate.js.map +1 -0
- package/dist/windowsSsh.js +209 -0
- package/dist/windowsSsh.js.map +1 -0
- package/dist/winshell/manager.js +267 -0
- package/dist/winshell/manager.js.map +1 -0
- package/dist/ws/protocol.js +4 -0
- package/dist/ws/protocol.js.map +1 -0
- package/dist/ws/sshState.js +35 -0
- package/dist/ws/sshState.js.map +1 -0
- package/dist/ws/terminalSession.js +204 -0
- package/dist/ws/terminalSession.js.map +1 -0
- package/dist/ws/wsServer.js +151 -0
- package/dist/ws/wsServer.js.map +1 -0
- package/dist/wsl.js +35 -0
- package/dist/wsl.js.map +1 -0
- package/package.json +49 -16
- package/public/assets/index-D_X5SnGx.css +1 -0
- package/public/assets/index-Dl69CPyt.js +44 -0
- package/{client → public}/index.html +3 -2
- package/.node-version +0 -1
- package/.nvmrc +0 -1
- package/.tmp-npm-cache/_cacache/content-v2/sha512/43/27/5e000b8b9c56a6ccc66f709485499f4304e2cb1982582ba571321c07b3ef56fcabd2c671898cc8003365a0485b6fd8e73e7b17b073cec0f7d1628c1a99df +0 -0
- package/.tmp-npm-cache/_cacache/content-v2/sha512/51/cf/4301295d74559ed494bae160d54d8741077f89faebb311882ac065019246951e7b53f3dcb913793c42b331e14c7070c4810c3cdc27a427d103a7db4614e0 +0 -0
- package/.tmp-npm-cache/_cacache/content-v2/sha512/c3/4d/d68a454a916e74c2617f586fbf770981b33811d667c2547eb0e9fc21938f4ee7e98f1ceee4bde8ad8815b5f6efe21b60eee798837d68f51a3340d7e5bb7a +0 -0
- package/.tmp-npm-cache/_cacache/content-v2/sha512/fe/40/2abfbefc96299e8bf714aa91d62607190ae299e102cf5933db2e2904640d65d25d67dbbb6fa2ddc92a17f00b9dbfdf2e37487f67d96ec36c64a285b59a7d +0 -0
- package/.tmp-npm-cache/_cacache/index-v5/27/fe/81a3de6ce7ae3d1e41a3421de20c5629998c4ee5d0ffe2037630f03b03b2 +0 -4
- package/.tmp-npm-cache/_cacache/index-v5/65/22/dd66711f62681fce09aabb2357a2907b4a0c778ac5227c4baf9603fd86e8 +0 -4
- package/.tmp-npm-cache/_update-notifier-last-checked +0 -0
- package/AGENTS.md +0 -15
- package/CLAUDE.md +0 -3
- package/README.en.md +0 -304
- package/SECURITY.md +0 -31
- package/client/package.json +0 -29
- package/client/src/App.tsx +0 -123
- package/client/src/activity.ts +0 -5
- package/client/src/api.ts +0 -130
- package/client/src/attention.tsx +0 -157
- package/client/src/components/FileExplorer.tsx +0 -156
- package/client/src/components/FileViewer.tsx +0 -194
- package/client/src/components/SessionRow.tsx +0 -108
- package/client/src/components/SessionTree.tsx +0 -197
- package/client/src/components/SettingsButton.tsx +0 -122
- package/client/src/components/Sidebar.tsx +0 -96
- package/client/src/components/StatusBanner.tsx +0 -31
- package/client/src/components/TargetGroup.tsx +0 -275
- package/client/src/components/TerminalPanel.tsx +0 -192
- package/client/src/folders.ts +0 -245
- package/client/src/hooks/useTerminal.ts +0 -67
- package/client/src/hooks/useTmuxSocket.ts +0 -65
- package/client/src/i18n.ts +0 -213
- package/client/src/main.tsx +0 -17
- package/client/src/settings.tsx +0 -87
- package/client/src/styles.css +0 -723
- package/client/src/types.ts +0 -93
- package/client/src/util.ts +0 -65
- package/client/tsconfig.json +0 -13
- package/client/vite.config.ts +0 -15
- package/fig/fig1.png +0 -0
- package/scripts/prepack.mjs +0 -35
- package/server/package.json +0 -61
- package/server/src/agentHooks.ts +0 -120
- package/server/src/agentOutput.ts +0 -36
- package/server/src/agentState.ts +0 -70
- package/server/src/config.ts +0 -31
- package/server/src/exe.ts +0 -34
- package/server/src/exec.ts +0 -61
- package/server/src/files.ts +0 -330
- package/server/src/foldersStore.ts +0 -114
- package/server/src/index.ts +0 -114
- package/server/src/logger.ts +0 -16
- package/server/src/openBrowser.ts +0 -28
- package/server/src/rest/router.ts +0 -290
- package/server/src/targetCommand.ts +0 -79
- package/server/src/targets.ts +0 -152
- package/server/src/tmux/builder.ts +0 -198
- package/server/src/tmux/formats.ts +0 -95
- package/server/src/tmux/sessions.ts +0 -204
- package/server/src/validate.ts +0 -79
- package/server/src/windowsSsh.ts +0 -239
- package/server/src/winshell/manager.ts +0 -296
- package/server/src/ws/protocol.ts +0 -15
- package/server/src/ws/sshState.ts +0 -36
- package/server/src/ws/terminalSession.ts +0 -207
- package/server/src/ws/wsServer.ts +0 -153
- package/server/src/wsl.ts +0 -38
- package/server/test/agentHooks.test.ts +0 -66
- package/server/test/agentOutput.test.ts +0 -26
- package/server/test/agentState.test.ts +0 -24
- package/server/test/builder.test.ts +0 -162
- package/server/test/files.test.ts +0 -81
- package/server/test/formats.test.ts +0 -123
- package/server/test/monitor.test.ts +0 -25
- package/server/test/validate.test.ts +0 -71
- package/server/test/wsl.test.ts +0 -18
- package/server/tsconfig.json +0 -9
- package/server/vitest.config.ts +0 -12
- package/start.cmd +0 -30
- package/start.command +0 -20
- package/start.sh +0 -20
- package/tsconfig.base.json +0 -19
- /package/{server/bin → bin}/tmuxes.js +0 -0
|
@@ -0,0 +1,204 @@
|
|
|
1
|
+
import * as pty from 'node-pty';
|
|
2
|
+
import { homedir } from 'node:os';
|
|
3
|
+
import { attachArgv } from '../tmux/builder.js';
|
|
4
|
+
import { resolveExecutable } from '../exe.js';
|
|
5
|
+
import { classifySsh } from './sshState.js';
|
|
6
|
+
import { log } from '../logger.js';
|
|
7
|
+
const HEARTBEAT_MS = 30_000;
|
|
8
|
+
const HIGH_WATER = 1 << 20; // 1 MiB buffered → pause the PTY
|
|
9
|
+
const LOW_WATER = 1 << 18; // 256 KiB → resume
|
|
10
|
+
const KILL_GRACE_MS = 2_000;
|
|
11
|
+
/** Owns exactly one PTY for one WebSocket. dispose() is idempotent. */
|
|
12
|
+
export class TerminalSession {
|
|
13
|
+
ws;
|
|
14
|
+
target;
|
|
15
|
+
session;
|
|
16
|
+
ptyProc;
|
|
17
|
+
disposed = false;
|
|
18
|
+
alive = true;
|
|
19
|
+
paused = false;
|
|
20
|
+
heartbeat;
|
|
21
|
+
drainTimer;
|
|
22
|
+
killTimer;
|
|
23
|
+
/** Scan ssh output for failure/prompt states until the link looks healthy. */
|
|
24
|
+
sshScanBudget;
|
|
25
|
+
constructor(ws, target, session, cols, rows) {
|
|
26
|
+
this.ws = ws;
|
|
27
|
+
this.target = target;
|
|
28
|
+
this.session = session;
|
|
29
|
+
this.sshScanBudget = target.kind === 'ssh' ? 8192 : 0;
|
|
30
|
+
const { file, args } = attachArgv(target, session);
|
|
31
|
+
// node-pty on Windows needs a full exe path (no PATH/.exe resolution).
|
|
32
|
+
this.ptyProc = pty.spawn(resolveExecutable(file), args, {
|
|
33
|
+
name: 'xterm-256color',
|
|
34
|
+
cols,
|
|
35
|
+
rows,
|
|
36
|
+
cwd: homedir(),
|
|
37
|
+
env: { ...process.env, TERM: 'xterm-256color' },
|
|
38
|
+
});
|
|
39
|
+
this.ptyProc.onData((data) => this.onPtyData(data));
|
|
40
|
+
this.ptyProc.onExit(({ exitCode }) => this.onPtyExit(exitCode));
|
|
41
|
+
this.ws.on('message', (data, isBinary) => this.onClientMessage(data, isBinary));
|
|
42
|
+
this.ws.on('close', () => this.dispose());
|
|
43
|
+
this.ws.on('error', () => this.dispose());
|
|
44
|
+
this.ws.on('pong', () => {
|
|
45
|
+
this.alive = true;
|
|
46
|
+
});
|
|
47
|
+
this.heartbeat = setInterval(() => this.tick(), HEARTBEAT_MS);
|
|
48
|
+
this.sendControl({ type: 'ready', target: target.id, session });
|
|
49
|
+
}
|
|
50
|
+
onPtyData(data) {
|
|
51
|
+
if (this.sshScanBudget > 0) {
|
|
52
|
+
this.sshScanBudget -= data.length;
|
|
53
|
+
const ssh = classifySsh(data);
|
|
54
|
+
if (ssh)
|
|
55
|
+
this.sendControl({ type: 'ssh', state: ssh.state, message: ssh.message });
|
|
56
|
+
}
|
|
57
|
+
this.sendBinary(Buffer.from(data, 'utf8'));
|
|
58
|
+
}
|
|
59
|
+
onPtyExit(exitCode) {
|
|
60
|
+
this.sendControl({ type: 'exit', code: exitCode });
|
|
61
|
+
this.closeWs(1000, 'pty exited');
|
|
62
|
+
this.dispose();
|
|
63
|
+
}
|
|
64
|
+
onClientMessage(data, isBinary) {
|
|
65
|
+
if (this.disposed)
|
|
66
|
+
return;
|
|
67
|
+
if (isBinary) {
|
|
68
|
+
// Raw keystrokes → straight into the PTY.
|
|
69
|
+
this.ptyProc.write(toBufferString(data));
|
|
70
|
+
return;
|
|
71
|
+
}
|
|
72
|
+
let msg;
|
|
73
|
+
try {
|
|
74
|
+
msg = JSON.parse(toBufferString(data));
|
|
75
|
+
}
|
|
76
|
+
catch {
|
|
77
|
+
return; // ignore malformed control frames
|
|
78
|
+
}
|
|
79
|
+
if (msg.type === 'resize') {
|
|
80
|
+
const cols = clampDim(msg.cols);
|
|
81
|
+
const rows = clampDim(msg.rows);
|
|
82
|
+
if (cols && rows) {
|
|
83
|
+
try {
|
|
84
|
+
this.ptyProc.resize(cols, rows);
|
|
85
|
+
}
|
|
86
|
+
catch {
|
|
87
|
+
/* pty may have exited */
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
else if (msg.type === 'ping') {
|
|
92
|
+
this.sendControl({ type: 'pong' });
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
sendBinary(buf) {
|
|
96
|
+
if (this.disposed || this.ws.readyState !== this.ws.OPEN)
|
|
97
|
+
return;
|
|
98
|
+
this.ws.send(buf, { binary: true });
|
|
99
|
+
if (!this.paused && this.ws.bufferedAmount > HIGH_WATER) {
|
|
100
|
+
this.paused = true;
|
|
101
|
+
this.ptyProc.pause();
|
|
102
|
+
this.drainTimer = setInterval(() => this.checkDrain(), 50);
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
checkDrain() {
|
|
106
|
+
if (this.disposed)
|
|
107
|
+
return;
|
|
108
|
+
if (this.ws.bufferedAmount < LOW_WATER) {
|
|
109
|
+
this.paused = false;
|
|
110
|
+
if (this.drainTimer)
|
|
111
|
+
clearInterval(this.drainTimer);
|
|
112
|
+
this.drainTimer = undefined;
|
|
113
|
+
this.ptyProc.resume();
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
sendControl(msg) {
|
|
117
|
+
if (this.disposed || this.ws.readyState !== this.ws.OPEN)
|
|
118
|
+
return;
|
|
119
|
+
this.ws.send(JSON.stringify(msg), { binary: false });
|
|
120
|
+
}
|
|
121
|
+
tick() {
|
|
122
|
+
if (!this.alive) {
|
|
123
|
+
log.warn(`heartbeat lost for ${this.target.id}/${this.session}, terminating`);
|
|
124
|
+
try {
|
|
125
|
+
this.ws.terminate();
|
|
126
|
+
}
|
|
127
|
+
catch {
|
|
128
|
+
/* ignore */
|
|
129
|
+
}
|
|
130
|
+
this.dispose();
|
|
131
|
+
return;
|
|
132
|
+
}
|
|
133
|
+
this.alive = false;
|
|
134
|
+
try {
|
|
135
|
+
this.ws.ping();
|
|
136
|
+
}
|
|
137
|
+
catch {
|
|
138
|
+
/* ignore */
|
|
139
|
+
}
|
|
140
|
+
}
|
|
141
|
+
closeWs(code, reason) {
|
|
142
|
+
try {
|
|
143
|
+
if (this.ws.readyState === this.ws.OPEN)
|
|
144
|
+
this.ws.close(code, reason);
|
|
145
|
+
}
|
|
146
|
+
catch {
|
|
147
|
+
/* ignore */
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
/** Idempotent teardown — called from pty exit, ws close, and shutdown. */
|
|
151
|
+
dispose() {
|
|
152
|
+
if (this.disposed)
|
|
153
|
+
return;
|
|
154
|
+
this.disposed = true;
|
|
155
|
+
if (this.heartbeat)
|
|
156
|
+
clearInterval(this.heartbeat);
|
|
157
|
+
if (this.drainTimer)
|
|
158
|
+
clearInterval(this.drainTimer);
|
|
159
|
+
try {
|
|
160
|
+
this.ptyProc.kill(); // SIGHUP → tmux client detaches; session keeps running
|
|
161
|
+
}
|
|
162
|
+
catch {
|
|
163
|
+
/* already gone */
|
|
164
|
+
}
|
|
165
|
+
// Force-kill if it lingers.
|
|
166
|
+
this.killTimer = setTimeout(() => {
|
|
167
|
+
try {
|
|
168
|
+
this.ptyProc.kill('SIGKILL');
|
|
169
|
+
}
|
|
170
|
+
catch {
|
|
171
|
+
/* already gone */
|
|
172
|
+
}
|
|
173
|
+
}, KILL_GRACE_MS);
|
|
174
|
+
this.killTimer.unref?.();
|
|
175
|
+
this.closeWs(1000, 'disposed');
|
|
176
|
+
registry.delete(this);
|
|
177
|
+
}
|
|
178
|
+
}
|
|
179
|
+
function toBufferString(data) {
|
|
180
|
+
if (typeof data === 'string')
|
|
181
|
+
return data;
|
|
182
|
+
if (Buffer.isBuffer(data))
|
|
183
|
+
return data.toString('utf8');
|
|
184
|
+
if (Array.isArray(data))
|
|
185
|
+
return Buffer.concat(data).toString('utf8');
|
|
186
|
+
if (data instanceof ArrayBuffer)
|
|
187
|
+
return Buffer.from(data).toString('utf8');
|
|
188
|
+
return String(data);
|
|
189
|
+
}
|
|
190
|
+
function clampDim(n) {
|
|
191
|
+
if (typeof n !== 'number' || !Number.isInteger(n) || n < 1 || n > 1000)
|
|
192
|
+
return null;
|
|
193
|
+
return n;
|
|
194
|
+
}
|
|
195
|
+
/** All live sessions, so the process can tear them down on shutdown. */
|
|
196
|
+
export const registry = new Set();
|
|
197
|
+
export function track(s) {
|
|
198
|
+
registry.add(s);
|
|
199
|
+
}
|
|
200
|
+
export function disposeAll() {
|
|
201
|
+
for (const s of [...registry])
|
|
202
|
+
s.dispose();
|
|
203
|
+
}
|
|
204
|
+
//# sourceMappingURL=terminalSession.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"terminalSession.js","sourceRoot":"","sources":["../../src/ws/terminalSession.ts"],"names":[],"mappings":"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"}
|
|
@@ -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
|
package/dist/wsl.js.map
ADDED
|
@@ -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
1
|
{
|
|
2
2
|
"name": "tmuxes",
|
|
3
|
-
"version": "0.1.
|
|
4
|
-
"
|
|
3
|
+
"version": "0.1.10",
|
|
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
|
+
],
|
|
5
15
|
"license": "MIT",
|
|
6
|
-
"
|
|
7
|
-
"
|
|
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
|
+
},
|
|
8
31
|
"engines": {
|
|
9
32
|
"node": ">=22.12.0 <23"
|
|
10
33
|
},
|
|
11
|
-
"
|
|
12
|
-
"
|
|
13
|
-
"
|
|
34
|
+
"files": [
|
|
35
|
+
"dist",
|
|
36
|
+
"public",
|
|
37
|
+
"bin",
|
|
38
|
+
"README.md",
|
|
39
|
+
"LICENSE"
|
|
14
40
|
],
|
|
15
41
|
"scripts": {
|
|
16
|
-
"dev": "
|
|
17
|
-
"
|
|
18
|
-
"
|
|
19
|
-
"
|
|
20
|
-
"
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
"
|
|
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"
|
|
24
52
|
},
|
|
25
53
|
"devDependencies": {
|
|
26
|
-
"
|
|
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"
|
|
27
60
|
}
|
|
28
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)}
|