panex 0.9.0 → 0.9.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/dist/cli.js +430 -9
- package/dist/cli.js.map +1 -1
- package/dist/index.d.ts +6 -2
- package/dist/index.js +418 -5
- package/dist/index.js.map +1 -1
- package/package.json +3 -4
- package/dist/chunk-Z7LPKPEA.js +0 -372
- package/dist/chunk-Z7LPKPEA.js.map +0 -1
- package/dist/cli.d.ts +0 -1
package/dist/index.js
CHANGED
|
@@ -1,8 +1,421 @@
|
|
|
1
|
-
|
|
2
|
-
import
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
} from "
|
|
1
|
+
// src/tui.ts
|
|
2
|
+
import blessed from "blessed";
|
|
3
|
+
|
|
4
|
+
// src/process-manager.ts
|
|
5
|
+
import { EventEmitter } from "events";
|
|
6
|
+
var ProcessManager = class extends EventEmitter {
|
|
7
|
+
constructor(procs) {
|
|
8
|
+
super();
|
|
9
|
+
this.procs = procs;
|
|
10
|
+
}
|
|
11
|
+
processes = /* @__PURE__ */ new Map();
|
|
12
|
+
maxOutputLines = 1e4;
|
|
13
|
+
async startAll() {
|
|
14
|
+
for (const [name, config] of Object.entries(this.procs)) {
|
|
15
|
+
await this.start(name, config);
|
|
16
|
+
}
|
|
17
|
+
}
|
|
18
|
+
async start(name, config) {
|
|
19
|
+
const existing = this.processes.get(name);
|
|
20
|
+
if (existing?.pty) {
|
|
21
|
+
existing.pty.kill();
|
|
22
|
+
}
|
|
23
|
+
const shell = process.platform === "win32" ? "powershell.exe" : "bash";
|
|
24
|
+
const args = config.shell ? ["-c", config.shell] : config.cmd ? ["-c", config.cmd.join(" ")] : [];
|
|
25
|
+
const cwd = config.cwd ?? process.cwd();
|
|
26
|
+
const env = { ...process.env, ...config.env };
|
|
27
|
+
const managed = {
|
|
28
|
+
name,
|
|
29
|
+
config,
|
|
30
|
+
pty: null,
|
|
31
|
+
status: "running",
|
|
32
|
+
output: [],
|
|
33
|
+
exitCode: null
|
|
34
|
+
};
|
|
35
|
+
this.processes.set(name, managed);
|
|
36
|
+
try {
|
|
37
|
+
const proc = Bun.spawn([shell, ...args], {
|
|
38
|
+
cwd,
|
|
39
|
+
env,
|
|
40
|
+
terminal: {
|
|
41
|
+
cols: 120,
|
|
42
|
+
rows: 30,
|
|
43
|
+
data: (_terminal, data) => {
|
|
44
|
+
const str = new TextDecoder().decode(data);
|
|
45
|
+
managed.output.push(str);
|
|
46
|
+
if (managed.output.length > this.maxOutputLines) {
|
|
47
|
+
managed.output = managed.output.slice(-this.maxOutputLines);
|
|
48
|
+
}
|
|
49
|
+
this.emit("output", name, str);
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
});
|
|
53
|
+
managed.pty = {
|
|
54
|
+
write: (data) => proc.terminal?.write(data),
|
|
55
|
+
resize: (cols, rows) => proc.terminal?.resize(cols, rows),
|
|
56
|
+
kill: () => proc.kill()
|
|
57
|
+
};
|
|
58
|
+
proc.exited.then((exitCode) => {
|
|
59
|
+
managed.status = exitCode === 0 ? "stopped" : "error";
|
|
60
|
+
managed.exitCode = exitCode;
|
|
61
|
+
managed.pty = null;
|
|
62
|
+
this.emit("exit", name, exitCode);
|
|
63
|
+
if (managed.config.autoRestart && exitCode !== 0) {
|
|
64
|
+
setTimeout(() => this.start(name, managed.config), 1e3);
|
|
65
|
+
}
|
|
66
|
+
});
|
|
67
|
+
this.emit("started", name);
|
|
68
|
+
} catch (error) {
|
|
69
|
+
managed.status = "error";
|
|
70
|
+
managed.output = [`Error starting process: ${error}`];
|
|
71
|
+
managed.exitCode = -1;
|
|
72
|
+
this.emit("error", name, error);
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
restart(name) {
|
|
76
|
+
const proc = this.processes.get(name);
|
|
77
|
+
if (proc) {
|
|
78
|
+
if (proc.pty) {
|
|
79
|
+
proc.pty.kill();
|
|
80
|
+
}
|
|
81
|
+
proc.output = [];
|
|
82
|
+
this.start(name, proc.config);
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
restartAll() {
|
|
86
|
+
for (const name of this.processes.keys()) {
|
|
87
|
+
this.restart(name);
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
kill(name) {
|
|
91
|
+
const proc = this.processes.get(name);
|
|
92
|
+
if (proc?.pty) {
|
|
93
|
+
proc.pty.kill();
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
killAll() {
|
|
97
|
+
for (const proc of this.processes.values()) {
|
|
98
|
+
if (proc.pty) {
|
|
99
|
+
proc.pty.kill();
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
write(name, data) {
|
|
104
|
+
const proc = this.processes.get(name);
|
|
105
|
+
if (proc?.pty) {
|
|
106
|
+
proc.pty.write(data);
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
resize(name, cols, rows) {
|
|
110
|
+
const proc = this.processes.get(name);
|
|
111
|
+
if (proc?.pty) {
|
|
112
|
+
proc.pty.resize(cols, rows);
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
getProcess(name) {
|
|
116
|
+
return this.processes.get(name);
|
|
117
|
+
}
|
|
118
|
+
getProcesses() {
|
|
119
|
+
return Array.from(this.processes.values());
|
|
120
|
+
}
|
|
121
|
+
getNames() {
|
|
122
|
+
return Array.from(this.processes.keys());
|
|
123
|
+
}
|
|
124
|
+
getOutput(name) {
|
|
125
|
+
return this.processes.get(name)?.output.join("") ?? "";
|
|
126
|
+
}
|
|
127
|
+
};
|
|
128
|
+
|
|
129
|
+
// src/tui.ts
|
|
130
|
+
async function createTUI(config) {
|
|
131
|
+
const processManager = new ProcessManager(config.procs);
|
|
132
|
+
const screen = blessed.screen({
|
|
133
|
+
smartCSR: true,
|
|
134
|
+
title: "panex",
|
|
135
|
+
fullUnicode: true
|
|
136
|
+
});
|
|
137
|
+
const processList = blessed.list({
|
|
138
|
+
parent: screen,
|
|
139
|
+
label: " PROCESSES ",
|
|
140
|
+
top: 0,
|
|
141
|
+
left: 0,
|
|
142
|
+
width: "20%",
|
|
143
|
+
height: "100%-1",
|
|
144
|
+
border: { type: "line" },
|
|
145
|
+
style: {
|
|
146
|
+
border: { fg: "blue" },
|
|
147
|
+
selected: { bg: "blue", fg: "white" },
|
|
148
|
+
item: { fg: "white" }
|
|
149
|
+
},
|
|
150
|
+
mouse: config.settings?.mouse ?? true,
|
|
151
|
+
scrollbar: {
|
|
152
|
+
ch: "\u2502",
|
|
153
|
+
style: { bg: "blue" }
|
|
154
|
+
}
|
|
155
|
+
});
|
|
156
|
+
const outputBox = blessed.box({
|
|
157
|
+
parent: screen,
|
|
158
|
+
label: " OUTPUT ",
|
|
159
|
+
top: 0,
|
|
160
|
+
left: "20%",
|
|
161
|
+
width: "80%",
|
|
162
|
+
height: "100%-1",
|
|
163
|
+
border: { type: "line" },
|
|
164
|
+
style: {
|
|
165
|
+
border: { fg: "green" }
|
|
166
|
+
},
|
|
167
|
+
scrollable: true,
|
|
168
|
+
alwaysScroll: true,
|
|
169
|
+
scrollbar: {
|
|
170
|
+
ch: "\u2502",
|
|
171
|
+
style: { bg: "green" }
|
|
172
|
+
},
|
|
173
|
+
mouse: config.settings?.mouse ?? true,
|
|
174
|
+
keys: true,
|
|
175
|
+
vi: true
|
|
176
|
+
});
|
|
177
|
+
const statusBar = blessed.box({
|
|
178
|
+
parent: screen,
|
|
179
|
+
bottom: 0,
|
|
180
|
+
left: 0,
|
|
181
|
+
width: "100%",
|
|
182
|
+
height: 1,
|
|
183
|
+
style: {
|
|
184
|
+
bg: "blue",
|
|
185
|
+
fg: "white"
|
|
186
|
+
},
|
|
187
|
+
content: " [\u2191\u2193/jk] select [Enter] focus [r] restart [A] restart All [x] kill [q] quit [?] help "
|
|
188
|
+
});
|
|
189
|
+
const helpBox = blessed.box({
|
|
190
|
+
parent: screen,
|
|
191
|
+
top: "center",
|
|
192
|
+
left: "center",
|
|
193
|
+
width: "60%",
|
|
194
|
+
height: "60%",
|
|
195
|
+
label: " Help ",
|
|
196
|
+
border: { type: "line" },
|
|
197
|
+
style: {
|
|
198
|
+
border: { fg: "yellow" },
|
|
199
|
+
bg: "black"
|
|
200
|
+
},
|
|
201
|
+
hidden: true,
|
|
202
|
+
content: `
|
|
203
|
+
Keyboard Shortcuts
|
|
204
|
+
\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500
|
|
205
|
+
|
|
206
|
+
Navigation
|
|
207
|
+
\u2191/\u2193 or j/k Navigate process list
|
|
208
|
+
g/G Scroll to top/bottom of output
|
|
209
|
+
PgUp/PgDn Scroll output
|
|
210
|
+
|
|
211
|
+
Process Control
|
|
212
|
+
Enter Focus process (interactive mode)
|
|
213
|
+
Esc Exit focus mode
|
|
214
|
+
r Restart selected process
|
|
215
|
+
A Restart all processes
|
|
216
|
+
x Kill selected process
|
|
217
|
+
|
|
218
|
+
General
|
|
219
|
+
? Toggle this help
|
|
220
|
+
q Quit panex
|
|
221
|
+
|
|
222
|
+
Press any key to close this help...
|
|
223
|
+
`
|
|
224
|
+
});
|
|
225
|
+
let selectedIndex = 0;
|
|
226
|
+
let focusMode = false;
|
|
227
|
+
const processNames = Object.keys(config.procs);
|
|
228
|
+
const scrollPositions = /* @__PURE__ */ new Map();
|
|
229
|
+
function updateProcessList() {
|
|
230
|
+
const items = processNames.map((name, i) => {
|
|
231
|
+
const proc = processManager.getProcess(name);
|
|
232
|
+
const status = proc?.status === "running" ? "\u25CF" : proc?.status === "error" ? "\u2717" : "\u25CB";
|
|
233
|
+
const prefix = i === selectedIndex ? "\u25B6" : " ";
|
|
234
|
+
return `${prefix} ${name} ${status}`;
|
|
235
|
+
});
|
|
236
|
+
processList.setItems(items);
|
|
237
|
+
processList.select(selectedIndex);
|
|
238
|
+
screen.render();
|
|
239
|
+
}
|
|
240
|
+
function updateOutput(autoScroll = false) {
|
|
241
|
+
const name = processNames[selectedIndex];
|
|
242
|
+
if (name) {
|
|
243
|
+
outputBox.setLabel(` OUTPUT: ${name} `);
|
|
244
|
+
const output = processManager.getOutput(name);
|
|
245
|
+
outputBox.setContent(output);
|
|
246
|
+
if (autoScroll) {
|
|
247
|
+
outputBox.setScrollPerc(100);
|
|
248
|
+
} else {
|
|
249
|
+
const savedPos = scrollPositions.get(name) ?? 100;
|
|
250
|
+
outputBox.setScrollPerc(savedPos);
|
|
251
|
+
}
|
|
252
|
+
}
|
|
253
|
+
screen.render();
|
|
254
|
+
}
|
|
255
|
+
function saveScrollPosition() {
|
|
256
|
+
const name = processNames[selectedIndex];
|
|
257
|
+
if (name) {
|
|
258
|
+
scrollPositions.set(name, outputBox.getScrollPerc());
|
|
259
|
+
}
|
|
260
|
+
}
|
|
261
|
+
processManager.on("output", (name) => {
|
|
262
|
+
if (name === processNames[selectedIndex]) {
|
|
263
|
+
updateOutput(true);
|
|
264
|
+
}
|
|
265
|
+
});
|
|
266
|
+
processManager.on("started", () => {
|
|
267
|
+
updateProcessList();
|
|
268
|
+
});
|
|
269
|
+
processManager.on("exit", () => {
|
|
270
|
+
updateProcessList();
|
|
271
|
+
});
|
|
272
|
+
processManager.on("error", (name) => {
|
|
273
|
+
updateProcessList();
|
|
274
|
+
if (name === processNames[selectedIndex]) {
|
|
275
|
+
updateOutput();
|
|
276
|
+
}
|
|
277
|
+
});
|
|
278
|
+
screen.key(["q", "C-c"], () => {
|
|
279
|
+
processManager.killAll();
|
|
280
|
+
process.exit(0);
|
|
281
|
+
});
|
|
282
|
+
screen.key(["?"], () => {
|
|
283
|
+
helpBox.toggle();
|
|
284
|
+
screen.render();
|
|
285
|
+
});
|
|
286
|
+
screen.key(["escape"], () => {
|
|
287
|
+
if (!helpBox.hidden) {
|
|
288
|
+
helpBox.hide();
|
|
289
|
+
screen.render();
|
|
290
|
+
return;
|
|
291
|
+
}
|
|
292
|
+
if (focusMode) {
|
|
293
|
+
focusMode = false;
|
|
294
|
+
statusBar.setContent(" [\u2191\u2193/jk] select [Enter] focus [r] restart [A] restart All [x] kill [q] quit [?] help ");
|
|
295
|
+
screen.render();
|
|
296
|
+
}
|
|
297
|
+
});
|
|
298
|
+
helpBox.key(["escape", "q", "?", "enter", "space"], () => {
|
|
299
|
+
helpBox.hide();
|
|
300
|
+
screen.render();
|
|
301
|
+
});
|
|
302
|
+
screen.key(["up", "k"], () => {
|
|
303
|
+
if (focusMode || !helpBox.hidden) return;
|
|
304
|
+
if (selectedIndex > 0) {
|
|
305
|
+
saveScrollPosition();
|
|
306
|
+
selectedIndex--;
|
|
307
|
+
updateProcessList();
|
|
308
|
+
updateOutput();
|
|
309
|
+
}
|
|
310
|
+
});
|
|
311
|
+
screen.key(["down", "j"], () => {
|
|
312
|
+
if (focusMode || !helpBox.hidden) return;
|
|
313
|
+
if (selectedIndex < processNames.length - 1) {
|
|
314
|
+
saveScrollPosition();
|
|
315
|
+
selectedIndex++;
|
|
316
|
+
updateProcessList();
|
|
317
|
+
updateOutput();
|
|
318
|
+
}
|
|
319
|
+
});
|
|
320
|
+
screen.key(["enter"], () => {
|
|
321
|
+
if (!helpBox.hidden) {
|
|
322
|
+
helpBox.hide();
|
|
323
|
+
screen.render();
|
|
324
|
+
return;
|
|
325
|
+
}
|
|
326
|
+
if (!focusMode) {
|
|
327
|
+
focusMode = true;
|
|
328
|
+
const name = processNames[selectedIndex];
|
|
329
|
+
statusBar.setContent(` FOCUS: ${name} - Type to interact, [Esc] to exit focus mode `);
|
|
330
|
+
screen.render();
|
|
331
|
+
} else {
|
|
332
|
+
const name = processNames[selectedIndex];
|
|
333
|
+
if (name) {
|
|
334
|
+
processManager.write(name, "\r");
|
|
335
|
+
}
|
|
336
|
+
}
|
|
337
|
+
});
|
|
338
|
+
screen.key(["r"], () => {
|
|
339
|
+
if (focusMode || !helpBox.hidden) return;
|
|
340
|
+
const name = processNames[selectedIndex];
|
|
341
|
+
if (name) {
|
|
342
|
+
processManager.restart(name);
|
|
343
|
+
}
|
|
344
|
+
});
|
|
345
|
+
screen.key(["S-a"], () => {
|
|
346
|
+
if (focusMode || !helpBox.hidden) return;
|
|
347
|
+
processManager.restartAll();
|
|
348
|
+
});
|
|
349
|
+
screen.key(["x"], () => {
|
|
350
|
+
if (focusMode || !helpBox.hidden) return;
|
|
351
|
+
const name = processNames[selectedIndex];
|
|
352
|
+
if (name) {
|
|
353
|
+
processManager.kill(name);
|
|
354
|
+
}
|
|
355
|
+
});
|
|
356
|
+
screen.key(["g"], () => {
|
|
357
|
+
if (focusMode || !helpBox.hidden) return;
|
|
358
|
+
outputBox.setScrollPerc(0);
|
|
359
|
+
screen.render();
|
|
360
|
+
});
|
|
361
|
+
screen.key(["S-g"], () => {
|
|
362
|
+
if (focusMode || !helpBox.hidden) return;
|
|
363
|
+
outputBox.setScrollPerc(100);
|
|
364
|
+
screen.render();
|
|
365
|
+
});
|
|
366
|
+
outputBox.on("click", () => {
|
|
367
|
+
if (!helpBox.hidden) return;
|
|
368
|
+
if (!focusMode) {
|
|
369
|
+
focusMode = true;
|
|
370
|
+
const name = processNames[selectedIndex];
|
|
371
|
+
statusBar.setContent(` FOCUS: ${name} - Type to interact, [Esc] to exit focus mode `);
|
|
372
|
+
screen.render();
|
|
373
|
+
}
|
|
374
|
+
});
|
|
375
|
+
processList.on("element click", (_el, data) => {
|
|
376
|
+
if (!helpBox.hidden) return;
|
|
377
|
+
const absTop = processList.atop ?? 0;
|
|
378
|
+
const clickedIndex = data.y - absTop - 1;
|
|
379
|
+
if (clickedIndex < 0 || clickedIndex >= processNames.length) return;
|
|
380
|
+
if (focusMode) {
|
|
381
|
+
focusMode = false;
|
|
382
|
+
statusBar.setContent(" [\u2191\u2193/jk] select [Enter] focus [r] restart [A] restart All [x] kill [q] quit [?] help ");
|
|
383
|
+
}
|
|
384
|
+
if (clickedIndex !== selectedIndex) {
|
|
385
|
+
saveScrollPosition();
|
|
386
|
+
selectedIndex = clickedIndex;
|
|
387
|
+
updateProcessList();
|
|
388
|
+
updateOutput();
|
|
389
|
+
}
|
|
390
|
+
screen.render();
|
|
391
|
+
});
|
|
392
|
+
screen.on("keypress", (ch, key) => {
|
|
393
|
+
if (focusMode && ch) {
|
|
394
|
+
if (key.name === "escape" || key.name === "return" || key.name === "enter") {
|
|
395
|
+
return;
|
|
396
|
+
}
|
|
397
|
+
const name = processNames[selectedIndex];
|
|
398
|
+
if (name) {
|
|
399
|
+
processManager.write(name, ch);
|
|
400
|
+
}
|
|
401
|
+
}
|
|
402
|
+
});
|
|
403
|
+
screen.on("resize", () => {
|
|
404
|
+
const name = processNames[selectedIndex];
|
|
405
|
+
if (name) {
|
|
406
|
+
const cols = Math.floor(screen.width * 0.8) - 2;
|
|
407
|
+
const rows = screen.height - 3;
|
|
408
|
+
processManager.resize(name, cols, rows);
|
|
409
|
+
}
|
|
410
|
+
});
|
|
411
|
+
updateProcessList();
|
|
412
|
+
updateOutput();
|
|
413
|
+
processList.focus();
|
|
414
|
+
await processManager.startAll();
|
|
415
|
+
updateProcessList();
|
|
416
|
+
updateOutput();
|
|
417
|
+
screen.render();
|
|
418
|
+
}
|
|
6
419
|
export {
|
|
7
420
|
ProcessManager,
|
|
8
421
|
createTUI
|
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":[],"sourcesContent":[],"mappings":"","names":[]}
|
|
1
|
+
{"version":3,"sources":["../src/tui.ts","../src/process-manager.ts"],"sourcesContent":["import blessed from 'blessed';\nimport type { PanexConfig } from './types';\nimport { ProcessManager } from './process-manager';\n\nexport async function createTUI(config: PanexConfig): Promise<void> {\n const processManager = new ProcessManager(config.procs);\n\n // Create screen\n const screen = blessed.screen({\n smartCSR: true,\n title: 'panex',\n fullUnicode: true,\n });\n\n // Process list (left panel)\n const processList = blessed.list({\n parent: screen,\n label: ' PROCESSES ',\n top: 0,\n left: 0,\n width: '20%',\n height: '100%-1',\n border: { type: 'line' },\n style: {\n border: { fg: 'blue' },\n selected: { bg: 'blue', fg: 'white' },\n item: { fg: 'white' },\n },\n mouse: config.settings?.mouse ?? true,\n scrollbar: {\n ch: '│',\n style: { bg: 'blue' },\n },\n });\n\n // Output panel (right panel)\n const outputBox = blessed.box({\n parent: screen,\n label: ' OUTPUT ',\n top: 0,\n left: '20%',\n width: '80%',\n height: '100%-1',\n border: { type: 'line' },\n style: {\n border: { fg: 'green' },\n },\n scrollable: true,\n alwaysScroll: true,\n scrollbar: {\n ch: '│',\n style: { bg: 'green' },\n },\n mouse: config.settings?.mouse ?? true,\n keys: true,\n vi: true,\n });\n\n // Status bar\n const statusBar = blessed.box({\n parent: screen,\n bottom: 0,\n left: 0,\n width: '100%',\n height: 1,\n style: {\n bg: 'blue',\n fg: 'white',\n },\n content: ' [↑↓/jk] select [Enter] focus [r] restart [A] restart All [x] kill [q] quit [?] help ',\n });\n\n // Help popup\n const helpBox = blessed.box({\n parent: screen,\n top: 'center',\n left: 'center',\n width: '60%',\n height: '60%',\n label: ' Help ',\n border: { type: 'line' },\n style: {\n border: { fg: 'yellow' },\n bg: 'black',\n },\n hidden: true,\n content: `\n Keyboard Shortcuts\n ──────────────────\n\n Navigation\n ↑/↓ or j/k Navigate process list\n g/G Scroll to top/bottom of output\n PgUp/PgDn Scroll output\n\n Process Control\n Enter Focus process (interactive mode)\n Esc Exit focus mode\n r Restart selected process\n A Restart all processes\n x Kill selected process\n\n General\n ? Toggle this help\n q Quit panex\n\n Press any key to close this help...\n `,\n });\n\n // State\n let selectedIndex = 0;\n let focusMode = false;\n const processNames = Object.keys(config.procs);\n const scrollPositions = new Map<string, number>(); // Track scroll % per process\n\n // Update process list UI\n function updateProcessList() {\n const items = processNames.map((name, i) => {\n const proc = processManager.getProcess(name);\n const status = proc?.status === 'running' ? '●' : proc?.status === 'error' ? '✗' : '○';\n const prefix = i === selectedIndex ? '▶' : ' ';\n return `${prefix} ${name} ${status}`;\n });\n processList.setItems(items);\n processList.select(selectedIndex);\n screen.render();\n }\n\n // Update output panel\n function updateOutput(autoScroll = false) {\n const name = processNames[selectedIndex];\n if (name) {\n outputBox.setLabel(` OUTPUT: ${name} `);\n const output = processManager.getOutput(name);\n outputBox.setContent(output);\n if (autoScroll) {\n outputBox.setScrollPerc(100); // Scroll to bottom for new output\n } else {\n // Restore saved scroll position or default to bottom\n const savedPos = scrollPositions.get(name) ?? 100;\n outputBox.setScrollPerc(savedPos);\n }\n }\n screen.render();\n }\n\n // Save current scroll position before switching\n function saveScrollPosition() {\n const name = processNames[selectedIndex];\n if (name) {\n scrollPositions.set(name, outputBox.getScrollPerc());\n }\n }\n\n // Event handlers\n processManager.on('output', (name: string) => {\n if (name === processNames[selectedIndex]) {\n updateOutput(true); // Auto-scroll for new output\n }\n });\n\n processManager.on('started', () => {\n updateProcessList();\n });\n\n processManager.on('exit', () => {\n updateProcessList();\n });\n\n processManager.on('error', (name: string) => {\n updateProcessList();\n if (name === processNames[selectedIndex]) {\n updateOutput();\n }\n });\n\n // Keyboard handling\n screen.key(['q', 'C-c'], () => {\n processManager.killAll();\n process.exit(0);\n });\n\n screen.key(['?'], () => {\n helpBox.toggle();\n screen.render();\n });\n\n screen.key(['escape'], () => {\n if (!helpBox.hidden) {\n helpBox.hide();\n screen.render();\n return;\n }\n if (focusMode) {\n focusMode = false;\n statusBar.setContent(' [↑↓/jk] select [Enter] focus [r] restart [A] restart All [x] kill [q] quit [?] help ');\n screen.render();\n }\n });\n\n helpBox.key(['escape', 'q', '?', 'enter', 'space'], () => {\n helpBox.hide();\n screen.render();\n });\n\n screen.key(['up', 'k'], () => {\n if (focusMode || !helpBox.hidden) return;\n if (selectedIndex > 0) {\n saveScrollPosition();\n selectedIndex--;\n updateProcessList();\n updateOutput();\n }\n });\n\n screen.key(['down', 'j'], () => {\n if (focusMode || !helpBox.hidden) return;\n if (selectedIndex < processNames.length - 1) {\n saveScrollPosition();\n selectedIndex++;\n updateProcessList();\n updateOutput();\n }\n });\n\n screen.key(['enter'], () => {\n if (!helpBox.hidden) {\n helpBox.hide();\n screen.render();\n return;\n }\n if (!focusMode) {\n // Enter focus mode (don't forward this Enter)\n focusMode = true;\n const name = processNames[selectedIndex];\n statusBar.setContent(` FOCUS: ${name} - Type to interact, [Esc] to exit focus mode `);\n screen.render();\n } else {\n // Already in focus mode - forward Enter to process\n const name = processNames[selectedIndex];\n if (name) {\n processManager.write(name, '\\r');\n }\n }\n });\n\n screen.key(['r'], () => {\n if (focusMode || !helpBox.hidden) return;\n const name = processNames[selectedIndex];\n if (name) {\n processManager.restart(name);\n }\n });\n\n screen.key(['S-a'], () => {\n if (focusMode || !helpBox.hidden) return;\n processManager.restartAll();\n });\n\n screen.key(['x'], () => {\n if (focusMode || !helpBox.hidden) return;\n const name = processNames[selectedIndex];\n if (name) {\n processManager.kill(name);\n }\n });\n\n screen.key(['g'], () => {\n if (focusMode || !helpBox.hidden) return;\n outputBox.setScrollPerc(0);\n screen.render();\n });\n\n screen.key(['S-g'], () => {\n if (focusMode || !helpBox.hidden) return;\n outputBox.setScrollPerc(100);\n screen.render();\n });\n\n // Mouse click on output box enables focus mode\n outputBox.on('click', () => {\n if (!helpBox.hidden) return;\n if (!focusMode) {\n focusMode = true;\n const name = processNames[selectedIndex];\n statusBar.setContent(` FOCUS: ${name} - Type to interact, [Esc] to exit focus mode `);\n screen.render();\n }\n });\n\n // Mouse click on process list (single click)\n processList.on('element click', (_el: blessed.Widgets.BlessedElement, data: { y: number }) => {\n if (!helpBox.hidden) return;\n\n // Calculate index: y is absolute, subtract list's absolute top and border\n const absTop = (processList.atop as number) ?? 0;\n const clickedIndex = data.y - absTop - 1; // -1 for border\n\n if (clickedIndex < 0 || clickedIndex >= processNames.length) return;\n\n // Exit focus mode on click\n if (focusMode) {\n focusMode = false;\n statusBar.setContent(' [↑↓/jk] select [Enter] focus [r] restart [A] restart All [x] kill [q] quit [?] help ');\n }\n if (clickedIndex !== selectedIndex) {\n saveScrollPosition();\n selectedIndex = clickedIndex;\n updateProcessList();\n updateOutput();\n }\n screen.render();\n });\n\n // Forward input in focus mode\n screen.on('keypress', (ch: string, key: { full: string; name?: string }) => {\n if (focusMode && ch) {\n // Don't forward Escape (used to exit focus) or Enter (handled separately)\n if (key.name === 'escape' || key.name === 'return' || key.name === 'enter') {\n return;\n }\n const name = processNames[selectedIndex];\n if (name) {\n processManager.write(name, ch);\n }\n }\n });\n\n // Handle resize\n screen.on('resize', () => {\n const name = processNames[selectedIndex];\n if (name) {\n const cols = Math.floor((screen.width as number) * 0.8) - 2;\n const rows = (screen.height as number) - 3;\n processManager.resize(name, cols, rows);\n }\n });\n\n // Initial render\n updateProcessList();\n updateOutput();\n processList.focus();\n\n // Start all processes\n await processManager.startAll();\n updateProcessList();\n updateOutput();\n\n screen.render();\n}","import { EventEmitter } from 'events';\nimport type { ProcessConfig } from './types';\n\ninterface PtyHandle {\n write(data: string): void;\n resize(cols: number, rows: number): void;\n kill(): void;\n}\n\nexport interface ManagedProcess {\n name: string;\n config: ProcessConfig;\n pty: PtyHandle | null;\n status: 'running' | 'stopped' | 'error';\n output: string[];\n exitCode: number | null;\n}\n\nexport class ProcessManager extends EventEmitter {\n private processes: Map<string, ManagedProcess> = new Map();\n private maxOutputLines = 10000;\n\n constructor(private procs: Record<string, ProcessConfig>) {\n super();\n }\n\n async startAll(): Promise<void> {\n for (const [name, config] of Object.entries(this.procs)) {\n await this.start(name, config);\n }\n }\n\n async start(name: string, config: ProcessConfig): Promise<void> {\n const existing = this.processes.get(name);\n if (existing?.pty) {\n existing.pty.kill();\n }\n\n const shell = process.platform === 'win32' ? 'powershell.exe' : 'bash';\n const args = config.shell\n ? ['-c', config.shell]\n : config.cmd\n ? ['-c', config.cmd.join(' ')]\n : [];\n\n const cwd = config.cwd ?? process.cwd();\n const env = { ...process.env, ...config.env };\n\n const managed: ManagedProcess = {\n name,\n config,\n pty: null,\n status: 'running',\n output: [],\n exitCode: null,\n };\n\n this.processes.set(name, managed);\n\n try {\n const proc = Bun.spawn([shell, ...args], {\n cwd,\n env: env as Record<string, string>,\n terminal: {\n cols: 120,\n rows: 30,\n data: (_terminal: unknown, data: Uint8Array) => {\n const str = new TextDecoder().decode(data);\n managed.output.push(str);\n if (managed.output.length > this.maxOutputLines) {\n managed.output = managed.output.slice(-this.maxOutputLines);\n }\n this.emit('output', name, str);\n },\n },\n });\n\n managed.pty = {\n write: (data: string) => proc.terminal?.write(data),\n resize: (cols: number, rows: number) => proc.terminal?.resize(cols, rows),\n kill: () => proc.kill(),\n };\n\n // Handle exit\n proc.exited.then((exitCode) => {\n managed.status = exitCode === 0 ? 'stopped' : 'error';\n managed.exitCode = exitCode;\n managed.pty = null;\n this.emit('exit', name, exitCode);\n\n if (managed.config.autoRestart && exitCode !== 0) {\n setTimeout(() => this.start(name, managed.config), 1000);\n }\n });\n\n this.emit('started', name);\n } catch (error) {\n managed.status = 'error';\n managed.output = [`Error starting process: ${error}`];\n managed.exitCode = -1;\n this.emit('error', name, error);\n }\n }\n\n restart(name: string): void {\n const proc = this.processes.get(name);\n if (proc) {\n if (proc.pty) {\n proc.pty.kill();\n }\n proc.output = [];\n this.start(name, proc.config);\n }\n }\n\n restartAll(): void {\n for (const name of this.processes.keys()) {\n this.restart(name);\n }\n }\n\n kill(name: string): void {\n const proc = this.processes.get(name);\n if (proc?.pty) {\n proc.pty.kill();\n }\n }\n\n killAll(): void {\n for (const proc of this.processes.values()) {\n if (proc.pty) {\n proc.pty.kill();\n }\n }\n }\n\n write(name: string, data: string): void {\n const proc = this.processes.get(name);\n if (proc?.pty) {\n proc.pty.write(data);\n }\n }\n\n resize(name: string, cols: number, rows: number): void {\n const proc = this.processes.get(name);\n if (proc?.pty) {\n proc.pty.resize(cols, rows);\n }\n }\n\n getProcess(name: string): ManagedProcess | undefined {\n return this.processes.get(name);\n }\n\n getProcesses(): ManagedProcess[] {\n return Array.from(this.processes.values());\n }\n\n getNames(): string[] {\n return Array.from(this.processes.keys());\n }\n\n getOutput(name: string): string {\n return this.processes.get(name)?.output.join('') ?? '';\n }\n}\n"],"mappings":"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","names":[]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "panex",
|
|
3
|
-
"version": "0.9.
|
|
3
|
+
"version": "0.9.2",
|
|
4
4
|
"description": "Terminal UI for running multiple processes in parallel",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"bin": {
|
|
@@ -36,13 +36,12 @@
|
|
|
36
36
|
"author": "Anton Veretennikov (king8fisher)",
|
|
37
37
|
"license": "MIT",
|
|
38
38
|
"engines": {
|
|
39
|
-
"
|
|
39
|
+
"bun": ">=1.3.5"
|
|
40
40
|
},
|
|
41
41
|
"dependencies": {
|
|
42
42
|
"blessed": "^0.1.81",
|
|
43
43
|
"chalk": "^5.3.0",
|
|
44
|
-
"commander": "^12.1.0"
|
|
45
|
-
"node-pty": "^1.0.0"
|
|
44
|
+
"commander": "^12.1.0"
|
|
46
45
|
},
|
|
47
46
|
"devDependencies": {
|
|
48
47
|
"@types/blessed": "^0.1.25",
|