kumo-cli 1.0.0
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/LICENSE +21 -0
- package/README.md +130 -0
- package/dist/ansiCodes.js +107 -0
- package/dist/ansiCodes.js.map +1 -0
- package/dist/credentialsStore.js +57 -0
- package/dist/credentialsStore.js.map +1 -0
- package/dist/generateHookSettings.js +30 -0
- package/dist/generateHookSettings.js.map +1 -0
- package/dist/hookServer.js +32 -0
- package/dist/hookServer.js.map +1 -0
- package/dist/ignorePaths.js +68 -0
- package/dist/ignorePaths.js.map +1 -0
- package/dist/index.js +1159 -0
- package/dist/index.js.map +1 -0
- package/dist/loadDotEnv.js +79 -0
- package/dist/loadDotEnv.js.map +1 -0
- package/dist/portForwardManager.js +97 -0
- package/dist/portForwardManager.js.map +1 -0
- package/dist/projectHistoryReader.js +85 -0
- package/dist/projectHistoryReader.js.map +1 -0
- package/dist/ptyMenuParser.js +470 -0
- package/dist/ptyMenuParser.js.map +1 -0
- package/dist/ptySnapshotRenderer.js +78 -0
- package/dist/ptySnapshotRenderer.js.map +1 -0
- package/dist/sessionScanner.js +145 -0
- package/dist/sessionScanner.js.map +1 -0
- package/dist/snapshotScanner.js +152 -0
- package/dist/snapshotScanner.js.map +1 -0
- package/dist/spawn.js +69 -0
- package/dist/spawn.js.map +1 -0
- package/dist/terminalManager.js +73 -0
- package/dist/terminalManager.js.map +1 -0
- package/package.json +65 -0
- package/scripts/claude_launcher.cjs +71 -0
- package/scripts/session_hook_forwarder.cjs +44 -0
|
@@ -0,0 +1,145 @@
|
|
|
1
|
+
import fs from 'fs';
|
|
2
|
+
import path from 'path';
|
|
3
|
+
import os from 'os';
|
|
4
|
+
import { EventEmitter } from 'events';
|
|
5
|
+
export class SessionScanner extends EventEmitter {
|
|
6
|
+
sessionId;
|
|
7
|
+
startAtEnd;
|
|
8
|
+
filePath = null;
|
|
9
|
+
offset = 0;
|
|
10
|
+
watcher = null;
|
|
11
|
+
seenUuids = new Set();
|
|
12
|
+
pollTimer = null;
|
|
13
|
+
constructor(sessionId, startAtEnd = false) {
|
|
14
|
+
super();
|
|
15
|
+
this.sessionId = sessionId;
|
|
16
|
+
this.startAtEnd = startAtEnd;
|
|
17
|
+
}
|
|
18
|
+
start() {
|
|
19
|
+
this.filePath = this.findJsonlFile();
|
|
20
|
+
if (!this.filePath) {
|
|
21
|
+
this.pollTimer = setInterval(() => {
|
|
22
|
+
const found = this.findJsonlFile();
|
|
23
|
+
if (found) {
|
|
24
|
+
this.filePath = found;
|
|
25
|
+
clearInterval(this.pollTimer);
|
|
26
|
+
this.pollTimer = null;
|
|
27
|
+
this.watchFile();
|
|
28
|
+
}
|
|
29
|
+
}, 500);
|
|
30
|
+
return;
|
|
31
|
+
}
|
|
32
|
+
this.watchFile();
|
|
33
|
+
}
|
|
34
|
+
stop() {
|
|
35
|
+
this.watcher?.close();
|
|
36
|
+
if (this.pollTimer)
|
|
37
|
+
clearInterval(this.pollTimer);
|
|
38
|
+
}
|
|
39
|
+
findJsonlFile() {
|
|
40
|
+
const projectsDir = path.join(os.homedir(), '.claude', 'projects');
|
|
41
|
+
if (!fs.existsSync(projectsDir))
|
|
42
|
+
return null;
|
|
43
|
+
const entries = fs.readdirSync(projectsDir, { withFileTypes: true });
|
|
44
|
+
for (const entry of entries) {
|
|
45
|
+
if (!entry.isDirectory())
|
|
46
|
+
continue;
|
|
47
|
+
const candidate = path.join(projectsDir, entry.name, `${this.sessionId}.jsonl`);
|
|
48
|
+
if (fs.existsSync(candidate))
|
|
49
|
+
return candidate;
|
|
50
|
+
}
|
|
51
|
+
return null;
|
|
52
|
+
}
|
|
53
|
+
watchFile() {
|
|
54
|
+
if (!this.filePath)
|
|
55
|
+
return;
|
|
56
|
+
if (this.startAtEnd) {
|
|
57
|
+
try {
|
|
58
|
+
this.offset = fs.statSync(this.filePath).size;
|
|
59
|
+
}
|
|
60
|
+
catch {
|
|
61
|
+
this.offset = 0;
|
|
62
|
+
}
|
|
63
|
+
this.startAtEnd = false;
|
|
64
|
+
}
|
|
65
|
+
else {
|
|
66
|
+
this.readIncremental();
|
|
67
|
+
}
|
|
68
|
+
try {
|
|
69
|
+
this.watcher = fs.watch(this.filePath, () => this.readIncremental());
|
|
70
|
+
}
|
|
71
|
+
catch { }
|
|
72
|
+
}
|
|
73
|
+
readIncremental() {
|
|
74
|
+
if (!this.filePath)
|
|
75
|
+
return;
|
|
76
|
+
try {
|
|
77
|
+
const stat = fs.statSync(this.filePath);
|
|
78
|
+
if (stat.size <= this.offset)
|
|
79
|
+
return;
|
|
80
|
+
const fd = fs.openSync(this.filePath, 'r');
|
|
81
|
+
const buf = Buffer.alloc(stat.size - this.offset);
|
|
82
|
+
fs.readSync(fd, buf, 0, buf.length, this.offset);
|
|
83
|
+
fs.closeSync(fd);
|
|
84
|
+
this.offset = stat.size;
|
|
85
|
+
const lines = buf.toString('utf-8').split('\n');
|
|
86
|
+
for (const line of lines) {
|
|
87
|
+
if (!line.trim())
|
|
88
|
+
continue;
|
|
89
|
+
try {
|
|
90
|
+
const msg = JSON.parse(line);
|
|
91
|
+
this.processMessage(msg);
|
|
92
|
+
}
|
|
93
|
+
catch { }
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
catch { }
|
|
97
|
+
}
|
|
98
|
+
processMessage(msg) {
|
|
99
|
+
if (msg.type !== 'user' && msg.type !== 'assistant')
|
|
100
|
+
return;
|
|
101
|
+
if (!msg.message)
|
|
102
|
+
return;
|
|
103
|
+
if (msg.isMeta || msg.sourceToolUseID)
|
|
104
|
+
return;
|
|
105
|
+
const role = msg.message.role;
|
|
106
|
+
if (!role || role === 'system')
|
|
107
|
+
return;
|
|
108
|
+
if (msg.uuid && this.seenUuids.has(msg.uuid))
|
|
109
|
+
return;
|
|
110
|
+
if (msg.uuid)
|
|
111
|
+
this.seenUuids.add(msg.uuid);
|
|
112
|
+
let content = '';
|
|
113
|
+
const toolUses = [];
|
|
114
|
+
if (typeof msg.message.content === 'string') {
|
|
115
|
+
content = msg.message.content;
|
|
116
|
+
}
|
|
117
|
+
else if (Array.isArray(msg.message.content)) {
|
|
118
|
+
content = msg.message.content
|
|
119
|
+
.filter((b) => b.type === 'text' && b.text)
|
|
120
|
+
.map((b) => b.text ?? '')
|
|
121
|
+
.join('');
|
|
122
|
+
for (const b of msg.message.content) {
|
|
123
|
+
if (b.type === 'tool_use' && b.name) {
|
|
124
|
+
toolUses.push({ toolName: b.name, input: b.input ?? {} });
|
|
125
|
+
}
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
if (!content && toolUses.length === 0)
|
|
129
|
+
return;
|
|
130
|
+
if (content && this.isNoiseTranscript(content))
|
|
131
|
+
return;
|
|
132
|
+
const stopReason = msg.message.stop_reason;
|
|
133
|
+
this.emit('message', { role, content, id: msg.uuid, timestamp: new Date().toISOString(), toolUses: toolUses.length > 0 ? toolUses : undefined, stopReason });
|
|
134
|
+
}
|
|
135
|
+
isNoiseTranscript(content) {
|
|
136
|
+
if (/<local-command-|<\/local-command-|<command-name>|<command-message>|<command-args>|<local-command-stdout>/i.test(content))
|
|
137
|
+
return true;
|
|
138
|
+
if (/request interrupted by user|interrupted by user|interrupted the running process/i.test(content))
|
|
139
|
+
return true;
|
|
140
|
+
if (/^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}\s*$/i.test(content.trim()))
|
|
141
|
+
return true;
|
|
142
|
+
return false;
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
//# sourceMappingURL=sessionScanner.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"sessionScanner.js","sourceRoot":"","sources":["../src/sessionScanner.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,152 @@
|
|
|
1
|
+
import fs from 'fs';
|
|
2
|
+
import path from 'path';
|
|
3
|
+
import { EventEmitter } from 'events';
|
|
4
|
+
export class SnapshotScanner extends EventEmitter {
|
|
5
|
+
sessionId;
|
|
6
|
+
snapshotDir;
|
|
7
|
+
filePath = null;
|
|
8
|
+
offset = 0;
|
|
9
|
+
watcher = null;
|
|
10
|
+
seenUuids = new Set();
|
|
11
|
+
pollTimer = null;
|
|
12
|
+
constructor(sessionId, snapshotDir) {
|
|
13
|
+
super();
|
|
14
|
+
this.sessionId = sessionId;
|
|
15
|
+
this.snapshotDir = snapshotDir;
|
|
16
|
+
}
|
|
17
|
+
start() {
|
|
18
|
+
this.filePath = this.findJsonlFile();
|
|
19
|
+
if (!this.filePath) {
|
|
20
|
+
this.pollTimer = setInterval(() => {
|
|
21
|
+
const found = this.findJsonlFile();
|
|
22
|
+
if (found) {
|
|
23
|
+
this.filePath = found;
|
|
24
|
+
clearInterval(this.pollTimer);
|
|
25
|
+
this.pollTimer = null;
|
|
26
|
+
this.watchFile();
|
|
27
|
+
}
|
|
28
|
+
}, 500);
|
|
29
|
+
return;
|
|
30
|
+
}
|
|
31
|
+
this.watchFile();
|
|
32
|
+
}
|
|
33
|
+
stop() {
|
|
34
|
+
this.watcher?.close();
|
|
35
|
+
if (this.pollTimer)
|
|
36
|
+
clearInterval(this.pollTimer);
|
|
37
|
+
}
|
|
38
|
+
findJsonlFile() {
|
|
39
|
+
const projectsDir = path.join(this.snapshotDir, 'projects');
|
|
40
|
+
if (!fs.existsSync(projectsDir))
|
|
41
|
+
return null;
|
|
42
|
+
const entries = fs.readdirSync(projectsDir, { withFileTypes: true });
|
|
43
|
+
for (const entry of entries) {
|
|
44
|
+
if (!entry.isDirectory())
|
|
45
|
+
continue;
|
|
46
|
+
const candidate = path.join(projectsDir, entry.name, `${this.sessionId}.jsonl`);
|
|
47
|
+
if (fs.existsSync(candidate))
|
|
48
|
+
return candidate;
|
|
49
|
+
}
|
|
50
|
+
return null;
|
|
51
|
+
}
|
|
52
|
+
watchFile() {
|
|
53
|
+
if (!this.filePath)
|
|
54
|
+
return;
|
|
55
|
+
this.readIncremental();
|
|
56
|
+
try {
|
|
57
|
+
this.watcher = fs.watch(this.filePath, () => this.readIncremental());
|
|
58
|
+
}
|
|
59
|
+
catch { }
|
|
60
|
+
}
|
|
61
|
+
readIncremental() {
|
|
62
|
+
if (!this.filePath)
|
|
63
|
+
return;
|
|
64
|
+
try {
|
|
65
|
+
const stat = fs.statSync(this.filePath);
|
|
66
|
+
if (stat.size <= this.offset)
|
|
67
|
+
return;
|
|
68
|
+
const fd = fs.openSync(this.filePath, 'r');
|
|
69
|
+
const buf = Buffer.alloc(stat.size - this.offset);
|
|
70
|
+
fs.readSync(fd, buf, 0, buf.length, this.offset);
|
|
71
|
+
fs.closeSync(fd);
|
|
72
|
+
this.offset = stat.size;
|
|
73
|
+
const lines = buf.toString('utf-8').split('\n');
|
|
74
|
+
for (const line of lines) {
|
|
75
|
+
if (!line.trim())
|
|
76
|
+
continue;
|
|
77
|
+
try {
|
|
78
|
+
const entry = JSON.parse(line);
|
|
79
|
+
this.processEntry(entry);
|
|
80
|
+
}
|
|
81
|
+
catch { }
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
catch { }
|
|
85
|
+
}
|
|
86
|
+
processEntry(entry) {
|
|
87
|
+
if (entry.type !== 'user' && entry.type !== 'assistant')
|
|
88
|
+
return;
|
|
89
|
+
if (!entry.message)
|
|
90
|
+
return;
|
|
91
|
+
const role = entry.message.role;
|
|
92
|
+
if (!role || role === 'system')
|
|
93
|
+
return;
|
|
94
|
+
if (entry.uuid && this.seenUuids.has(entry.uuid))
|
|
95
|
+
return;
|
|
96
|
+
if (entry.uuid)
|
|
97
|
+
this.seenUuids.add(entry.uuid);
|
|
98
|
+
const content = entry.message.content;
|
|
99
|
+
if (typeof content === 'string') {
|
|
100
|
+
if (!content)
|
|
101
|
+
return;
|
|
102
|
+
if (this.isNoiseTranscript(content))
|
|
103
|
+
return;
|
|
104
|
+
this.emit('jsonl_message', { role, content, id: entry.uuid, timestamp: entry.timestamp });
|
|
105
|
+
return;
|
|
106
|
+
}
|
|
107
|
+
if (!Array.isArray(content))
|
|
108
|
+
return;
|
|
109
|
+
for (const block of content) {
|
|
110
|
+
if (block.type === 'text' && block.text) {
|
|
111
|
+
if (this.isNoiseTranscript(block.text))
|
|
112
|
+
continue;
|
|
113
|
+
this.emit('jsonl_message', {
|
|
114
|
+
role,
|
|
115
|
+
content: block.text,
|
|
116
|
+
id: entry.uuid,
|
|
117
|
+
timestamp: entry.timestamp,
|
|
118
|
+
});
|
|
119
|
+
}
|
|
120
|
+
else if (block.type === 'tool_use') {
|
|
121
|
+
const toolContent = JSON.stringify({ name: block.name, input: block.input });
|
|
122
|
+
this.emit('jsonl_message', {
|
|
123
|
+
role: 'tool_use',
|
|
124
|
+
content: toolContent,
|
|
125
|
+
id: block.id ?? entry.uuid,
|
|
126
|
+
timestamp: entry.timestamp,
|
|
127
|
+
});
|
|
128
|
+
}
|
|
129
|
+
else if (block.type === 'tool_result') {
|
|
130
|
+
const resultContent = typeof block.content === 'string'
|
|
131
|
+
? block.content
|
|
132
|
+
: JSON.stringify(block.content);
|
|
133
|
+
this.emit('jsonl_message', {
|
|
134
|
+
role: 'tool_result',
|
|
135
|
+
content: resultContent,
|
|
136
|
+
id: block.tool_use_id ?? entry.uuid,
|
|
137
|
+
timestamp: entry.timestamp,
|
|
138
|
+
});
|
|
139
|
+
}
|
|
140
|
+
}
|
|
141
|
+
}
|
|
142
|
+
isNoiseTranscript(content) {
|
|
143
|
+
if (/<local-command-|<\/local-command-|<command-name>|<command-message>|<command-args>|<local-command-stdout>/i.test(content))
|
|
144
|
+
return true;
|
|
145
|
+
if (/request interrupted by user|interrupted by user|interrupted the running process/i.test(content))
|
|
146
|
+
return true;
|
|
147
|
+
if (/^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}\s*$/i.test(content.trim()))
|
|
148
|
+
return true;
|
|
149
|
+
return false;
|
|
150
|
+
}
|
|
151
|
+
}
|
|
152
|
+
//# sourceMappingURL=snapshotScanner.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"snapshotScanner.js","sourceRoot":"","sources":["../src/snapshotScanner.ts"],"names":[],"mappings":"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"}
|
package/dist/spawn.js
ADDED
|
@@ -0,0 +1,69 @@
|
|
|
1
|
+
import { execSync } from 'child_process';
|
|
2
|
+
import os from 'os';
|
|
3
|
+
import * as pty from 'node-pty';
|
|
4
|
+
function getClaudeCmd() {
|
|
5
|
+
if (os.platform() === 'win32') {
|
|
6
|
+
try {
|
|
7
|
+
const r = execSync('where claude.cmd', { encoding: 'utf-8', stdio: ['pipe', 'pipe', 'pipe'] }).trim();
|
|
8
|
+
const first = r.split('\n')[0].trim();
|
|
9
|
+
if (first)
|
|
10
|
+
return first;
|
|
11
|
+
}
|
|
12
|
+
catch { }
|
|
13
|
+
const appData = process.env['APPDATA'] ?? '';
|
|
14
|
+
return `${appData}\\npm\\claude.cmd`;
|
|
15
|
+
}
|
|
16
|
+
try {
|
|
17
|
+
const r = execSync('which claude', { encoding: 'utf-8', stdio: ['pipe', 'pipe', 'pipe'] }).trim();
|
|
18
|
+
if (r)
|
|
19
|
+
return r.split('\n')[0].trim();
|
|
20
|
+
}
|
|
21
|
+
catch { }
|
|
22
|
+
return 'claude';
|
|
23
|
+
}
|
|
24
|
+
const CLAUDE_CMD = getClaudeCmd();
|
|
25
|
+
export function spawnClaude(args, _onThinking, onPtyData, onResize, onSpawn, cwdOverride) {
|
|
26
|
+
const shell = os.platform() === 'win32' ? 'cmd.exe' : undefined;
|
|
27
|
+
const cmdArgs = os.platform() === 'win32'
|
|
28
|
+
? ['/c', CLAUDE_CMD, ...args]
|
|
29
|
+
: args;
|
|
30
|
+
const cmd = os.platform() === 'win32' ? shell : CLAUDE_CMD;
|
|
31
|
+
const child = pty.spawn(cmd, cmdArgs, {
|
|
32
|
+
name: 'xterm-256color',
|
|
33
|
+
cols: process.stdout.columns ?? 120,
|
|
34
|
+
rows: process.stdout.rows ?? 30,
|
|
35
|
+
cwd: cwdOverride ?? process.cwd(),
|
|
36
|
+
env: process.env,
|
|
37
|
+
useConpty: process.stdout.isTTY === true,
|
|
38
|
+
});
|
|
39
|
+
onSpawn?.(child);
|
|
40
|
+
child.onData((data) => {
|
|
41
|
+
onPtyData?.(data);
|
|
42
|
+
process.stdout.write(data);
|
|
43
|
+
});
|
|
44
|
+
if (process.stdin.isTTY) {
|
|
45
|
+
process.stdin.setRawMode(true);
|
|
46
|
+
}
|
|
47
|
+
process.stdin.resume();
|
|
48
|
+
const stdinHandler = (data) => child.write(data.toString());
|
|
49
|
+
process.stdin.on('data', stdinHandler);
|
|
50
|
+
let ptyAlive = true;
|
|
51
|
+
child.onExit(() => { ptyAlive = false; });
|
|
52
|
+
process.stdout.on('resize', () => {
|
|
53
|
+
if (!ptyAlive)
|
|
54
|
+
return;
|
|
55
|
+
const cols = process.stdout.columns ?? 120;
|
|
56
|
+
const rows = process.stdout.rows ?? 30;
|
|
57
|
+
try {
|
|
58
|
+
child.resize(cols, rows);
|
|
59
|
+
onResize?.(cols, rows);
|
|
60
|
+
}
|
|
61
|
+
catch {
|
|
62
|
+
}
|
|
63
|
+
});
|
|
64
|
+
return { pty: child, cleanup: () => { process.stdin.removeListener('data', stdinHandler); } };
|
|
65
|
+
}
|
|
66
|
+
export function writeToProcess(child, text) {
|
|
67
|
+
child.write(text + '\r');
|
|
68
|
+
}
|
|
69
|
+
//# sourceMappingURL=spawn.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"spawn.js","sourceRoot":"","sources":["../src/spawn.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,73 @@
|
|
|
1
|
+
import * as pty from 'node-pty';
|
|
2
|
+
import os from 'os';
|
|
3
|
+
import { randomUUID } from 'crypto';
|
|
4
|
+
export class TerminalManager {
|
|
5
|
+
terminals = new Map();
|
|
6
|
+
onData;
|
|
7
|
+
onExit;
|
|
8
|
+
constructor(onData, onExit) {
|
|
9
|
+
this.onData = onData;
|
|
10
|
+
this.onExit = onExit;
|
|
11
|
+
}
|
|
12
|
+
create(cols = 120, rows = 30, cwd) {
|
|
13
|
+
const id = randomUUID();
|
|
14
|
+
const shell = os.platform() === 'win32' ? 'cmd.exe' : process.env.SHELL || '/bin/bash';
|
|
15
|
+
const child = pty.spawn(shell, [], {
|
|
16
|
+
name: 'xterm-256color',
|
|
17
|
+
cols,
|
|
18
|
+
rows,
|
|
19
|
+
cwd: cwd || process.cwd(),
|
|
20
|
+
env: process.env,
|
|
21
|
+
});
|
|
22
|
+
child.onData((data) => {
|
|
23
|
+
this.onData(id, data);
|
|
24
|
+
});
|
|
25
|
+
child.onExit(({ exitCode }) => {
|
|
26
|
+
this.terminals.delete(id);
|
|
27
|
+
this.onExit(id, exitCode);
|
|
28
|
+
});
|
|
29
|
+
this.terminals.set(id, { id, pty: child, cols, rows });
|
|
30
|
+
return id;
|
|
31
|
+
}
|
|
32
|
+
write(id, data) {
|
|
33
|
+
const session = this.terminals.get(id);
|
|
34
|
+
if (!session)
|
|
35
|
+
return false;
|
|
36
|
+
session.pty.write(data);
|
|
37
|
+
return true;
|
|
38
|
+
}
|
|
39
|
+
resize(id, cols, rows) {
|
|
40
|
+
const session = this.terminals.get(id);
|
|
41
|
+
if (!session)
|
|
42
|
+
return false;
|
|
43
|
+
try {
|
|
44
|
+
session.pty.resize(cols, rows);
|
|
45
|
+
session.cols = cols;
|
|
46
|
+
session.rows = rows;
|
|
47
|
+
return true;
|
|
48
|
+
}
|
|
49
|
+
catch {
|
|
50
|
+
return false;
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
close(id) {
|
|
54
|
+
const session = this.terminals.get(id);
|
|
55
|
+
if (!session)
|
|
56
|
+
return false;
|
|
57
|
+
try {
|
|
58
|
+
session.pty.kill();
|
|
59
|
+
}
|
|
60
|
+
catch { }
|
|
61
|
+
this.terminals.delete(id);
|
|
62
|
+
return true;
|
|
63
|
+
}
|
|
64
|
+
list() {
|
|
65
|
+
return Array.from(this.terminals.keys());
|
|
66
|
+
}
|
|
67
|
+
closeAll() {
|
|
68
|
+
for (const [id] of this.terminals) {
|
|
69
|
+
this.close(id);
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
//# sourceMappingURL=terminalManager.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"terminalManager.js","sourceRoot":"","sources":["../src/terminalManager.ts"],"names":[],"mappings":"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"}
|
package/package.json
ADDED
|
@@ -0,0 +1,65 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "kumo-cli",
|
|
3
|
+
"version": "1.0.0",
|
|
4
|
+
"description": "Kumo CLI — where your code drifts between devices, softly. A pairing-based companion for the Kumo app that bridges your terminal sessions to remote devices.",
|
|
5
|
+
"type": "module",
|
|
6
|
+
"bin": {
|
|
7
|
+
"kumo": "dist/index.js"
|
|
8
|
+
},
|
|
9
|
+
"main": "dist/index.js",
|
|
10
|
+
"files": [
|
|
11
|
+
"dist",
|
|
12
|
+
"scripts",
|
|
13
|
+
"README.md",
|
|
14
|
+
"LICENSE"
|
|
15
|
+
],
|
|
16
|
+
"scripts": {
|
|
17
|
+
"dev": "tsx src/index.ts",
|
|
18
|
+
"build": "tsc",
|
|
19
|
+
"start": "node dist/index.js",
|
|
20
|
+
"test": "node test/test-pty-parser.mjs",
|
|
21
|
+
"fake-ws": "node test/fake-ws.mjs",
|
|
22
|
+
"prepublishOnly": "npm run build"
|
|
23
|
+
},
|
|
24
|
+
"engines": {
|
|
25
|
+
"node": ">=18"
|
|
26
|
+
},
|
|
27
|
+
"license": "MIT",
|
|
28
|
+
"author": "Kumo",
|
|
29
|
+
"homepage": "https://github.com/kumo-app/kumo#readme",
|
|
30
|
+
"repository": {
|
|
31
|
+
"type": "git",
|
|
32
|
+
"url": "git+https://github.com/kumo-app/kumo.git",
|
|
33
|
+
"directory": "cli"
|
|
34
|
+
},
|
|
35
|
+
"bugs": {
|
|
36
|
+
"url": "https://github.com/kumo-app/kumo/issues"
|
|
37
|
+
},
|
|
38
|
+
"keywords": [
|
|
39
|
+
"kumo",
|
|
40
|
+
"cli",
|
|
41
|
+
"claude",
|
|
42
|
+
"claude-code",
|
|
43
|
+
"pty",
|
|
44
|
+
"remote",
|
|
45
|
+
"pairing",
|
|
46
|
+
"terminal",
|
|
47
|
+
"websocket"
|
|
48
|
+
],
|
|
49
|
+
"dependencies": {
|
|
50
|
+
"@xterm/headless": "^6.0.0",
|
|
51
|
+
"bonjour-service": "^1.3.0",
|
|
52
|
+
"http-proxy": "^1.18.1",
|
|
53
|
+
"node-pty": "^1.1.0",
|
|
54
|
+
"uuid": "^10.0.0",
|
|
55
|
+
"ws": "^8.18.0"
|
|
56
|
+
},
|
|
57
|
+
"devDependencies": {
|
|
58
|
+
"@types/http-proxy": "^1.17.17",
|
|
59
|
+
"@types/node": "^22.0.0",
|
|
60
|
+
"@types/uuid": "^10.0.0",
|
|
61
|
+
"@types/ws": "^8.5.12",
|
|
62
|
+
"tsx": "^4.19.2",
|
|
63
|
+
"typescript": "^5.6.0"
|
|
64
|
+
}
|
|
65
|
+
}
|
|
@@ -0,0 +1,71 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
'use strict';
|
|
3
|
+
|
|
4
|
+
const { spawn, execSync } = require('child_process');
|
|
5
|
+
const os = require('os');
|
|
6
|
+
const fs = require('fs');
|
|
7
|
+
|
|
8
|
+
const originalFetch = global.fetch;
|
|
9
|
+
|
|
10
|
+
let fd3Write = null;
|
|
11
|
+
try {
|
|
12
|
+
const stream = fs.createWriteStream(null, { fd: 3 });
|
|
13
|
+
fd3Write = (data) => { try { stream.write(data); } catch {} };
|
|
14
|
+
} catch {
|
|
15
|
+
fd3Write = null;
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
function emitFd3(event) {
|
|
19
|
+
if (fd3Write) fd3Write(JSON.stringify(event) + '\n');
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
if (typeof originalFetch === 'function') {
|
|
23
|
+
global.fetch = async function (...args) {
|
|
24
|
+
const url = typeof args[0] === 'string' ? args[0] : (args[0]?.url ?? '');
|
|
25
|
+
emitFd3({ event: 'fetch-start', url });
|
|
26
|
+
try {
|
|
27
|
+
const result = await originalFetch.apply(this, args);
|
|
28
|
+
emitFd3({ event: 'fetch-end', url });
|
|
29
|
+
return result;
|
|
30
|
+
} catch (err) {
|
|
31
|
+
emitFd3({ event: 'fetch-end', error: true });
|
|
32
|
+
throw err;
|
|
33
|
+
}
|
|
34
|
+
};
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
function getClaudeCmd() {
|
|
38
|
+
if (os.platform() === 'win32') {
|
|
39
|
+
try {
|
|
40
|
+
const r = execSync('where claude.cmd', { encoding: 'utf-8', stdio: ['pipe', 'pipe', 'pipe'] }).trim();
|
|
41
|
+
const first = r.split('\n')[0].trim();
|
|
42
|
+
if (first) return first;
|
|
43
|
+
} catch {}
|
|
44
|
+
const appData = process.env.APPDATA ?? '';
|
|
45
|
+
return require('path').join(appData, 'npm', 'claude.cmd');
|
|
46
|
+
}
|
|
47
|
+
try {
|
|
48
|
+
const r = execSync('which claude', { encoding: 'utf-8', stdio: ['pipe', 'pipe', 'pipe'] }).trim();
|
|
49
|
+
if (r) return r.split('\n')[0].trim();
|
|
50
|
+
} catch {}
|
|
51
|
+
return 'claude';
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
const claudeCmd = getClaudeCmd();
|
|
55
|
+
const args = process.argv.slice(2);
|
|
56
|
+
|
|
57
|
+
const child = spawn(claudeCmd, args, {
|
|
58
|
+
stdio: 'inherit',
|
|
59
|
+
shell: false,
|
|
60
|
+
windowsHide: false,
|
|
61
|
+
});
|
|
62
|
+
|
|
63
|
+
child.on('exit', (code, signal) => {
|
|
64
|
+
if (signal) process.kill(process.pid, signal);
|
|
65
|
+
else process.exit(code ?? 0);
|
|
66
|
+
});
|
|
67
|
+
|
|
68
|
+
child.on('error', (err) => {
|
|
69
|
+
console.error('[launcher] failed to start claude:', err.message);
|
|
70
|
+
process.exit(1);
|
|
71
|
+
});
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
'use strict';
|
|
3
|
+
|
|
4
|
+
const http = require('http');
|
|
5
|
+
const fs = require('fs');
|
|
6
|
+
|
|
7
|
+
const hookPort = parseInt(process.env.KUMO_HOOK_PORT ?? '0', 10)
|
|
8
|
+
|| (() => { const i = process.argv.indexOf('--port'); return i >= 0 ? parseInt(process.argv[i + 1], 10) : 0; })();
|
|
9
|
+
if (!hookPort) process.exit(0);
|
|
10
|
+
|
|
11
|
+
let stdin = '';
|
|
12
|
+
try {
|
|
13
|
+
stdin = fs.readFileSync(0, 'utf-8');
|
|
14
|
+
} catch {}
|
|
15
|
+
|
|
16
|
+
let sessionId = '';
|
|
17
|
+
try {
|
|
18
|
+
const input = JSON.parse(stdin);
|
|
19
|
+
sessionId = input.session_id ?? '';
|
|
20
|
+
} catch {}
|
|
21
|
+
|
|
22
|
+
if (!sessionId) {
|
|
23
|
+
sessionId = process.env.CLAUDE_SESSION_ID ?? process.argv[2] ?? '';
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
if (!sessionId) process.exit(0);
|
|
27
|
+
|
|
28
|
+
const body = JSON.stringify({ sessionId });
|
|
29
|
+
const req = http.request(
|
|
30
|
+
{
|
|
31
|
+
hostname: '127.0.0.1',
|
|
32
|
+
port: hookPort,
|
|
33
|
+
path: '/hook/session-start',
|
|
34
|
+
method: 'POST',
|
|
35
|
+
headers: { 'Content-Type': 'application/json', 'Content-Length': Buffer.byteLength(body) },
|
|
36
|
+
},
|
|
37
|
+
(res) => {
|
|
38
|
+
res.resume();
|
|
39
|
+
res.on('end', () => process.exit(0));
|
|
40
|
+
}
|
|
41
|
+
);
|
|
42
|
+
req.on('error', () => process.exit(0));
|
|
43
|
+
req.write(body);
|
|
44
|
+
req.end();
|