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.
@@ -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();