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 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,7 @@
1
+ export interface CommandInfo {
2
+ id: number;
3
+ name: string;
4
+ command: string;
5
+ }
6
+ export declare function parseCommands(): CommandInfo[];
7
+ //# sourceMappingURL=cli.d.ts.map
@@ -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
@@ -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"}
@@ -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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"}
@@ -0,0 +1,3 @@
1
+ #!/usr/bin/env node
2
+ export {};
3
+ //# sourceMappingURL=index.d.ts.map
@@ -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
@@ -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":"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"}
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
+ }