conqr 0.0.1
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/README.md +113 -0
- package/dist/cli.d.ts +7 -0
- package/dist/cli.d.ts.map +1 -0
- package/dist/cli.js +45 -0
- package/dist/cli.js.map +1 -0
- package/dist/config.d.ts +9 -0
- package/dist/config.d.ts.map +1 -0
- package/dist/config.js +62 -0
- package/dist/config.js.map +1 -0
- package/dist/index.d.ts +3 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +23 -0
- package/dist/index.js.map +1 -0
- package/dist/log-buffer.d.ts +17 -0
- package/dist/log-buffer.d.ts.map +1 -0
- package/dist/log-buffer.js +41 -0
- package/dist/log-buffer.js.map +1 -0
- package/dist/process-manager.d.ts +26 -0
- package/dist/process-manager.d.ts.map +1 -0
- package/dist/process-manager.js +97 -0
- package/dist/process-manager.js.map +1 -0
- package/dist/ui.d.ts +12 -0
- package/dist/ui.d.ts.map +1 -0
- package/dist/ui.js +224 -0
- package/dist/ui.js.map +1 -0
- package/package.json +48 -0
package/README.md
ADDED
|
@@ -0,0 +1,113 @@
|
|
|
1
|
+
# conqr
|
|
2
|
+
|
|
3
|
+
Dead-simple TUI process runner for Node.js.
|
|
4
|
+
|
|
5
|
+
## Usage
|
|
6
|
+
|
|
7
|
+
### Command Line Arguments
|
|
8
|
+
|
|
9
|
+
```bash
|
|
10
|
+
conqr 'npm run dev' 'npm run build:emails' 'npm run worker'
|
|
11
|
+
```
|
|
12
|
+
|
|
13
|
+
You can customize process names using the `'name'='command'` syntax:
|
|
14
|
+
|
|
15
|
+
```bash
|
|
16
|
+
conqr 'Dev Server'='npm run dev' 'Build Process'='npm run build' 'Worker'='npm run worker'
|
|
17
|
+
```
|
|
18
|
+
|
|
19
|
+
### Configuration File
|
|
20
|
+
|
|
21
|
+
Create a `conqr.json` or `.conqr.json` file in your project directory:
|
|
22
|
+
|
|
23
|
+
**Array of commands:**
|
|
24
|
+
```json
|
|
25
|
+
{
|
|
26
|
+
"commands": [
|
|
27
|
+
"npm run dev",
|
|
28
|
+
"npm run build",
|
|
29
|
+
"npm run worker"
|
|
30
|
+
]
|
|
31
|
+
}
|
|
32
|
+
```
|
|
33
|
+
|
|
34
|
+
**Object with custom names:**
|
|
35
|
+
```json
|
|
36
|
+
{
|
|
37
|
+
"commands": {
|
|
38
|
+
"Dev Server": "npm run dev",
|
|
39
|
+
"Build Process": "npm run build",
|
|
40
|
+
"Worker": "npm run worker"
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
```
|
|
44
|
+
|
|
45
|
+
**Array of objects:**
|
|
46
|
+
```json
|
|
47
|
+
{
|
|
48
|
+
"commands": [
|
|
49
|
+
{
|
|
50
|
+
"name": "Dev Server",
|
|
51
|
+
"command": "npm run dev"
|
|
52
|
+
},
|
|
53
|
+
{
|
|
54
|
+
"name": "Build Process",
|
|
55
|
+
"command": "npm run build"
|
|
56
|
+
}
|
|
57
|
+
]
|
|
58
|
+
}
|
|
59
|
+
```
|
|
60
|
+
|
|
61
|
+
Then simply run:
|
|
62
|
+
```bash
|
|
63
|
+
conqr
|
|
64
|
+
```
|
|
65
|
+
|
|
66
|
+
CLI arguments take precedence over the config file if both are provided.
|
|
67
|
+
|
|
68
|
+
## Demo
|
|
69
|
+
|
|
70
|
+
Try it with the included demo scripts:
|
|
71
|
+
|
|
72
|
+
```bash
|
|
73
|
+
npm start 'node demo/logger1.js' 'node demo/logger2.js' 'node demo/logger3.js'
|
|
74
|
+
```
|
|
75
|
+
|
|
76
|
+
## Features
|
|
77
|
+
|
|
78
|
+
- Run multiple commands concurrently
|
|
79
|
+
- Two-pane interface:
|
|
80
|
+
- **Sidebar**: "All processes" menu item and list of commands with status indicators (▲ = running, ▼ = stopped/error)
|
|
81
|
+
- **Main pane**: Logs from selected command or unified view when "All processes" is selected
|
|
82
|
+
- Keyboard controls:
|
|
83
|
+
- **Arrow Up/Down**: Navigate between commands (including "All processes" menu item)
|
|
84
|
+
- **q**: Quit application
|
|
85
|
+
|
|
86
|
+
## Installation
|
|
87
|
+
|
|
88
|
+
```bash
|
|
89
|
+
npm install
|
|
90
|
+
```
|
|
91
|
+
|
|
92
|
+
## Build
|
|
93
|
+
|
|
94
|
+
```bash
|
|
95
|
+
npm run build
|
|
96
|
+
```
|
|
97
|
+
|
|
98
|
+
## Run
|
|
99
|
+
|
|
100
|
+
Development (using tsx):
|
|
101
|
+
```bash
|
|
102
|
+
npm start 'command1' 'command2' 'command3'
|
|
103
|
+
```
|
|
104
|
+
|
|
105
|
+
Or after building:
|
|
106
|
+
```bash
|
|
107
|
+
node dist/index.js 'command1' 'command2' 'command3'
|
|
108
|
+
```
|
|
109
|
+
|
|
110
|
+
Or after installing globally:
|
|
111
|
+
```bash
|
|
112
|
+
conqr 'command1' 'command2' 'command3'
|
|
113
|
+
```
|
package/dist/cli.d.ts
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"cli.d.ts","sourceRoot":"","sources":["../src/cli.ts"],"names":[],"mappings":"AAAA,MAAM,WAAW,WAAW;IAC1B,EAAE,EAAE,MAAM,CAAC;IACX,IAAI,EAAE,MAAM,CAAC;IACb,OAAO,EAAE,MAAM,CAAC;CACjB;AAED,wBAAgB,aAAa,IAAI,WAAW,EAAE,CA0C7C"}
|
package/dist/cli.js
ADDED
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
export function parseCommands() {
|
|
2
|
+
const args = process.argv.slice(2);
|
|
3
|
+
if (args.length === 0) {
|
|
4
|
+
return [];
|
|
5
|
+
}
|
|
6
|
+
return args.map((arg, index) => {
|
|
7
|
+
const equalsIndex = arg.indexOf('=');
|
|
8
|
+
let command;
|
|
9
|
+
let name;
|
|
10
|
+
if (equalsIndex > 0 && equalsIndex < arg.length - 1) {
|
|
11
|
+
name = arg.substring(0, equalsIndex).trim();
|
|
12
|
+
command = arg.substring(equalsIndex + 1).trim();
|
|
13
|
+
if (name.startsWith("'") && name.endsWith("'")) {
|
|
14
|
+
name = name.slice(1, -1);
|
|
15
|
+
}
|
|
16
|
+
else if (name.startsWith('"') && name.endsWith('"')) {
|
|
17
|
+
name = name.slice(1, -1);
|
|
18
|
+
}
|
|
19
|
+
if (command.startsWith("'") && command.endsWith("'")) {
|
|
20
|
+
command = command.slice(1, -1);
|
|
21
|
+
}
|
|
22
|
+
else if (command.startsWith('"') && command.endsWith('"')) {
|
|
23
|
+
command = command.slice(1, -1);
|
|
24
|
+
}
|
|
25
|
+
if (name.length === 0) {
|
|
26
|
+
name = extractCommandName(command);
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
else {
|
|
30
|
+
command = arg;
|
|
31
|
+
name = extractCommandName(command);
|
|
32
|
+
}
|
|
33
|
+
return {
|
|
34
|
+
id: index,
|
|
35
|
+
name,
|
|
36
|
+
command
|
|
37
|
+
};
|
|
38
|
+
});
|
|
39
|
+
}
|
|
40
|
+
function extractCommandName(command) {
|
|
41
|
+
const firstWord = command.trim().split(/\s+/)[0];
|
|
42
|
+
const basename = firstWord.split('/').pop();
|
|
43
|
+
return basename || `cmd${Date.now()}`;
|
|
44
|
+
}
|
|
45
|
+
//# sourceMappingURL=cli.js.map
|
package/dist/cli.js.map
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"cli.js","sourceRoot":"","sources":["../src/cli.ts"],"names":[],"mappings":"AAMA,MAAM,UAAU,aAAa;IAC3B,MAAM,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;IAEnC,IAAI,IAAI,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACtB,OAAO,EAAE,CAAC;IACZ,CAAC;IAED,OAAO,IAAI,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,KAAK,EAAE,EAAE;QAC7B,MAAM,WAAW,GAAG,GAAG,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;QACrC,IAAI,OAAe,CAAC;QACpB,IAAI,IAAY,CAAC;QAEjB,IAAI,WAAW,GAAG,CAAC,IAAI,WAAW,GAAG,GAAG,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACpD,IAAI,GAAG,GAAG,CAAC,SAAS,CAAC,CAAC,EAAE,WAAW,CAAC,CAAC,IAAI,EAAE,CAAC;YAC5C,OAAO,GAAG,GAAG,CAAC,SAAS,CAAC,WAAW,GAAG,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC;YAEhD,IAAI,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,IAAI,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE,CAAC;gBAC/C,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;YAC3B,CAAC;iBAAM,IAAI,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,IAAI,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE,CAAC;gBACtD,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;YAC3B,CAAC;YAED,IAAI,OAAO,CAAC,UAAU,CAAC,GAAG,CAAC,IAAI,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE,CAAC;gBACrD,OAAO,GAAG,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;YACjC,CAAC;iBAAM,IAAI,OAAO,CAAC,UAAU,CAAC,GAAG,CAAC,IAAI,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE,CAAC;gBAC5D,OAAO,GAAG,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;YACjC,CAAC;YAED,IAAI,IAAI,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBACtB,IAAI,GAAG,kBAAkB,CAAC,OAAO,CAAC,CAAC;YACrC,CAAC;QACH,CAAC;aAAM,CAAC;YACN,OAAO,GAAG,GAAG,CAAC;YACd,IAAI,GAAG,kBAAkB,CAAC,OAAO,CAAC,CAAC;QACrC,CAAC;QAED,OAAO;YACL,EAAE,EAAE,KAAK;YACT,IAAI;YACJ,OAAO;SACR,CAAC;IACJ,CAAC,CAAC,CAAC;AACL,CAAC;AAED,SAAS,kBAAkB,CAAC,OAAe;IACzC,MAAM,SAAS,GAAG,OAAO,CAAC,IAAI,EAAE,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;IACjD,MAAM,QAAQ,GAAG,SAAS,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,CAAC;IAC5C,OAAO,QAAQ,IAAI,MAAM,IAAI,CAAC,GAAG,EAAE,EAAE,CAAC;AACxC,CAAC"}
|
package/dist/config.d.ts
ADDED
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
import { CommandInfo } from './cli.js';
|
|
2
|
+
export interface ConfigFile {
|
|
3
|
+
commands?: Array<string | {
|
|
4
|
+
name: string;
|
|
5
|
+
command: string;
|
|
6
|
+
}> | Record<string, string>;
|
|
7
|
+
}
|
|
8
|
+
export declare function loadConfig(): CommandInfo[] | null;
|
|
9
|
+
//# sourceMappingURL=config.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"config.d.ts","sourceRoot":"","sources":["../src/config.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,WAAW,EAAE,MAAM,UAAU,CAAC;AAIvC,MAAM,WAAW,UAAU;IACzB,QAAQ,CAAC,EAAE,KAAK,CAAC,MAAM,GAAG;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,OAAO,EAAE,MAAM,CAAA;KAAE,CAAC,GAAG,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;CACvF;AAED,wBAAgB,UAAU,IAAI,WAAW,EAAE,GAAG,IAAI,CAkBjD"}
|
package/dist/config.js
ADDED
|
@@ -0,0 +1,62 @@
|
|
|
1
|
+
import { readFileSync, existsSync } from 'fs';
|
|
2
|
+
import { join } from 'path';
|
|
3
|
+
const CONFIG_FILES = ['.conqr.json', 'conqr.json'];
|
|
4
|
+
export function loadConfig() {
|
|
5
|
+
const cwd = process.cwd();
|
|
6
|
+
for (const configFile of CONFIG_FILES) {
|
|
7
|
+
const configPath = join(cwd, configFile);
|
|
8
|
+
if (existsSync(configPath)) {
|
|
9
|
+
try {
|
|
10
|
+
const content = readFileSync(configPath, 'utf-8');
|
|
11
|
+
const config = JSON.parse(content);
|
|
12
|
+
return parseConfigCommands(config);
|
|
13
|
+
}
|
|
14
|
+
catch (err) {
|
|
15
|
+
console.error(`Error reading config file ${configFile}:`, err);
|
|
16
|
+
return null;
|
|
17
|
+
}
|
|
18
|
+
}
|
|
19
|
+
}
|
|
20
|
+
return null;
|
|
21
|
+
}
|
|
22
|
+
function parseConfigCommands(config) {
|
|
23
|
+
if (!config.commands) {
|
|
24
|
+
return [];
|
|
25
|
+
}
|
|
26
|
+
const commands = [];
|
|
27
|
+
if (Array.isArray(config.commands)) {
|
|
28
|
+
config.commands.forEach((cmd, index) => {
|
|
29
|
+
if (typeof cmd === 'string') {
|
|
30
|
+
commands.push({
|
|
31
|
+
id: index,
|
|
32
|
+
name: extractCommandName(cmd),
|
|
33
|
+
command: cmd
|
|
34
|
+
});
|
|
35
|
+
}
|
|
36
|
+
else if (typeof cmd === 'object' && cmd.name && cmd.command) {
|
|
37
|
+
commands.push({
|
|
38
|
+
id: index,
|
|
39
|
+
name: cmd.name,
|
|
40
|
+
command: cmd.command
|
|
41
|
+
});
|
|
42
|
+
}
|
|
43
|
+
});
|
|
44
|
+
}
|
|
45
|
+
else if (typeof config.commands === 'object') {
|
|
46
|
+
let index = 0;
|
|
47
|
+
for (const [name, command] of Object.entries(config.commands)) {
|
|
48
|
+
commands.push({
|
|
49
|
+
id: index++,
|
|
50
|
+
name,
|
|
51
|
+
command: typeof command === 'string' ? command : String(command)
|
|
52
|
+
});
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
return commands;
|
|
56
|
+
}
|
|
57
|
+
function extractCommandName(command) {
|
|
58
|
+
const firstWord = command.trim().split(/\s+/)[0];
|
|
59
|
+
const basename = firstWord.split('/').pop();
|
|
60
|
+
return basename || `cmd${Date.now()}`;
|
|
61
|
+
}
|
|
62
|
+
//# sourceMappingURL=config.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"config.js","sourceRoot":"","sources":["../src/config.ts"],"names":[],"mappings":"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"}
|
package/dist/index.d.ts
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":""}
|
package/dist/index.js
ADDED
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
import { parseCommands } from './cli.js';
|
|
3
|
+
import { loadConfig } from './config.js';
|
|
4
|
+
import { ProcessManager } from './process-manager.js';
|
|
5
|
+
import { LogBuffer } from './log-buffer.js';
|
|
6
|
+
import { renderTUI } from './ui.jsx';
|
|
7
|
+
const cliCommands = parseCommands();
|
|
8
|
+
const configCommands = loadConfig();
|
|
9
|
+
let commands;
|
|
10
|
+
if (cliCommands.length > 0) {
|
|
11
|
+
commands = cliCommands;
|
|
12
|
+
}
|
|
13
|
+
else if (configCommands && configCommands.length > 0) {
|
|
14
|
+
commands = configCommands;
|
|
15
|
+
}
|
|
16
|
+
else {
|
|
17
|
+
console.error('No commands provided. Use CLI arguments or create a conqr.json config file.');
|
|
18
|
+
process.exit(1);
|
|
19
|
+
}
|
|
20
|
+
const logBuffer = new LogBuffer();
|
|
21
|
+
const processManager = new ProcessManager(logBuffer);
|
|
22
|
+
renderTUI(commands, processManager, logBuffer);
|
|
23
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";AAEA,OAAO,EAAE,aAAa,EAAE,MAAM,UAAU,CAAC;AACzC,OAAO,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AACzC,OAAO,EAAE,cAAc,EAAE,MAAM,sBAAsB,CAAC;AACtD,OAAO,EAAE,SAAS,EAAE,MAAM,iBAAiB,CAAC;AAC5C,OAAO,EAAE,SAAS,EAAE,MAAM,UAAU,CAAC;AAErC,MAAM,WAAW,GAAG,aAAa,EAAE,CAAC;AACpC,MAAM,cAAc,GAAG,UAAU,EAAE,CAAC;AAEpC,IAAI,QAAQ,CAAC;AACb,IAAI,WAAW,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;IAC3B,QAAQ,GAAG,WAAW,CAAC;AACzB,CAAC;KAAM,IAAI,cAAc,IAAI,cAAc,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;IACvD,QAAQ,GAAG,cAAc,CAAC;AAC5B,CAAC;KAAM,CAAC;IACN,OAAO,CAAC,KAAK,CAAC,6EAA6E,CAAC,CAAC;IAC7F,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;AAClB,CAAC;AAED,MAAM,SAAS,GAAG,IAAI,SAAS,EAAE,CAAC;AAClC,MAAM,cAAc,GAAG,IAAI,cAAc,CAAC,SAAS,CAAC,CAAC;AAErD,SAAS,CAAC,QAAQ,EAAE,cAAc,EAAE,SAAS,CAAC,CAAC"}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
export interface LogEntry {
|
|
2
|
+
line: string;
|
|
3
|
+
source: 'stdout' | 'stderr';
|
|
4
|
+
timestamp: number;
|
|
5
|
+
processId?: number;
|
|
6
|
+
}
|
|
7
|
+
export declare class LogBuffer {
|
|
8
|
+
private buffers;
|
|
9
|
+
private unifiedBuffer;
|
|
10
|
+
private maxLines;
|
|
11
|
+
constructor();
|
|
12
|
+
addLog(processId: number, line: string, source?: 'stdout' | 'stderr'): void;
|
|
13
|
+
getLogs(processId: number): LogEntry[];
|
|
14
|
+
getUnifiedLogs(): LogEntry[];
|
|
15
|
+
clear(processId?: number): void;
|
|
16
|
+
}
|
|
17
|
+
//# sourceMappingURL=log-buffer.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"log-buffer.d.ts","sourceRoot":"","sources":["../src/log-buffer.ts"],"names":[],"mappings":"AAEA,MAAM,WAAW,QAAQ;IACvB,IAAI,EAAE,MAAM,CAAC;IACb,MAAM,EAAE,QAAQ,GAAG,QAAQ,CAAC;IAC5B,SAAS,EAAE,MAAM,CAAC;IAClB,SAAS,CAAC,EAAE,MAAM,CAAC;CACpB;AAED,qBAAa,SAAS;IACpB,OAAO,CAAC,OAAO,CAA0B;IACzC,OAAO,CAAC,aAAa,CAAa;IAClC,OAAO,CAAC,QAAQ,CAAS;;IAQzB,MAAM,CAAC,SAAS,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,EAAE,MAAM,GAAE,QAAQ,GAAG,QAAmB,GAAG,IAAI;IAkBrF,OAAO,CAAC,SAAS,EAAE,MAAM,GAAG,QAAQ,EAAE;IAItC,cAAc,IAAI,QAAQ,EAAE;IAI5B,KAAK,CAAC,SAAS,CAAC,EAAE,MAAM,GAAG,IAAI;CAQhC"}
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
const MAX_LINES_PER_PROCESS = 1000;
|
|
2
|
+
export class LogBuffer {
|
|
3
|
+
buffers;
|
|
4
|
+
unifiedBuffer;
|
|
5
|
+
maxLines;
|
|
6
|
+
constructor() {
|
|
7
|
+
this.buffers = new Map();
|
|
8
|
+
this.unifiedBuffer = [];
|
|
9
|
+
this.maxLines = MAX_LINES_PER_PROCESS;
|
|
10
|
+
}
|
|
11
|
+
addLog(processId, line, source = 'stdout') {
|
|
12
|
+
if (!this.buffers.has(processId)) {
|
|
13
|
+
this.buffers.set(processId, []);
|
|
14
|
+
}
|
|
15
|
+
const buffer = this.buffers.get(processId);
|
|
16
|
+
buffer.push({ line, source, timestamp: Date.now() });
|
|
17
|
+
if (buffer.length > this.maxLines) {
|
|
18
|
+
buffer.shift();
|
|
19
|
+
}
|
|
20
|
+
this.unifiedBuffer.push({ processId, line, source, timestamp: Date.now() });
|
|
21
|
+
if (this.unifiedBuffer.length > this.maxLines * 10) {
|
|
22
|
+
this.unifiedBuffer.shift();
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
getLogs(processId) {
|
|
26
|
+
return this.buffers.get(processId) || [];
|
|
27
|
+
}
|
|
28
|
+
getUnifiedLogs() {
|
|
29
|
+
return this.unifiedBuffer;
|
|
30
|
+
}
|
|
31
|
+
clear(processId) {
|
|
32
|
+
if (processId !== undefined) {
|
|
33
|
+
this.buffers.delete(processId);
|
|
34
|
+
}
|
|
35
|
+
else {
|
|
36
|
+
this.buffers.clear();
|
|
37
|
+
this.unifiedBuffer = [];
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
//# sourceMappingURL=log-buffer.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"log-buffer.js","sourceRoot":"","sources":["../src/log-buffer.ts"],"names":[],"mappings":"AAAA,MAAM,qBAAqB,GAAG,IAAI,CAAC;AASnC,MAAM,OAAO,SAAS;IACZ,OAAO,CAA0B;IACjC,aAAa,CAAa;IAC1B,QAAQ,CAAS;IAEzB;QACE,IAAI,CAAC,OAAO,GAAG,IAAI,GAAG,EAAE,CAAC;QACzB,IAAI,CAAC,aAAa,GAAG,EAAE,CAAC;QACxB,IAAI,CAAC,QAAQ,GAAG,qBAAqB,CAAC;IACxC,CAAC;IAED,MAAM,CAAC,SAAiB,EAAE,IAAY,EAAE,SAA8B,QAAQ;QAC5E,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,SAAS,CAAC,EAAE,CAAC;YACjC,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,SAAS,EAAE,EAAE,CAAC,CAAC;QAClC,CAAC;QAED,MAAM,MAAM,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,SAAS,CAAE,CAAC;QAC5C,MAAM,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,SAAS,EAAE,IAAI,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;QAErD,IAAI,MAAM,CAAC,MAAM,GAAG,IAAI,CAAC,QAAQ,EAAE,CAAC;YAClC,MAAM,CAAC,KAAK,EAAE,CAAC;QACjB,CAAC;QAED,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,EAAE,SAAS,EAAE,IAAI,EAAE,MAAM,EAAE,SAAS,EAAE,IAAI,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;QAC5E,IAAI,IAAI,CAAC,aAAa,CAAC,MAAM,GAAG,IAAI,CAAC,QAAQ,GAAG,EAAE,EAAE,CAAC;YACnD,IAAI,CAAC,aAAa,CAAC,KAAK,EAAE,CAAC;QAC7B,CAAC;IACH,CAAC;IAED,OAAO,CAAC,SAAiB;QACvB,OAAO,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,SAAS,CAAC,IAAI,EAAE,CAAC;IAC3C,CAAC;IAED,cAAc;QACZ,OAAO,IAAI,CAAC,aAAa,CAAC;IAC5B,CAAC;IAED,KAAK,CAAC,SAAkB;QACtB,IAAI,SAAS,KAAK,SAAS,EAAE,CAAC;YAC5B,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;QACjC,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,OAAO,CAAC,KAAK,EAAE,CAAC;YACrB,IAAI,CAAC,aAAa,GAAG,EAAE,CAAC;QAC1B,CAAC;IACH,CAAC;CACF"}
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
import { ChildProcess } from 'child_process';
|
|
2
|
+
import { EventEmitter } from 'events';
|
|
3
|
+
import { CommandInfo } from './cli.js';
|
|
4
|
+
import { LogBuffer } from './log-buffer.js';
|
|
5
|
+
export type ProcessStatus = 'running' | 'stopped' | 'error' | 'unknown';
|
|
6
|
+
export interface LogEvent {
|
|
7
|
+
processId: number;
|
|
8
|
+
line: string;
|
|
9
|
+
source: 'stdout' | 'stderr';
|
|
10
|
+
}
|
|
11
|
+
export interface StatusChangeEvent {
|
|
12
|
+
processId: number;
|
|
13
|
+
status: ProcessStatus;
|
|
14
|
+
}
|
|
15
|
+
export declare class ProcessManager extends EventEmitter {
|
|
16
|
+
private processes;
|
|
17
|
+
private logBuffer;
|
|
18
|
+
private buffers;
|
|
19
|
+
constructor(logBuffer: LogBuffer);
|
|
20
|
+
startCommand(commandInfo: CommandInfo): ChildProcess;
|
|
21
|
+
startAll(commands: CommandInfo[]): void;
|
|
22
|
+
getStatus(processId: number): ProcessStatus;
|
|
23
|
+
getAllStatuses(): Map<number, ProcessStatus>;
|
|
24
|
+
killAll(): void;
|
|
25
|
+
}
|
|
26
|
+
//# sourceMappingURL=process-manager.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"process-manager.d.ts","sourceRoot":"","sources":["../src/process-manager.ts"],"names":[],"mappings":"AAAA,OAAO,EAAS,YAAY,EAAE,MAAM,eAAe,CAAC;AACpD,OAAO,EAAE,YAAY,EAAE,MAAM,QAAQ,CAAC;AACtC,OAAO,EAAE,WAAW,EAAE,MAAM,UAAU,CAAC;AACvC,OAAO,EAAE,SAAS,EAAE,MAAM,iBAAiB,CAAC;AAE5C,MAAM,MAAM,aAAa,GAAG,SAAS,GAAG,SAAS,GAAG,OAAO,GAAG,SAAS,CAAC;AAYxE,MAAM,WAAW,QAAQ;IACvB,SAAS,EAAE,MAAM,CAAC;IAClB,IAAI,EAAE,MAAM,CAAC;IACb,MAAM,EAAE,QAAQ,GAAG,QAAQ,CAAC;CAC7B;AAED,MAAM,WAAW,iBAAiB;IAChC,SAAS,EAAE,MAAM,CAAC;IAClB,MAAM,EAAE,aAAa,CAAC;CACvB;AAED,qBAAa,cAAe,SAAQ,YAAY;IAC9C,OAAO,CAAC,SAAS,CAA2B;IAC5C,OAAO,CAAC,SAAS,CAAY;IAC7B,OAAO,CAAC,OAAO,CAA6B;gBAEhC,SAAS,EAAE,SAAS;IAOhC,YAAY,CAAC,WAAW,EAAE,WAAW,GAAG,YAAY;IA8DpD,QAAQ,CAAC,QAAQ,EAAE,WAAW,EAAE,GAAG,IAAI;IAIvC,SAAS,CAAC,SAAS,EAAE,MAAM,GAAG,aAAa;IAK3C,cAAc,IAAI,GAAG,CAAC,MAAM,EAAE,aAAa,CAAC;IAQ5C,OAAO,IAAI,IAAI;CAchB"}
|
|
@@ -0,0 +1,97 @@
|
|
|
1
|
+
import { spawn } from 'child_process';
|
|
2
|
+
import { EventEmitter } from 'events';
|
|
3
|
+
export class ProcessManager extends EventEmitter {
|
|
4
|
+
processes;
|
|
5
|
+
logBuffer;
|
|
6
|
+
buffers;
|
|
7
|
+
constructor(logBuffer) {
|
|
8
|
+
super();
|
|
9
|
+
this.processes = new Map();
|
|
10
|
+
this.logBuffer = logBuffer;
|
|
11
|
+
this.buffers = new Map();
|
|
12
|
+
}
|
|
13
|
+
startCommand(commandInfo) {
|
|
14
|
+
const { id, name, command } = commandInfo;
|
|
15
|
+
const parts = command.split(/\s+/);
|
|
16
|
+
const cmd = parts[0];
|
|
17
|
+
const args = parts.slice(1);
|
|
18
|
+
const proc = spawn(cmd, args, {
|
|
19
|
+
shell: true,
|
|
20
|
+
stdio: ['ignore', 'pipe', 'pipe']
|
|
21
|
+
});
|
|
22
|
+
proc.stdout?.setEncoding('utf8');
|
|
23
|
+
proc.stderr?.setEncoding('utf8');
|
|
24
|
+
this.buffers.set(id, { stdout: '', stderr: '' });
|
|
25
|
+
proc.stdout?.on('data', (data) => {
|
|
26
|
+
const buffer = this.buffers.get(id);
|
|
27
|
+
if (!buffer)
|
|
28
|
+
return;
|
|
29
|
+
buffer.stdout += data.toString();
|
|
30
|
+
const lines = buffer.stdout.split('\n');
|
|
31
|
+
buffer.stdout = lines.pop() || '';
|
|
32
|
+
lines.forEach(line => {
|
|
33
|
+
if (line.length > 0) {
|
|
34
|
+
this.logBuffer.addLog(id, line, 'stdout');
|
|
35
|
+
this.emit('log', { processId: id, line, source: 'stdout' });
|
|
36
|
+
}
|
|
37
|
+
});
|
|
38
|
+
});
|
|
39
|
+
proc.stderr?.on('data', (data) => {
|
|
40
|
+
const buffer = this.buffers.get(id);
|
|
41
|
+
if (!buffer)
|
|
42
|
+
return;
|
|
43
|
+
buffer.stderr += data.toString();
|
|
44
|
+
const lines = buffer.stderr.split('\n');
|
|
45
|
+
buffer.stderr = lines.pop() || '';
|
|
46
|
+
lines.forEach(line => {
|
|
47
|
+
if (line.length > 0) {
|
|
48
|
+
this.logBuffer.addLog(id, line, 'stderr');
|
|
49
|
+
this.emit('log', { processId: id, line, source: 'stderr' });
|
|
50
|
+
}
|
|
51
|
+
});
|
|
52
|
+
});
|
|
53
|
+
proc.on('error', (err) => {
|
|
54
|
+
this.logBuffer.addLog(id, `Process error: ${err.message}`, 'stderr');
|
|
55
|
+
this.processes.set(id, { ...commandInfo, status: 'error', process: proc });
|
|
56
|
+
this.emit('status-change', { processId: id, status: 'error' });
|
|
57
|
+
});
|
|
58
|
+
proc.on('exit', (code) => {
|
|
59
|
+
this.processes.set(id, { ...commandInfo, status: code === 0 ? 'stopped' : 'error', process: proc });
|
|
60
|
+
this.emit('status-change', { processId: id, status: code === 0 ? 'stopped' : 'error' });
|
|
61
|
+
});
|
|
62
|
+
this.processes.set(id, { ...commandInfo, status: 'running', process: proc });
|
|
63
|
+
this.emit('status-change', { processId: id, status: 'running' });
|
|
64
|
+
return proc;
|
|
65
|
+
}
|
|
66
|
+
startAll(commands) {
|
|
67
|
+
commands.forEach(cmd => this.startCommand(cmd));
|
|
68
|
+
}
|
|
69
|
+
getStatus(processId) {
|
|
70
|
+
const procInfo = this.processes.get(processId);
|
|
71
|
+
return procInfo ? procInfo.status : 'unknown';
|
|
72
|
+
}
|
|
73
|
+
getAllStatuses() {
|
|
74
|
+
const statuses = new Map();
|
|
75
|
+
this.processes.forEach((procInfo, id) => {
|
|
76
|
+
statuses.set(id, procInfo.status);
|
|
77
|
+
});
|
|
78
|
+
return statuses;
|
|
79
|
+
}
|
|
80
|
+
killAll() {
|
|
81
|
+
this.processes.forEach((procInfo) => {
|
|
82
|
+
if (procInfo.process && !procInfo.process.killed) {
|
|
83
|
+
try {
|
|
84
|
+
procInfo.process.kill('SIGKILL');
|
|
85
|
+
}
|
|
86
|
+
catch (err) {
|
|
87
|
+
try {
|
|
88
|
+
procInfo.process.kill('SIGTERM');
|
|
89
|
+
}
|
|
90
|
+
catch (e) {
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
});
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
//# sourceMappingURL=process-manager.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"process-manager.js","sourceRoot":"","sources":["../src/process-manager.ts"],"names":[],"mappings":"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"}
|
package/dist/ui.d.ts
ADDED
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import { CommandInfo } from './cli.js';
|
|
2
|
+
import { ProcessManager } from './process-manager.js';
|
|
3
|
+
import { LogBuffer } from './log-buffer.js';
|
|
4
|
+
interface TUIProps {
|
|
5
|
+
commands: CommandInfo[];
|
|
6
|
+
processManager: ProcessManager;
|
|
7
|
+
logBuffer: LogBuffer;
|
|
8
|
+
}
|
|
9
|
+
export declare function TUI({ commands, processManager, logBuffer }: TUIProps): import("react/jsx-runtime").JSX.Element;
|
|
10
|
+
export declare function renderTUI(commands: CommandInfo[], processManager: ProcessManager, logBuffer: LogBuffer): void;
|
|
11
|
+
export {};
|
|
12
|
+
//# sourceMappingURL=ui.d.ts.map
|
package/dist/ui.d.ts.map
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ui.d.ts","sourceRoot":"","sources":["../src/ui.tsx"],"names":[],"mappings":"AAEA,OAAO,EAAE,WAAW,EAAE,MAAM,UAAU,CAAC;AACvC,OAAO,EAAE,cAAc,EAAE,MAAM,sBAAsB,CAAC;AACtD,OAAO,EAAE,SAAS,EAAY,MAAM,iBAAiB,CAAC;AAItD,UAAU,QAAQ;IAChB,QAAQ,EAAE,WAAW,EAAE,CAAC;IACxB,cAAc,EAAE,cAAc,CAAC;IAC/B,SAAS,EAAE,SAAS,CAAC;CACtB;AAID,wBAAgB,GAAG,CAAC,EAAE,QAAQ,EAAE,cAAc,EAAE,SAAS,EAAE,EAAE,QAAQ,2CAoLpE;AA4JD,wBAAgB,SAAS,CAAC,QAAQ,EAAE,WAAW,EAAE,EAAE,cAAc,EAAE,cAAc,EAAE,SAAS,EAAE,SAAS,QAiBtG"}
|
package/dist/ui.js
ADDED
|
@@ -0,0 +1,224 @@
|
|
|
1
|
+
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
2
|
+
import { useState, useEffect } from 'react';
|
|
3
|
+
import { render, Box, Text, useInput, useApp } from 'ink';
|
|
4
|
+
const ALL_PROCESSES_INDEX = -1;
|
|
5
|
+
export function TUI({ commands, processManager, logBuffer }) {
|
|
6
|
+
const [selectedIndex, setSelectedIndex] = useState(0);
|
|
7
|
+
const [focusedPane, setFocusedPane] = useState('sidebar');
|
|
8
|
+
const [logScrollOffset, setLogScrollOffset] = useState(0);
|
|
9
|
+
const [statuses, setStatuses] = useState(new Map());
|
|
10
|
+
const [logs, setLogs] = useState([]);
|
|
11
|
+
const { exit } = useApp();
|
|
12
|
+
const sidebarWidth = 30;
|
|
13
|
+
useEffect(() => {
|
|
14
|
+
const updateStatuses = () => {
|
|
15
|
+
const newStatuses = new Map();
|
|
16
|
+
commands.forEach(cmd => {
|
|
17
|
+
newStatuses.set(cmd.id, processManager.getStatus(cmd.id));
|
|
18
|
+
});
|
|
19
|
+
setStatuses(newStatuses);
|
|
20
|
+
};
|
|
21
|
+
const updateLogs = () => {
|
|
22
|
+
const unifiedView = selectedIndex === ALL_PROCESSES_INDEX;
|
|
23
|
+
if (unifiedView) {
|
|
24
|
+
setLogs([...logBuffer.getUnifiedLogs()]);
|
|
25
|
+
}
|
|
26
|
+
else {
|
|
27
|
+
const selectedCmd = commands[selectedIndex];
|
|
28
|
+
setLogs([...logBuffer.getLogs(selectedCmd.id)]);
|
|
29
|
+
}
|
|
30
|
+
};
|
|
31
|
+
processManager.on('status-change', updateStatuses);
|
|
32
|
+
processManager.on('log', updateLogs);
|
|
33
|
+
updateStatuses();
|
|
34
|
+
updateLogs();
|
|
35
|
+
const interval = setInterval(() => {
|
|
36
|
+
updateLogs();
|
|
37
|
+
updateStatuses();
|
|
38
|
+
}, 100);
|
|
39
|
+
return () => {
|
|
40
|
+
clearInterval(interval);
|
|
41
|
+
processManager.removeAllListeners('status-change');
|
|
42
|
+
processManager.removeAllListeners('log');
|
|
43
|
+
};
|
|
44
|
+
}, [commands, processManager, logBuffer, selectedIndex]);
|
|
45
|
+
useEffect(() => {
|
|
46
|
+
setLogScrollOffset(Infinity);
|
|
47
|
+
}, [selectedIndex]);
|
|
48
|
+
useInput((input, key) => {
|
|
49
|
+
if (key.leftArrow && focusedPane === 'main') {
|
|
50
|
+
setFocusedPane('sidebar');
|
|
51
|
+
}
|
|
52
|
+
else if (key.rightArrow && focusedPane === 'sidebar') {
|
|
53
|
+
setFocusedPane('main');
|
|
54
|
+
}
|
|
55
|
+
else if (key.upArrow) {
|
|
56
|
+
if (focusedPane === 'sidebar') {
|
|
57
|
+
if (selectedIndex === ALL_PROCESSES_INDEX) {
|
|
58
|
+
setSelectedIndex(commands.length - 1);
|
|
59
|
+
}
|
|
60
|
+
else if (selectedIndex === 0) {
|
|
61
|
+
setSelectedIndex(ALL_PROCESSES_INDEX);
|
|
62
|
+
}
|
|
63
|
+
else {
|
|
64
|
+
setSelectedIndex(selectedIndex - 1);
|
|
65
|
+
}
|
|
66
|
+
setLogScrollOffset(Infinity);
|
|
67
|
+
}
|
|
68
|
+
else {
|
|
69
|
+
setLogScrollOffset((prev) => {
|
|
70
|
+
if (prev === Infinity) {
|
|
71
|
+
const currentLogs = unifiedView ? logBuffer.getUnifiedLogs() : logBuffer.getLogs(commands[selectedIndex].id);
|
|
72
|
+
return Math.max(0, currentLogs.length - displayHeight - 1);
|
|
73
|
+
}
|
|
74
|
+
return Math.max(0, prev - 1);
|
|
75
|
+
});
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
else if (key.downArrow) {
|
|
79
|
+
if (focusedPane === 'sidebar') {
|
|
80
|
+
if (selectedIndex === ALL_PROCESSES_INDEX) {
|
|
81
|
+
setSelectedIndex(0);
|
|
82
|
+
}
|
|
83
|
+
else if (selectedIndex === commands.length - 1) {
|
|
84
|
+
setSelectedIndex(ALL_PROCESSES_INDEX);
|
|
85
|
+
}
|
|
86
|
+
else {
|
|
87
|
+
setSelectedIndex(selectedIndex + 1);
|
|
88
|
+
}
|
|
89
|
+
setLogScrollOffset(Infinity);
|
|
90
|
+
}
|
|
91
|
+
else {
|
|
92
|
+
setLogScrollOffset((prev) => {
|
|
93
|
+
if (prev === Infinity) {
|
|
94
|
+
return Infinity;
|
|
95
|
+
}
|
|
96
|
+
const currentLogs = unifiedView ? logBuffer.getUnifiedLogs() : logBuffer.getLogs(commands[selectedIndex].id);
|
|
97
|
+
const maxScroll = Math.max(0, currentLogs.length - displayHeight);
|
|
98
|
+
return Math.min(prev + 1, maxScroll);
|
|
99
|
+
});
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
else if (key.pageUp && focusedPane === 'main') {
|
|
103
|
+
setLogScrollOffset((prev) => {
|
|
104
|
+
if (prev === Infinity) {
|
|
105
|
+
const currentLogs = unifiedView ? logBuffer.getUnifiedLogs() : logBuffer.getLogs(commands[selectedIndex].id);
|
|
106
|
+
return Math.max(0, currentLogs.length - displayHeight - 10);
|
|
107
|
+
}
|
|
108
|
+
return Math.max(0, prev - 10);
|
|
109
|
+
});
|
|
110
|
+
}
|
|
111
|
+
else if (key.pageDown && focusedPane === 'main') {
|
|
112
|
+
setLogScrollOffset((prev) => {
|
|
113
|
+
if (prev === Infinity) {
|
|
114
|
+
return Infinity;
|
|
115
|
+
}
|
|
116
|
+
const currentLogs = unifiedView ? logBuffer.getUnifiedLogs() : logBuffer.getLogs(commands[selectedIndex].id);
|
|
117
|
+
const maxScroll = Math.max(0, currentLogs.length - displayHeight);
|
|
118
|
+
return Math.min(prev + 10, maxScroll);
|
|
119
|
+
});
|
|
120
|
+
}
|
|
121
|
+
else if (key.home && focusedPane === 'main') {
|
|
122
|
+
setLogScrollOffset(0);
|
|
123
|
+
}
|
|
124
|
+
else if (key.end && focusedPane === 'main') {
|
|
125
|
+
setLogScrollOffset(Infinity);
|
|
126
|
+
}
|
|
127
|
+
else if (input === 'q' || input === 'Q' || (key.ctrl && input === 'c')) {
|
|
128
|
+
processManager.killAll();
|
|
129
|
+
exit();
|
|
130
|
+
}
|
|
131
|
+
});
|
|
132
|
+
const unifiedView = selectedIndex === ALL_PROCESSES_INDEX;
|
|
133
|
+
const displayHeight = (process.stdout.rows || 24) - 1;
|
|
134
|
+
let startIndex;
|
|
135
|
+
const wasAtBottom = logScrollOffset === Infinity ||
|
|
136
|
+
(logScrollOffset >= logs.length - displayHeight && logs.length > displayHeight);
|
|
137
|
+
if (wasAtBottom) {
|
|
138
|
+
startIndex = Math.max(0, logs.length - displayHeight);
|
|
139
|
+
if (logScrollOffset !== Infinity && logs.length > 0) {
|
|
140
|
+
setLogScrollOffset(Infinity);
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
else {
|
|
144
|
+
startIndex = Math.min(logScrollOffset, Math.max(0, logs.length - displayHeight));
|
|
145
|
+
}
|
|
146
|
+
const displayLogs = logs.slice(startIndex, startIndex + displayHeight);
|
|
147
|
+
const terminalWidth = process.stdout.columns || 80;
|
|
148
|
+
const terminalHeight = process.stdout.rows || 24;
|
|
149
|
+
const contentHeight = terminalHeight - 1;
|
|
150
|
+
const helpText = focusedPane === 'sidebar'
|
|
151
|
+
? '←→: switch | q: quit'
|
|
152
|
+
: '←→: switch | ↑↓: scroll | PageUp/Down: 10 lines | Home/End: top/bottom | q: quit';
|
|
153
|
+
return (_jsxs(Box, { flexDirection: "column", width: terminalWidth, height: terminalHeight, children: [_jsxs(Box, { flexDirection: "row", width: terminalWidth, height: contentHeight, children: [_jsx(Sidebar, { width: sidebarWidth, height: contentHeight, commands: commands, selectedIndex: selectedIndex, statuses: statuses, focusedPane: focusedPane }), _jsx(Separator, { height: contentHeight }), _jsx(MainPane, { width: terminalWidth - sidebarWidth - 1, height: contentHeight, unifiedView: unifiedView, selectedCommand: unifiedView ? null : commands[selectedIndex], logs: displayLogs, focusedPane: focusedPane, commands: commands, logScrollOffset: logScrollOffset, totalLogs: logs.length, displayHeight: displayHeight })] }), _jsx(Box, { width: terminalWidth, height: 1, children: _jsx(Text, { backgroundColor: "#585858", color: "#ffffff", children: helpText.padEnd(terminalWidth) }) })] }));
|
|
154
|
+
}
|
|
155
|
+
function Sidebar({ width, height, commands, selectedIndex, statuses, focusedPane }) {
|
|
156
|
+
const headerBg = focusedPane === 'sidebar' ? '#0055ff' : '#585858';
|
|
157
|
+
const headerFg = '#ffffff';
|
|
158
|
+
const headerText = focusedPane === 'sidebar' ? '▶ Commands ' : ' Commands ';
|
|
159
|
+
const availableHeight = height - 1;
|
|
160
|
+
let renderIndex = 0;
|
|
161
|
+
const isAllProcessesSelected = selectedIndex === ALL_PROCESSES_INDEX;
|
|
162
|
+
const allProcessesBg = isAllProcessesSelected ? '#eeeeee' : undefined;
|
|
163
|
+
const allProcessesFg = isAllProcessesSelected ? '#000000' : '#d7d7d7';
|
|
164
|
+
const allProcessesDot = isAllProcessesSelected ? '• ' : ' ';
|
|
165
|
+
const allProcessesName = 'All processes'.substring(0, width - 8);
|
|
166
|
+
const allProcessesNamePadded = allProcessesName.padEnd(width - 8);
|
|
167
|
+
const maxCommandsToShow = availableHeight - renderIndex;
|
|
168
|
+
let startCmdIndex = 0;
|
|
169
|
+
if (selectedIndex !== ALL_PROCESSES_INDEX && selectedIndex >= maxCommandsToShow) {
|
|
170
|
+
startCmdIndex = selectedIndex - maxCommandsToShow + 1;
|
|
171
|
+
}
|
|
172
|
+
const visibleCommands = commands.slice(startCmdIndex, startCmdIndex + maxCommandsToShow);
|
|
173
|
+
const helpText = focusedPane === 'sidebar'
|
|
174
|
+
? '←→: switch | q: quit'
|
|
175
|
+
: '←→: switch | ↑↓: scroll | Home/End: top/bottom | q: quit';
|
|
176
|
+
return (_jsxs(Box, { flexDirection: "column", width: width, height: height, children: [_jsx(Box, { width: width, height: 1, children: _jsx(Text, { backgroundColor: headerBg, color: headerFg, bold: focusedPane === 'sidebar', children: headerText.padEnd(width) }) }), _jsxs(Box, { width: width, height: 1, children: [_jsx(Text, { backgroundColor: allProcessesBg, color: allProcessesFg, bold: isAllProcessesSelected, children: allProcessesDot }), _jsx(Text, { backgroundColor: allProcessesBg, color: allProcessesFg, bold: isAllProcessesSelected, children: allProcessesNamePadded }), _jsx(Text, { backgroundColor: allProcessesBg, color: "#d7d7d7", children: ' ' }), _jsx(Text, { backgroundColor: allProcessesBg, color: "#d7d7d7", children: "---" })] }), visibleCommands.map((cmd, idx) => {
|
|
177
|
+
const actualIndex = startCmdIndex + idx;
|
|
178
|
+
const status = statuses.get(cmd.id) || 'unknown';
|
|
179
|
+
const isSelected = actualIndex === selectedIndex;
|
|
180
|
+
const itemBg = isSelected ? '#eeeeee' : undefined;
|
|
181
|
+
const itemFg = isSelected ? '#000000' : '#d7d7d7';
|
|
182
|
+
const dotText = isSelected ? '• ' : ' ';
|
|
183
|
+
const name = cmd.name.substring(0, width - 8);
|
|
184
|
+
const statusText = status === 'running' ? 'UP' : 'DOWN';
|
|
185
|
+
const statusColor = status === 'running' ? '#00ff00' : '#ff0000';
|
|
186
|
+
return (_jsxs(Box, { width: width, height: 1, children: [_jsx(Text, { backgroundColor: itemBg, color: itemFg, bold: isSelected, children: dotText }), _jsx(Text, { backgroundColor: itemBg, color: itemFg, bold: isSelected, children: name.padEnd(width - 8) }), _jsx(Text, { backgroundColor: itemBg, color: statusColor, children: ' ' + statusText })] }, cmd.id));
|
|
187
|
+
})] }));
|
|
188
|
+
}
|
|
189
|
+
function Separator({ height }) {
|
|
190
|
+
return (_jsx(Box, { flexDirection: "column", width: 1, height: height, children: Array.from({ length: height }, (_, i) => (_jsx(Text, { color: "#585858", children: "\u2502" }, i))) }));
|
|
191
|
+
}
|
|
192
|
+
function MainPane({ width, height, unifiedView, selectedCommand, logs, focusedPane, commands, logScrollOffset, totalLogs, displayHeight }) {
|
|
193
|
+
const title = unifiedView ? ' All Logs ' : ` ${selectedCommand?.name || ''} - ${selectedCommand?.command || ''} `;
|
|
194
|
+
const headerBg = focusedPane === 'main' ? '#0055ff' : '#585858';
|
|
195
|
+
const headerFg = '#ffffff';
|
|
196
|
+
const headerText = focusedPane === 'main' ? '▶' + title : ' ' + title;
|
|
197
|
+
const isScrolledUp = logScrollOffset !== Infinity;
|
|
198
|
+
const hasMoreBelow = isScrolledUp && totalLogs > displayHeight && logScrollOffset < totalLogs - displayHeight;
|
|
199
|
+
const logAreaHeight = height - 1 - (hasMoreBelow ? 1 : 0);
|
|
200
|
+
return (_jsxs(Box, { flexDirection: "column", width: width, height: height, children: [_jsx(Box, { width: width, height: 1, children: _jsx(Text, { backgroundColor: headerBg, color: headerFg, bold: focusedPane === 'main', children: headerText.padEnd(width) }) }), _jsx(Box, { flexDirection: "column", width: width, height: logAreaHeight, children: logs.slice(0, logAreaHeight).map((log, i) => {
|
|
201
|
+
let line = log.line;
|
|
202
|
+
if (unifiedView && log.processId !== undefined) {
|
|
203
|
+
const cmd = commands.find(c => c.id === log.processId);
|
|
204
|
+
const prefix = `[${cmd ? cmd.name : log.processId}] `;
|
|
205
|
+
line = prefix + line;
|
|
206
|
+
}
|
|
207
|
+
const lineColor = log.source === 'stderr' ? '#ff0000' : '#ffffff';
|
|
208
|
+
const truncated = line.substring(0, width);
|
|
209
|
+
return (_jsx(Text, { color: lineColor, children: truncated.padEnd(width) }, i));
|
|
210
|
+
}) }), hasMoreBelow && (_jsx(Box, { width: width, height: 1, children: _jsx(Text, { backgroundColor: "#ffaa00", color: "#000000", bold: true, children: ' ▼ More logs below - Press END to jump to bottom '.padEnd(width) }) }))] }));
|
|
211
|
+
}
|
|
212
|
+
export function renderTUI(commands, processManager, logBuffer) {
|
|
213
|
+
processManager.startAll(commands);
|
|
214
|
+
const cleanup = () => {
|
|
215
|
+
processManager.killAll();
|
|
216
|
+
};
|
|
217
|
+
process.on('SIGINT', cleanup);
|
|
218
|
+
process.on('SIGTERM', cleanup);
|
|
219
|
+
const { waitUntilExit } = render(_jsx(TUI, { commands: commands, processManager: processManager, logBuffer: logBuffer }));
|
|
220
|
+
waitUntilExit().then(() => {
|
|
221
|
+
cleanup();
|
|
222
|
+
});
|
|
223
|
+
}
|
|
224
|
+
//# sourceMappingURL=ui.js.map
|
package/dist/ui.js.map
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ui.js","sourceRoot":"","sources":["../src/ui.tsx"],"names":[],"mappings":";AAAA,OAAO,EAAE,QAAQ,EAAE,SAAS,EAAE,MAAM,OAAO,CAAC;AAC5C,OAAO,EAAE,MAAM,EAAE,GAAG,EAAE,IAAI,EAAE,QAAQ,EAAE,MAAM,EAAE,MAAM,KAAK,CAAC;AAa1D,MAAM,mBAAmB,GAAG,CAAC,CAAC,CAAC;AAE/B,MAAM,UAAU,GAAG,CAAC,EAAE,QAAQ,EAAE,cAAc,EAAE,SAAS,EAAY;IACnE,MAAM,CAAC,aAAa,EAAE,gBAAgB,CAAC,GAAG,QAAQ,CAAC,CAAC,CAAC,CAAC;IACtD,MAAM,CAAC,WAAW,EAAE,cAAc,CAAC,GAAG,QAAQ,CAAY,SAAS,CAAC,CAAC;IACrE,MAAM,CAAC,eAAe,EAAE,kBAAkB,CAAC,GAAG,QAAQ,CAAC,CAAC,CAAC,CAAC;IAC1D,MAAM,CAAC,QAAQ,EAAE,WAAW,CAAC,GAAG,QAAQ,CAAsB,IAAI,GAAG,EAAE,CAAC,CAAC;IACzE,MAAM,CAAC,IAAI,EAAE,OAAO,CAAC,GAAG,QAAQ,CAAa,EAAE,CAAC,CAAC;IACjD,MAAM,EAAE,IAAI,EAAE,GAAG,MAAM,EAAE,CAAC;IAE1B,MAAM,YAAY,GAAG,EAAE,CAAC;IAExB,SAAS,CAAC,GAAG,EAAE;QACb,MAAM,cAAc,GAAG,GAAG,EAAE;YAC1B,MAAM,WAAW,GAAG,IAAI,GAAG,EAAkB,CAAC;YAC9C,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE;gBACrB,WAAW,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,EAAE,cAAc,CAAC,SAAS,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC;YAC5D,CAAC,CAAC,CAAC;YACH,WAAW,CAAC,WAAW,CAAC,CAAC;QAC3B,CAAC,CAAC;QAEF,MAAM,UAAU,GAAG,GAAG,EAAE;YACtB,MAAM,WAAW,GAAG,aAAa,KAAK,mBAAmB,CAAC;YAC1D,IAAI,WAAW,EAAE,CAAC;gBAChB,OAAO,CAAC,CAAC,GAAG,SAAS,CAAC,cAAc,EAAE,CAAC,CAAC,CAAC;YAC3C,CAAC;iBAAM,CAAC;gBACN,MAAM,WAAW,GAAG,QAAQ,CAAC,aAAa,CAAC,CAAC;gBAC5C,OAAO,CAAC,CAAC,GAAG,SAAS,CAAC,OAAO,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;YAClD,CAAC;QACH,CAAC,CAAC;QAEF,cAAc,CAAC,EAAE,CAAC,eAAe,EAAE,cAAc,CAAC,CAAC;QACnD,cAAc,CAAC,EAAE,CAAC,KAAK,EAAE,UAAU,CAAC,CAAC;QAErC,cAAc,EAAE,CAAC;QACjB,UAAU,EAAE,CAAC;QAEb,MAAM,QAAQ,GAAG,WAAW,CAAC,GAAG,EAAE;YAChC,UAAU,EAAE,CAAC;YACb,cAAc,EAAE,CAAC;QACnB,CAAC,EAAE,GAAG,CAAC,CAAC;QAER,OAAO,GAAG,EAAE;YACV,aAAa,CAAC,QAAQ,CAAC,CAAC;YACxB,cAAc,CAAC,kBAAkB,CAAC,eAAe,CAAC,CAAC;YACnD,cAAc,CAAC,kBAAkB,CAAC,KAAK,CAAC,CAAC;QAC3C,CAAC,CAAC;IACJ,CAAC,EAAE,CAAC,QAAQ,EAAE,cAAc,EAAE,SAAS,EAAE,aAAa,CAAC,CAAC,CAAC;IAEzD,SAAS,CAAC,GAAG,EAAE;QACb,kBAAkB,CAAC,QAAQ,CAAC,CAAC;IAC/B,CAAC,EAAE,CAAC,aAAa,CAAC,CAAC,CAAC;IAEpB,QAAQ,CAAC,CAAC,KAAa,EAAE,GAAQ,EAAE,EAAE;QACnC,IAAI,GAAG,CAAC,SAAS,IAAI,WAAW,KAAK,MAAM,EAAE,CAAC;YAC5C,cAAc,CAAC,SAAS,CAAC,CAAC;QAC5B,CAAC;aAAM,IAAI,GAAG,CAAC,UAAU,IAAI,WAAW,KAAK,SAAS,EAAE,CAAC;YACvD,cAAc,CAAC,MAAM,CAAC,CAAC;QACzB,CAAC;aAAM,IAAI,GAAG,CAAC,OAAO,EAAE,CAAC;YACvB,IAAI,WAAW,KAAK,SAAS,EAAE,CAAC;gBAC9B,IAAI,aAAa,KAAK,mBAAmB,EAAE,CAAC;oBAC1C,gBAAgB,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;gBACxC,CAAC;qBAAM,IAAI,aAAa,KAAK,CAAC,EAAE,CAAC;oBAC/B,gBAAgB,CAAC,mBAAmB,CAAC,CAAC;gBACxC,CAAC;qBAAM,CAAC;oBACN,gBAAgB,CAAC,aAAa,GAAG,CAAC,CAAC,CAAC;gBACtC,CAAC;gBACD,kBAAkB,CAAC,QAAQ,CAAC,CAAC;YAC/B,CAAC;iBAAM,CAAC;gBACN,kBAAkB,CAAC,CAAC,IAAY,EAAE,EAAE;oBAClC,IAAI,IAAI,KAAK,QAAQ,EAAE,CAAC;wBACtB,MAAM,WAAW,GAAG,WAAW,CAAC,CAAC,CAAC,SAAS,CAAC,cAAc,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,OAAO,CAAC,QAAQ,CAAC,aAAa,CAAC,CAAC,EAAE,CAAC,CAAC;wBAC7G,OAAO,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,WAAW,CAAC,MAAM,GAAG,aAAa,GAAG,CAAC,CAAC,CAAC;oBAC7D,CAAC;oBACD,OAAO,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,GAAG,CAAC,CAAC,CAAC;gBAC/B,CAAC,CAAC,CAAC;YACL,CAAC;QACH,CAAC;aAAM,IAAI,GAAG,CAAC,SAAS,EAAE,CAAC;YACzB,IAAI,WAAW,KAAK,SAAS,EAAE,CAAC;gBAC9B,IAAI,aAAa,KAAK,mBAAmB,EAAE,CAAC;oBAC1C,gBAAgB,CAAC,CAAC,CAAC,CAAC;gBACtB,CAAC;qBAAM,IAAI,aAAa,KAAK,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;oBACjD,gBAAgB,CAAC,mBAAmB,CAAC,CAAC;gBACxC,CAAC;qBAAM,CAAC;oBACN,gBAAgB,CAAC,aAAa,GAAG,CAAC,CAAC,CAAC;gBACtC,CAAC;gBACD,kBAAkB,CAAC,QAAQ,CAAC,CAAC;YAC/B,CAAC;iBAAM,CAAC;gBACN,kBAAkB,CAAC,CAAC,IAAY,EAAE,EAAE;oBAClC,IAAI,IAAI,KAAK,QAAQ,EAAE,CAAC;wBACtB,OAAO,QAAQ,CAAC;oBAClB,CAAC;oBACD,MAAM,WAAW,GAAG,WAAW,CAAC,CAAC,CAAC,SAAS,CAAC,cAAc,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,OAAO,CAAC,QAAQ,CAAC,aAAa,CAAC,CAAC,EAAE,CAAC,CAAC;oBAC7G,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,WAAW,CAAC,MAAM,GAAG,aAAa,CAAC,CAAC;oBAClE,OAAO,IAAI,CAAC,GAAG,CAAC,IAAI,GAAG,CAAC,EAAE,SAAS,CAAC,CAAC;gBACvC,CAAC,CAAC,CAAC;YACL,CAAC;QACH,CAAC;aAAM,IAAI,GAAG,CAAC,MAAM,IAAI,WAAW,KAAK,MAAM,EAAE,CAAC;YAChD,kBAAkB,CAAC,CAAC,IAAY,EAAE,EAAE;gBAClC,IAAI,IAAI,KAAK,QAAQ,EAAE,CAAC;oBACtB,MAAM,WAAW,GAAG,WAAW,CAAC,CAAC,CAAC,SAAS,CAAC,cAAc,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,OAAO,CAAC,QAAQ,CAAC,aAAa,CAAC,CAAC,EAAE,CAAC,CAAC;oBAC7G,OAAO,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,WAAW,CAAC,MAAM,GAAG,aAAa,GAAG,EAAE,CAAC,CAAC;gBAC9D,CAAC;gBACD,OAAO,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,GAAG,EAAE,CAAC,CAAC;YAChC,CAAC,CAAC,CAAC;QACL,CAAC;aAAM,IAAI,GAAG,CAAC,QAAQ,IAAI,WAAW,KAAK,MAAM,EAAE,CAAC;YAClD,kBAAkB,CAAC,CAAC,IAAY,EAAE,EAAE;gBAClC,IAAI,IAAI,KAAK,QAAQ,EAAE,CAAC;oBACtB,OAAO,QAAQ,CAAC;gBAClB,CAAC;gBACD,MAAM,WAAW,GAAG,WAAW,CAAC,CAAC,CAAC,SAAS,CAAC,cAAc,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,OAAO,CAAC,QAAQ,CAAC,aAAa,CAAC,CAAC,EAAE,CAAC,CAAC;gBAC7G,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,WAAW,CAAC,MAAM,GAAG,aAAa,CAAC,CAAC;gBAClE,OAAO,IAAI,CAAC,GAAG,CAAC,IAAI,GAAG,EAAE,EAAE,SAAS,CAAC,CAAC;YACxC,CAAC,CAAC,CAAC;QACL,CAAC;aAAM,IAAI,GAAG,CAAC,IAAI,IAAI,WAAW,KAAK,MAAM,EAAE,CAAC;YAC9C,kBAAkB,CAAC,CAAC,CAAC,CAAC;QACxB,CAAC;aAAM,IAAI,GAAG,CAAC,GAAG,IAAI,WAAW,KAAK,MAAM,EAAE,CAAC;YAC7C,kBAAkB,CAAC,QAAQ,CAAC,CAAC;QAC/B,CAAC;aAAM,IAAI,KAAK,KAAK,GAAG,IAAI,KAAK,KAAK,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,IAAI,KAAK,KAAK,GAAG,CAAC,EAAE,CAAC;YACzE,cAAc,CAAC,OAAO,EAAE,CAAC;YACzB,IAAI,EAAE,CAAC;QACT,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,MAAM,WAAW,GAAG,aAAa,KAAK,mBAAmB,CAAC;IAC1D,MAAM,aAAa,GAAG,CAAC,OAAO,CAAC,MAAM,CAAC,IAAI,IAAI,EAAE,CAAC,GAAG,CAAC,CAAC;IACtD,IAAI,UAAkB,CAAC;IAEvB,MAAM,WAAW,GAAG,eAAe,KAAK,QAAQ;QAC9C,CAAC,eAAe,IAAI,IAAI,CAAC,MAAM,GAAG,aAAa,IAAI,IAAI,CAAC,MAAM,GAAG,aAAa,CAAC,CAAC;IAElF,IAAI,WAAW,EAAE,CAAC;QAChB,UAAU,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,MAAM,GAAG,aAAa,CAAC,CAAC;QACtD,IAAI,eAAe,KAAK,QAAQ,IAAI,IAAI,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACpD,kBAAkB,CAAC,QAAQ,CAAC,CAAC;QAC/B,CAAC;IACH,CAAC;SAAM,CAAC;QACN,UAAU,GAAG,IAAI,CAAC,GAAG,CAAC,eAAe,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,MAAM,GAAG,aAAa,CAAC,CAAC,CAAC;IACnF,CAAC;IAED,MAAM,WAAW,GAAG,IAAI,CAAC,KAAK,CAAC,UAAU,EAAE,UAAU,GAAG,aAAa,CAAC,CAAC;IAEvE,MAAM,aAAa,GAAG,OAAO,CAAC,MAAM,CAAC,OAAO,IAAI,EAAE,CAAC;IACnD,MAAM,cAAc,GAAG,OAAO,CAAC,MAAM,CAAC,IAAI,IAAI,EAAE,CAAC;IACjD,MAAM,aAAa,GAAG,cAAc,GAAG,CAAC,CAAC;IAEzC,MAAM,QAAQ,GAAG,WAAW,KAAK,SAAS;QACxC,CAAC,CAAC,sBAAsB;QACxB,CAAC,CAAC,kFAAkF,CAAC;IAEvF,OAAO,CACL,MAAC,GAAG,IAAC,aAAa,EAAC,QAAQ,EAAC,KAAK,EAAE,aAAa,EAAE,MAAM,EAAE,cAAc,aACtE,MAAC,GAAG,IAAC,aAAa,EAAC,KAAK,EAAC,KAAK,EAAE,aAAa,EAAE,MAAM,EAAE,aAAa,aAClE,KAAC,OAAO,IACN,KAAK,EAAE,YAAY,EACnB,MAAM,EAAE,aAAa,EACrB,QAAQ,EAAE,QAAQ,EAClB,aAAa,EAAE,aAAa,EAC5B,QAAQ,EAAE,QAAQ,EAClB,WAAW,EAAE,WAAW,GACxB,EACF,KAAC,SAAS,IAAC,MAAM,EAAE,aAAa,GAAI,EACtC,KAAC,QAAQ,IACP,KAAK,EAAE,aAAa,GAAG,YAAY,GAAG,CAAC,EACvC,MAAM,EAAE,aAAa,EACrB,WAAW,EAAE,WAAW,EACxB,eAAe,EAAE,WAAW,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,QAAQ,CAAC,aAAa,CAAC,EAC7D,IAAI,EAAE,WAAW,EACjB,WAAW,EAAE,WAAW,EACxB,QAAQ,EAAE,QAAQ,EAClB,eAAe,EAAE,eAAe,EAChC,SAAS,EAAE,IAAI,CAAC,MAAM,EACtB,aAAa,EAAE,aAAa,GAC5B,IACI,EACN,KAAC,GAAG,IAAC,KAAK,EAAE,aAAa,EAAE,MAAM,EAAE,CAAC,YAClC,KAAC,IAAI,IAAC,eAAe,EAAC,SAAS,EAAC,KAAK,EAAC,SAAS,YAC5C,QAAQ,CAAC,MAAM,CAAC,aAAa,CAAC,GAC1B,GACH,IACF,CACP,CAAC;AACJ,CAAC;AAWD,SAAS,OAAO,CAAC,EAAE,KAAK,EAAE,MAAM,EAAE,QAAQ,EAAE,aAAa,EAAE,QAAQ,EAAE,WAAW,EAAgB;IAC9F,MAAM,QAAQ,GAAG,WAAW,KAAK,SAAS,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,SAAS,CAAC;IACnE,MAAM,QAAQ,GAAG,SAAS,CAAC;IAC3B,MAAM,UAAU,GAAG,WAAW,KAAK,SAAS,CAAC,CAAC,CAAC,aAAa,CAAC,CAAC,CAAC,aAAa,CAAC;IAE7E,MAAM,eAAe,GAAG,MAAM,GAAG,CAAC,CAAC;IACnC,IAAI,WAAW,GAAG,CAAC,CAAC;IAEpB,MAAM,sBAAsB,GAAG,aAAa,KAAK,mBAAmB,CAAC;IACrE,MAAM,cAAc,GAAG,sBAAsB,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,SAAS,CAAC;IACtE,MAAM,cAAc,GAAG,sBAAsB,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,SAAS,CAAC;IACtE,MAAM,eAAe,GAAG,sBAAsB,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC;IAC7D,MAAM,gBAAgB,GAAG,eAAe,CAAC,SAAS,CAAC,CAAC,EAAE,KAAK,GAAG,CAAC,CAAC,CAAC;IACjE,MAAM,sBAAsB,GAAG,gBAAgB,CAAC,MAAM,CAAC,KAAK,GAAG,CAAC,CAAC,CAAC;IAElE,MAAM,iBAAiB,GAAG,eAAe,GAAG,WAAW,CAAC;IACxD,IAAI,aAAa,GAAG,CAAC,CAAC;IAEtB,IAAI,aAAa,KAAK,mBAAmB,IAAI,aAAa,IAAI,iBAAiB,EAAE,CAAC;QAChF,aAAa,GAAG,aAAa,GAAG,iBAAiB,GAAG,CAAC,CAAC;IACxD,CAAC;IAED,MAAM,eAAe,GAAG,QAAQ,CAAC,KAAK,CAAC,aAAa,EAAE,aAAa,GAAG,iBAAiB,CAAC,CAAC;IAEzF,MAAM,QAAQ,GAAG,WAAW,KAAK,SAAS;QACxC,CAAC,CAAC,sBAAsB;QACxB,CAAC,CAAC,0DAA0D,CAAC;IAE/D,OAAO,CACL,MAAC,GAAG,IAAC,aAAa,EAAC,QAAQ,EAAC,KAAK,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,aACtD,KAAC,GAAG,IAAC,KAAK,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC,YAC1B,KAAC,IAAI,IAAC,eAAe,EAAE,QAAQ,EAAE,KAAK,EAAE,QAAQ,EAAE,IAAI,EAAE,WAAW,KAAK,SAAS,YAC9E,UAAU,CAAC,MAAM,CAAC,KAAK,CAAC,GACpB,GACH,EACN,MAAC,GAAG,IAAC,KAAK,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC,aAC1B,KAAC,IAAI,IAAC,eAAe,EAAE,cAAc,EAAE,KAAK,EAAE,cAAc,EAAE,IAAI,EAAE,sBAAsB,YACvF,eAAe,GACX,EACP,KAAC,IAAI,IAAC,eAAe,EAAE,cAAc,EAAE,KAAK,EAAE,cAAc,EAAE,IAAI,EAAE,sBAAsB,YACvF,sBAAsB,GAClB,EACP,KAAC,IAAI,IAAC,eAAe,EAAE,cAAc,EAAE,KAAK,EAAC,SAAS,YACnD,GAAG,GACC,EACP,KAAC,IAAI,IAAC,eAAe,EAAE,cAAc,EAAE,KAAK,EAAC,SAAS,oBAAW,IAC7D,EACL,eAAe,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,GAAG,EAAE,EAAE;gBAChC,MAAM,WAAW,GAAG,aAAa,GAAG,GAAG,CAAC;gBACxC,MAAM,MAAM,GAAG,QAAQ,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,SAAS,CAAC;gBACjD,MAAM,UAAU,GAAG,WAAW,KAAK,aAAa,CAAC;gBACjD,MAAM,MAAM,GAAG,UAAU,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,SAAS,CAAC;gBAClD,MAAM,MAAM,GAAG,UAAU,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,SAAS,CAAC;gBAClD,MAAM,OAAO,GAAG,UAAU,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC;gBACzC,MAAM,IAAI,GAAG,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,EAAE,KAAK,GAAG,CAAC,CAAC,CAAC;gBAC9C,MAAM,UAAU,GAAG,MAAM,KAAK,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC;gBACxD,MAAM,WAAW,GAAG,MAAM,KAAK,SAAS,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,SAAS,CAAC;gBAEjE,OAAO,CACL,MAAC,GAAG,IAAc,KAAK,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC,aACvC,KAAC,IAAI,IAAC,eAAe,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,IAAI,EAAE,UAAU,YAC3D,OAAO,GACH,EACP,KAAC,IAAI,IAAC,eAAe,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,IAAI,EAAE,UAAU,YAC3D,IAAI,CAAC,MAAM,CAAC,KAAK,GAAG,CAAC,CAAC,GAClB,EACP,KAAC,IAAI,IAAC,eAAe,EAAE,MAAM,EAAE,KAAK,EAAE,WAAW,YAC9C,GAAG,GAAG,UAAU,GACZ,KATC,GAAG,CAAC,EAAE,CAUV,CACP,CAAC;YACJ,CAAC,CAAC,IACE,CACP,CAAC;AACJ,CAAC;AAED,SAAS,SAAS,CAAC,EAAE,MAAM,EAAsB;IAC/C,OAAO,CACL,KAAC,GAAG,IAAC,aAAa,EAAC,QAAQ,EAAC,KAAK,EAAE,CAAC,EAAE,MAAM,EAAE,MAAM,YACjD,KAAK,CAAC,IAAI,CAAC,EAAE,MAAM,EAAE,MAAM,EAAE,EAAE,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CACxC,KAAC,IAAI,IAAS,KAAK,EAAC,SAAS,wBAAlB,CAAC,CAA0B,CACvC,CAAC,GACE,CACP,CAAC;AACJ,CAAC;AAeD,SAAS,QAAQ,CAAC,EAAE,KAAK,EAAE,MAAM,EAAE,WAAW,EAAE,eAAe,EAAE,IAAI,EAAE,WAAW,EAAE,QAAQ,EAAE,eAAe,EAAE,SAAS,EAAE,aAAa,EAAiB;IACtJ,MAAM,KAAK,GAAG,WAAW,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,IAAI,eAAe,EAAE,IAAI,IAAI,EAAE,MAAM,eAAe,EAAE,OAAO,IAAI,EAAE,GAAG,CAAC;IAClH,MAAM,QAAQ,GAAG,WAAW,KAAK,MAAM,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,SAAS,CAAC;IAChE,MAAM,QAAQ,GAAG,SAAS,CAAC;IAC3B,MAAM,UAAU,GAAG,WAAW,KAAK,MAAM,CAAC,CAAC,CAAC,GAAG,GAAG,KAAK,CAAC,CAAC,CAAC,GAAG,GAAG,KAAK,CAAC;IAEtE,MAAM,YAAY,GAAG,eAAe,KAAK,QAAQ,CAAC;IAClD,MAAM,YAAY,GAAG,YAAY,IAAI,SAAS,GAAG,aAAa,IAAI,eAAe,GAAG,SAAS,GAAG,aAAa,CAAC;IAC9G,MAAM,aAAa,GAAG,MAAM,GAAG,CAAC,GAAG,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IAE1D,OAAO,CACL,MAAC,GAAG,IAAC,aAAa,EAAC,QAAQ,EAAC,KAAK,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,aACtD,KAAC,GAAG,IAAC,KAAK,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC,YAC1B,KAAC,IAAI,IAAC,eAAe,EAAE,QAAQ,EAAE,KAAK,EAAE,QAAQ,EAAE,IAAI,EAAE,WAAW,KAAK,MAAM,YAC3E,UAAU,CAAC,MAAM,CAAC,KAAK,CAAC,GACpB,GACH,EACN,KAAC,GAAG,IAAC,aAAa,EAAC,QAAQ,EAAC,KAAK,EAAE,KAAK,EAAE,MAAM,EAAE,aAAa,YAC5D,IAAI,CAAC,KAAK,CAAC,CAAC,EAAE,aAAa,CAAC,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE;oBAC3C,IAAI,IAAI,GAAG,GAAG,CAAC,IAAI,CAAC;oBACpB,IAAI,WAAW,IAAI,GAAG,CAAC,SAAS,KAAK,SAAS,EAAE,CAAC;wBAC/C,MAAM,GAAG,GAAG,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,GAAG,CAAC,SAAS,CAAC,CAAC;wBACvD,MAAM,MAAM,GAAG,IAAI,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,GAAG,CAAC,SAAS,IAAI,CAAC;wBACtD,IAAI,GAAG,MAAM,GAAG,IAAI,CAAC;oBACvB,CAAC;oBACD,MAAM,SAAS,GAAG,GAAG,CAAC,MAAM,KAAK,QAAQ,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,SAAS,CAAC;oBAClE,MAAM,SAAS,GAAG,IAAI,CAAC,SAAS,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC;oBAE3C,OAAO,CACL,KAAC,IAAI,IAAS,KAAK,EAAE,SAAS,YAC3B,SAAS,CAAC,MAAM,CAAC,KAAK,CAAC,IADf,CAAC,CAEL,CACR,CAAC;gBACJ,CAAC,CAAC,GACE,EACL,YAAY,IAAI,CACf,KAAC,GAAG,IAAC,KAAK,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC,YAC1B,KAAC,IAAI,IAAC,eAAe,EAAC,SAAS,EAAC,KAAK,EAAC,SAAS,EAAC,IAAI,kBACjD,mDAAmD,CAAC,MAAM,CAAC,KAAK,CAAC,GAC7D,GACH,CACP,IACG,CACP,CAAC;AACJ,CAAC;AAED,MAAM,UAAU,SAAS,CAAC,QAAuB,EAAE,cAA8B,EAAE,SAAoB;IACrG,cAAc,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC;IAElC,MAAM,OAAO,GAAG,GAAG,EAAE;QACnB,cAAc,CAAC,OAAO,EAAE,CAAC;IAC3B,CAAC,CAAC;IAEF,OAAO,CAAC,EAAE,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;IAC9B,OAAO,CAAC,EAAE,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC;IAE/B,MAAM,EAAE,aAAa,EAAE,GAAG,MAAM,CAC9B,KAAC,GAAG,IAAC,QAAQ,EAAE,QAAQ,EAAE,cAAc,EAAE,cAAc,EAAE,SAAS,EAAE,SAAS,GAAI,CAClF,CAAC;IAEF,aAAa,EAAE,CAAC,IAAI,CAAC,GAAG,EAAE;QACxB,OAAO,EAAE,CAAC;IACZ,CAAC,CAAC,CAAC;AACL,CAAC"}
|
package/package.json
ADDED
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "conqr",
|
|
3
|
+
"version": "0.0.1",
|
|
4
|
+
"description": "Dead-simple TUI process runner",
|
|
5
|
+
"main": "dist/index.js",
|
|
6
|
+
"type": "module",
|
|
7
|
+
"bin": {
|
|
8
|
+
"conqr": "./dist/index.js"
|
|
9
|
+
},
|
|
10
|
+
"files": [
|
|
11
|
+
"dist",
|
|
12
|
+
"README.md"
|
|
13
|
+
],
|
|
14
|
+
"scripts": {
|
|
15
|
+
"build": "tsc",
|
|
16
|
+
"prepublishOnly": "npm run build",
|
|
17
|
+
"start": "tsx src/index.ts",
|
|
18
|
+
"dev": "tsx src/index.ts"
|
|
19
|
+
},
|
|
20
|
+
"keywords": [
|
|
21
|
+
"tui",
|
|
22
|
+
"process",
|
|
23
|
+
"runner",
|
|
24
|
+
"terminal",
|
|
25
|
+
"concurrent",
|
|
26
|
+
"monitor"
|
|
27
|
+
],
|
|
28
|
+
"author": "",
|
|
29
|
+
"license": "ISC",
|
|
30
|
+
"repository": {
|
|
31
|
+
"type": "git",
|
|
32
|
+
"url": "https://github.com/bohdan-shulha/conqr.git"
|
|
33
|
+
},
|
|
34
|
+
"homepage": "https://github.com/bohdan-shulha/conqr#readme",
|
|
35
|
+
"engines": {
|
|
36
|
+
"node": ">=18.0.0"
|
|
37
|
+
},
|
|
38
|
+
"dependencies": {
|
|
39
|
+
"ink": "^4.4.1",
|
|
40
|
+
"react": "^18.2.0"
|
|
41
|
+
},
|
|
42
|
+
"devDependencies": {
|
|
43
|
+
"@types/node": "^24.10.1",
|
|
44
|
+
"@types/react": "^19.2.4",
|
|
45
|
+
"tsx": "^4.20.6",
|
|
46
|
+
"typescript": "^5.9.3"
|
|
47
|
+
}
|
|
48
|
+
}
|