kumo-cli 1.0.1 → 1.0.3
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 +20 -31
- package/dist/auth/credentialsStore.js +41 -0
- package/dist/auth/credentialsStore.js.map +1 -0
- package/dist/claude/generateHookSettings.js +23 -0
- package/dist/claude/generateHookSettings.js.map +1 -0
- package/dist/claude/hookServer.js +33 -0
- package/dist/claude/hookServer.js.map +1 -0
- package/dist/claude/sessionScanner.js +250 -0
- package/dist/claude/sessionScanner.js.map +1 -0
- package/dist/config.js +40 -0
- package/dist/config.js.map +1 -0
- package/dist/core/config.js +48 -0
- package/dist/core/config.js.map +1 -0
- package/dist/core/loadDotEnv.js +46 -0
- package/dist/core/loadDotEnv.js.map +1 -0
- package/dist/core/logger.js +29 -0
- package/dist/core/logger.js.map +1 -0
- package/dist/handlers/messageRouter.js +225 -0
- package/dist/handlers/messageRouter.js.map +1 -0
- package/dist/http/httpClient.js +41 -0
- package/dist/http/httpClient.js.map +1 -0
- package/dist/index.js +120 -1364
- package/dist/index.js.map +1 -1
- package/dist/logger.js +29 -0
- package/dist/logger.js.map +1 -0
- package/dist/pty/ansiCodes.js +14 -0
- package/dist/pty/ansiCodes.js.map +1 -0
- package/dist/pty/ptyMenuParser.js +357 -0
- package/dist/pty/ptyMenuParser.js.map +1 -0
- package/dist/pty/ptySnapshotRenderer.js +71 -0
- package/dist/pty/ptySnapshotRenderer.js.map +1 -0
- package/dist/pty/spawn.js +77 -0
- package/dist/pty/spawn.js.map +1 -0
- package/dist/pty/terminalManager.js +66 -0
- package/dist/pty/terminalManager.js.map +1 -0
- package/dist/services/claudeAdapterProxy.js +104 -0
- package/dist/services/claudeAdapterProxy.js.map +1 -0
- package/dist/services/claudeService.js +239 -0
- package/dist/services/claudeService.js.map +1 -0
- package/dist/services/claudeService.test.js +39 -0
- package/dist/services/claudeService.test.js.map +1 -0
- package/dist/services/effortService.js +89 -0
- package/dist/services/effortService.js.map +1 -0
- package/dist/services/fileService.js +129 -0
- package/dist/services/fileService.js.map +1 -0
- package/dist/services/modelService.js +115 -0
- package/dist/services/modelService.js.map +1 -0
- package/dist/services/modelService.test.js +76 -0
- package/dist/services/modelService.test.js.map +1 -0
- package/dist/services/pairingService.js +129 -0
- package/dist/services/pairingService.js.map +1 -0
- package/dist/services/sessionService.js +168 -0
- package/dist/services/sessionService.js.map +1 -0
- package/dist/services/tunnelService.js +47 -0
- package/dist/services/tunnelService.js.map +1 -0
- package/dist/sessionScanner.js +130 -4
- package/dist/sessionScanner.js.map +1 -1
- package/dist/snapshotScanner.js +3 -1
- package/dist/snapshotScanner.js.map +1 -1
- package/dist/transport/directWsServer.js +135 -0
- package/dist/transport/directWsServer.js.map +1 -0
- package/dist/transport/wsClient.js +87 -0
- package/dist/transport/wsClient.js.map +1 -0
- package/dist/utils/ignorePaths.js +54 -0
- package/dist/utils/ignorePaths.js.map +1 -0
- package/dist/utils/ptyBuffer.js +46 -0
- package/dist/utils/ptyBuffer.js.map +1 -0
- package/dist/utils/safePath.js +43 -0
- package/dist/utils/safePath.js.map +1 -0
- package/package.json +6 -5
|
@@ -0,0 +1,87 @@
|
|
|
1
|
+
import WebSocket from 'ws';
|
|
2
|
+
import { log } from '../core/logger.js';
|
|
3
|
+
import { SERVER_WS } from '../core/config.js';
|
|
4
|
+
function stringifyPayload(data) {
|
|
5
|
+
try {
|
|
6
|
+
return JSON.stringify(data);
|
|
7
|
+
}
|
|
8
|
+
catch (error) {
|
|
9
|
+
return JSON.stringify({ error: error instanceof Error ? error.message : String(error) });
|
|
10
|
+
}
|
|
11
|
+
}
|
|
12
|
+
export class WsClient {
|
|
13
|
+
onOpen;
|
|
14
|
+
onMessage;
|
|
15
|
+
onClose;
|
|
16
|
+
ws = null;
|
|
17
|
+
reconnectTimer = null;
|
|
18
|
+
pingInterval = null;
|
|
19
|
+
onSendListeners = [];
|
|
20
|
+
constructor(onOpen, onMessage, onClose) {
|
|
21
|
+
this.onOpen = onOpen;
|
|
22
|
+
this.onMessage = onMessage;
|
|
23
|
+
this.onClose = onClose;
|
|
24
|
+
}
|
|
25
|
+
onSent(cb) { this.onSendListeners.push(cb); }
|
|
26
|
+
send(data) {
|
|
27
|
+
if (this.ws && this.ws.readyState === WebSocket.OPEN) {
|
|
28
|
+
const payload = stringifyPayload(data);
|
|
29
|
+
log(`[ws] outgoing ${payload}`);
|
|
30
|
+
this.ws.send(payload);
|
|
31
|
+
for (const cb of this.onSendListeners)
|
|
32
|
+
cb();
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
connect() {
|
|
36
|
+
this.ws = new WebSocket(SERVER_WS());
|
|
37
|
+
this.ws.on('open', () => {
|
|
38
|
+
log(`[ws] connected to ${SERVER_WS()}`);
|
|
39
|
+
if (this.pingInterval)
|
|
40
|
+
clearInterval(this.pingInterval);
|
|
41
|
+
this.pingInterval = setInterval(() => {
|
|
42
|
+
if (this.ws && this.ws.readyState === WebSocket.OPEN) {
|
|
43
|
+
this.ws.ping();
|
|
44
|
+
this.send({ type: 'cli_heartbeat' });
|
|
45
|
+
}
|
|
46
|
+
}, 5000);
|
|
47
|
+
Promise.resolve(this.onOpen()).catch((e) => log(`[onOpen] error: ${e}`));
|
|
48
|
+
});
|
|
49
|
+
this.ws.on('ping', () => this.ws?.pong());
|
|
50
|
+
this.ws.on('message', (raw) => {
|
|
51
|
+
const rawText = raw.toString();
|
|
52
|
+
log(`[ws] incoming ${rawText}`);
|
|
53
|
+
try {
|
|
54
|
+
this.onMessage(JSON.parse(rawText));
|
|
55
|
+
}
|
|
56
|
+
catch (error) {
|
|
57
|
+
log(`[ws] incoming parse error: ${error}`);
|
|
58
|
+
}
|
|
59
|
+
});
|
|
60
|
+
this.ws.on('close', () => {
|
|
61
|
+
log(`[ws] disconnected, reconnecting in 3s`);
|
|
62
|
+
if (this.pingInterval) {
|
|
63
|
+
clearInterval(this.pingInterval);
|
|
64
|
+
this.pingInterval = null;
|
|
65
|
+
}
|
|
66
|
+
this.onClose?.();
|
|
67
|
+
this.reconnectTimer = setTimeout(() => this.connect(), 3000);
|
|
68
|
+
});
|
|
69
|
+
this.ws.on('error', () => { });
|
|
70
|
+
}
|
|
71
|
+
close() {
|
|
72
|
+
if (this.reconnectTimer) {
|
|
73
|
+
clearTimeout(this.reconnectTimer);
|
|
74
|
+
this.reconnectTimer = null;
|
|
75
|
+
}
|
|
76
|
+
if (this.pingInterval) {
|
|
77
|
+
clearInterval(this.pingInterval);
|
|
78
|
+
this.pingInterval = null;
|
|
79
|
+
}
|
|
80
|
+
try {
|
|
81
|
+
this.ws?.close();
|
|
82
|
+
}
|
|
83
|
+
catch { }
|
|
84
|
+
this.ws = null;
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
//# sourceMappingURL=wsClient.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"wsClient.js","sourceRoot":"","sources":["../../src/transport/wsClient.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
import fs from 'node:fs';
|
|
2
|
+
import path from 'node:path';
|
|
3
|
+
// common noisy folders/files to always ignore across ecosystems
|
|
4
|
+
const DEFAULT_IGNORES = [
|
|
5
|
+
'.git', '.svn', '.hg', '.idea', '.vscode', '.DS_Store', 'Thumbs.db',
|
|
6
|
+
'node_modules', 'bower_components', 'dist', 'build', 'out', 'coverage',
|
|
7
|
+
'.nyc_output', '.next', '.nuxt', '.turbo', '.cache', '.parcel-cache',
|
|
8
|
+
'.svelte-kit', '.vercel', '.netlify', '.output', '.astro',
|
|
9
|
+
'__pycache__', '.venv', 'venv', 'env', '.pytest_cache', '.mypy_cache',
|
|
10
|
+
'.tox', '.ruff_cache', '.ipynb_checkpoints',
|
|
11
|
+
'target', '.gradle', '.mvn', '.kotlin',
|
|
12
|
+
'.dart_tool', '.pub-cache', '.pub', '.flutter-plugins',
|
|
13
|
+
'.flutter-plugins-dependencies',
|
|
14
|
+
'.expo', '.expo-shared', 'Pods', 'DerivedData',
|
|
15
|
+
'tmp', 'temp', 'logs', 'vendor', '.terraform', '.serverless',
|
|
16
|
+
];
|
|
17
|
+
let cache = null;
|
|
18
|
+
// parse a .gitignore into a set of simple name patterns (no globs)
|
|
19
|
+
function parseGitignore(content) {
|
|
20
|
+
const names = [];
|
|
21
|
+
for (const rawLine of content.split(/\r?\n/)) {
|
|
22
|
+
const line = rawLine.trim();
|
|
23
|
+
if (!line || line.startsWith('#') || line.startsWith('!'))
|
|
24
|
+
continue;
|
|
25
|
+
const p = line.replace(/^\/+/, '').replace(/\/+$/, '');
|
|
26
|
+
if (p.includes('*') || p.includes('?') || p.includes('[') || p.includes('/'))
|
|
27
|
+
continue;
|
|
28
|
+
if (p)
|
|
29
|
+
names.push(p);
|
|
30
|
+
}
|
|
31
|
+
return names;
|
|
32
|
+
}
|
|
33
|
+
// load ignored names combining defaults + .gitignore in given root, cached by mtime
|
|
34
|
+
export function loadIgnoredNames(projectRoot) {
|
|
35
|
+
const gitignore = path.join(projectRoot, '.gitignore');
|
|
36
|
+
let mtime = 0;
|
|
37
|
+
try {
|
|
38
|
+
mtime = fs.statSync(gitignore).mtimeMs;
|
|
39
|
+
}
|
|
40
|
+
catch { }
|
|
41
|
+
if (cache && cache.mtime === mtime)
|
|
42
|
+
return cache.set;
|
|
43
|
+
const set = new Set(DEFAULT_IGNORES);
|
|
44
|
+
if (mtime > 0) {
|
|
45
|
+
try {
|
|
46
|
+
for (const name of parseGitignore(fs.readFileSync(gitignore, 'utf-8')))
|
|
47
|
+
set.add(name);
|
|
48
|
+
}
|
|
49
|
+
catch { }
|
|
50
|
+
}
|
|
51
|
+
cache = { mtime, set };
|
|
52
|
+
return set;
|
|
53
|
+
}
|
|
54
|
+
//# sourceMappingURL=ignorePaths.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ignorePaths.js","sourceRoot":"","sources":["../../src/utils/ignorePaths.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,SAAS,CAAC;AACzB,OAAO,IAAI,MAAM,WAAW,CAAC;AAE7B,gEAAgE;AAChE,MAAM,eAAe,GAAa;IAChC,MAAM,EAAE,MAAM,EAAE,KAAK,EAAE,OAAO,EAAE,SAAS,EAAE,WAAW,EAAE,WAAW;IACnE,cAAc,EAAE,kBAAkB,EAAE,MAAM,EAAE,OAAO,EAAE,KAAK,EAAE,UAAU;IACtE,aAAa,EAAE,OAAO,EAAE,OAAO,EAAE,QAAQ,EAAE,QAAQ,EAAE,eAAe;IACpE,aAAa,EAAE,SAAS,EAAE,UAAU,EAAE,SAAS,EAAE,QAAQ;IACzD,aAAa,EAAE,OAAO,EAAE,MAAM,EAAE,KAAK,EAAE,eAAe,EAAE,aAAa;IACrE,MAAM,EAAE,aAAa,EAAE,oBAAoB;IAC3C,QAAQ,EAAE,SAAS,EAAE,MAAM,EAAE,SAAS;IACtC,YAAY,EAAE,YAAY,EAAE,MAAM,EAAE,kBAAkB;IACtD,+BAA+B;IAC/B,OAAO,EAAE,cAAc,EAAE,MAAM,EAAE,aAAa;IAC9C,KAAK,EAAE,MAAM,EAAE,MAAM,EAAE,QAAQ,EAAE,YAAY,EAAE,aAAa;CAC7D,CAAC;AAEF,IAAI,KAAK,GAA+C,IAAI,CAAC;AAE7D,mEAAmE;AACnE,SAAS,cAAc,CAAC,OAAe;IACrC,MAAM,KAAK,GAAa,EAAE,CAAC;IAC3B,KAAK,MAAM,OAAO,IAAI,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,EAAE,CAAC;QAC7C,MAAM,IAAI,GAAG,OAAO,CAAC,IAAI,EAAE,CAAC;QAC5B,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,IAAI,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC;YAAE,SAAS;QACpE,MAAM,CAAC,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC,OAAO,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC;QACvD,IAAI,CAAC,CAAC,QAAQ,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,QAAQ,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,QAAQ,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,QAAQ,CAAC,GAAG,CAAC;YAAE,SAAS;QACvF,IAAI,CAAC;YAAE,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IACvB,CAAC;IACD,OAAO,KAAK,CAAC;AACf,CAAC;AAED,oFAAoF;AACpF,MAAM,UAAU,gBAAgB,CAAC,WAAmB;IAClD,MAAM,SAAS,GAAG,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,YAAY,CAAC,CAAC;IACvD,IAAI,KAAK,GAAG,CAAC,CAAC;IACd,IAAI,CAAC;QAAC,KAAK,GAAG,EAAE,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC,OAAO,CAAC;IAAC,CAAC;IAAC,MAAM,CAAC,CAAA,CAAC;IACxD,IAAI,KAAK,IAAI,KAAK,CAAC,KAAK,KAAK,KAAK;QAAE,OAAO,KAAK,CAAC,GAAG,CAAC;IACrD,MAAM,GAAG,GAAG,IAAI,GAAG,CAAS,eAAe,CAAC,CAAC;IAC7C,IAAI,KAAK,GAAG,CAAC,EAAE,CAAC;QACd,IAAI,CAAC;YACH,KAAK,MAAM,IAAI,IAAI,cAAc,CAAC,EAAE,CAAC,YAAY,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC;gBAAE,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QACxF,CAAC;QAAC,MAAM,CAAC,CAAA,CAAC;IACZ,CAAC;IACD,KAAK,GAAG,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC;IACvB,OAAO,GAAG,CAAC;AACb,CAAC"}
|
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
import { PTY_FLUSH_MS, PTY_MAX_CHUNK } from '../core/config.js';
|
|
2
|
+
// buffer pty output and trigger snapshot flush either by max chunk or debounce
|
|
3
|
+
export class PtyBuffer {
|
|
4
|
+
renderer;
|
|
5
|
+
buffer = '';
|
|
6
|
+
timer = null;
|
|
7
|
+
constructor(renderer) {
|
|
8
|
+
this.renderer = renderer;
|
|
9
|
+
}
|
|
10
|
+
feed(chunk) {
|
|
11
|
+
this.buffer += chunk;
|
|
12
|
+
this.renderer.feed(chunk);
|
|
13
|
+
if (this.buffer.length >= PTY_MAX_CHUNK) {
|
|
14
|
+
this.clearTimer();
|
|
15
|
+
this.flush();
|
|
16
|
+
return;
|
|
17
|
+
}
|
|
18
|
+
if (!this.timer) {
|
|
19
|
+
this.timer = setTimeout(() => {
|
|
20
|
+
this.timer = null;
|
|
21
|
+
this.flush();
|
|
22
|
+
}, PTY_FLUSH_MS);
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
flush() {
|
|
26
|
+
if (!this.buffer)
|
|
27
|
+
return;
|
|
28
|
+
this.buffer = '';
|
|
29
|
+
this.renderer.flushSnapshot();
|
|
30
|
+
}
|
|
31
|
+
clearTimer() {
|
|
32
|
+
if (this.timer) {
|
|
33
|
+
clearTimeout(this.timer);
|
|
34
|
+
this.timer = null;
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
// strip ansi escape sequences for short, log-friendly preview
|
|
39
|
+
export function sanitizePtyPreview(data) {
|
|
40
|
+
return data
|
|
41
|
+
.replace(/\x1B\][^\x07]*(?:\x07|\x1B\\)/g, '')
|
|
42
|
+
.replace(/\x1B\[[0-?]*[ -/]*[@-~]/g, '')
|
|
43
|
+
.replace(/\s+/g, ' ')
|
|
44
|
+
.slice(0, 100);
|
|
45
|
+
}
|
|
46
|
+
//# sourceMappingURL=ptyBuffer.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ptyBuffer.js","sourceRoot":"","sources":["../../src/utils/ptyBuffer.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,YAAY,EAAE,aAAa,EAAE,MAAM,mBAAmB,CAAC;AAEhE,+EAA+E;AAC/E,MAAM,OAAO,SAAS;IAIA;IAHZ,MAAM,GAAG,EAAE,CAAC;IACZ,KAAK,GAA0B,IAAI,CAAC;IAE5C,YAAoB,QAA6B;QAA7B,aAAQ,GAAR,QAAQ,CAAqB;IAAG,CAAC;IAErD,IAAI,CAAC,KAAa;QAChB,IAAI,CAAC,MAAM,IAAI,KAAK,CAAC;QACrB,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAC1B,IAAI,IAAI,CAAC,MAAM,CAAC,MAAM,IAAI,aAAa,EAAE,CAAC;YACxC,IAAI,CAAC,UAAU,EAAE,CAAC;YAClB,IAAI,CAAC,KAAK,EAAE,CAAC;YACb,OAAO;QACT,CAAC;QACD,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC;YAChB,IAAI,CAAC,KAAK,GAAG,UAAU,CAAC,GAAG,EAAE;gBAC3B,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC;gBAClB,IAAI,CAAC,KAAK,EAAE,CAAC;YACf,CAAC,EAAE,YAAY,CAAC,CAAC;QACnB,CAAC;IACH,CAAC;IAED,KAAK;QACH,IAAI,CAAC,IAAI,CAAC,MAAM;YAAE,OAAO;QACzB,IAAI,CAAC,MAAM,GAAG,EAAE,CAAC;QACjB,IAAI,CAAC,QAAQ,CAAC,aAAa,EAAE,CAAC;IAChC,CAAC;IAEO,UAAU;QAChB,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;YACf,YAAY,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YACzB,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC;QACpB,CAAC;IACH,CAAC;CACF;AAED,8DAA8D;AAC9D,MAAM,UAAU,kBAAkB,CAAC,IAAY;IAC7C,OAAO,IAAI;SACR,OAAO,CAAC,gCAAgC,EAAE,EAAE,CAAC;SAC7C,OAAO,CAAC,0BAA0B,EAAE,EAAE,CAAC;SACvC,OAAO,CAAC,MAAM,EAAE,GAAG,CAAC;SACpB,KAAK,CAAC,CAAC,EAAE,GAAG,CAAC,CAAC;AACnB,CAAC"}
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
import fs from 'node:fs';
|
|
2
|
+
import os from 'node:os';
|
|
3
|
+
import path from 'node:path';
|
|
4
|
+
// resolve a user-supplied path safely, allowing absolute paths and blocking traversal for relative ones.
|
|
5
|
+
// handles msys2/git-bash style paths like /e/Data/... → E:\Data\... on windows.
|
|
6
|
+
export function safePath(requestedPath) {
|
|
7
|
+
let p = requestedPath || '.';
|
|
8
|
+
if (process.platform === 'win32') {
|
|
9
|
+
const msys = p.match(/^\/([a-zA-Z])\/(.*)/);
|
|
10
|
+
if (msys)
|
|
11
|
+
p = `${msys[1].toUpperCase()}:\\${msys[2].replace(/\//g, '\\')}`;
|
|
12
|
+
}
|
|
13
|
+
const normalized = path.normalize(p);
|
|
14
|
+
const projectRoot = process.cwd();
|
|
15
|
+
if (path.isAbsolute(normalized))
|
|
16
|
+
return normalized;
|
|
17
|
+
const resolved = path.resolve(projectRoot, normalized);
|
|
18
|
+
if (!resolved.startsWith(projectRoot + path.sep) && resolved !== projectRoot) {
|
|
19
|
+
throw new Error('path traversal blocked');
|
|
20
|
+
}
|
|
21
|
+
return resolved;
|
|
22
|
+
}
|
|
23
|
+
// recursively copy a file or directory tree
|
|
24
|
+
export function copyRecursive(src, dest) {
|
|
25
|
+
const stat = fs.statSync(src);
|
|
26
|
+
if (stat.isDirectory()) {
|
|
27
|
+
fs.mkdirSync(dest, { recursive: true });
|
|
28
|
+
for (const child of fs.readdirSync(src)) {
|
|
29
|
+
copyRecursive(path.join(src, child), path.join(dest, child));
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
else {
|
|
33
|
+
fs.copyFileSync(src, dest);
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
// claude-cli encodes a cwd into a folder name by replacing separators, colons and underscores with dashes
|
|
37
|
+
export function cwdToProjectFolder(cwd) {
|
|
38
|
+
if (os.platform() === 'win32') {
|
|
39
|
+
return cwd.replace(/:/g, '-').replace(/[\\/]/g, '-').replace(/_/g, '-');
|
|
40
|
+
}
|
|
41
|
+
return cwd.replace(/\//g, '-').replace(/_/g, '-');
|
|
42
|
+
}
|
|
43
|
+
//# sourceMappingURL=safePath.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"safePath.js","sourceRoot":"","sources":["../../src/utils/safePath.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,SAAS,CAAC;AACzB,OAAO,EAAE,MAAM,SAAS,CAAC;AACzB,OAAO,IAAI,MAAM,WAAW,CAAC;AAE7B,yGAAyG;AACzG,gFAAgF;AAChF,MAAM,UAAU,QAAQ,CAAC,aAAqB;IAC5C,IAAI,CAAC,GAAG,aAAa,IAAI,GAAG,CAAC;IAC7B,IAAI,OAAO,CAAC,QAAQ,KAAK,OAAO,EAAE,CAAC;QACjC,MAAM,IAAI,GAAG,CAAC,CAAC,KAAK,CAAC,qBAAqB,CAAC,CAAC;QAC5C,IAAI,IAAI;YAAE,CAAC,GAAG,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,WAAW,EAAE,MAAM,IAAI,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,KAAK,EAAE,IAAI,CAAC,EAAE,CAAC;IAC7E,CAAC;IACD,MAAM,UAAU,GAAG,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC;IACrC,MAAM,WAAW,GAAG,OAAO,CAAC,GAAG,EAAE,CAAC;IAClC,IAAI,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC;QAAE,OAAO,UAAU,CAAC;IACnD,MAAM,QAAQ,GAAG,IAAI,CAAC,OAAO,CAAC,WAAW,EAAE,UAAU,CAAC,CAAC;IACvD,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,WAAW,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,QAAQ,KAAK,WAAW,EAAE,CAAC;QAC7E,MAAM,IAAI,KAAK,CAAC,wBAAwB,CAAC,CAAC;IAC5C,CAAC;IACD,OAAO,QAAQ,CAAC;AAClB,CAAC;AAED,4CAA4C;AAC5C,MAAM,UAAU,aAAa,CAAC,GAAW,EAAE,IAAY;IACrD,MAAM,IAAI,GAAG,EAAE,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC;IAC9B,IAAI,IAAI,CAAC,WAAW,EAAE,EAAE,CAAC;QACvB,EAAE,CAAC,SAAS,CAAC,IAAI,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QACxC,KAAK,MAAM,KAAK,IAAI,EAAE,CAAC,WAAW,CAAC,GAAG,CAAC,EAAE,CAAC;YACxC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE,KAAK,CAAC,EAAE,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC,CAAC;QAC/D,CAAC;IACH,CAAC;SAAM,CAAC;QACN,EAAE,CAAC,YAAY,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;IAC7B,CAAC;AACH,CAAC;AAED,0GAA0G;AAC1G,MAAM,UAAU,kBAAkB,CAAC,GAAW;IAC5C,IAAI,EAAE,CAAC,QAAQ,EAAE,KAAK,OAAO,EAAE,CAAC;QAC9B,OAAO,GAAG,CAAC,OAAO,CAAC,IAAI,EAAE,GAAG,CAAC,CAAC,OAAO,CAAC,QAAQ,EAAE,GAAG,CAAC,CAAC,OAAO,CAAC,IAAI,EAAE,GAAG,CAAC,CAAC;IAC1E,CAAC;IACD,OAAO,GAAG,CAAC,OAAO,CAAC,KAAK,EAAE,GAAG,CAAC,CAAC,OAAO,CAAC,IAAI,EAAE,GAAG,CAAC,CAAC;AACpD,CAAC"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "kumo-cli",
|
|
3
|
-
"version": "1.0.
|
|
3
|
+
"version": "1.0.3",
|
|
4
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
5
|
"type": "module",
|
|
6
6
|
"bin": {
|
|
@@ -14,15 +14,13 @@
|
|
|
14
14
|
"LICENSE"
|
|
15
15
|
],
|
|
16
16
|
"scripts": {
|
|
17
|
-
"dev": "tsx src/index.ts",
|
|
17
|
+
"dev": "tsx src/index.ts --env-file ../.env",
|
|
18
18
|
"build": "tsc",
|
|
19
19
|
"start": "node dist/index.js",
|
|
20
|
-
"test": "node test/test-pty-parser.mjs",
|
|
21
|
-
"fake-ws": "node test/fake-ws.mjs",
|
|
22
20
|
"prepublishOnly": "npm run build"
|
|
23
21
|
},
|
|
24
22
|
"engines": {
|
|
25
|
-
"node": ">=
|
|
23
|
+
"node": ">=20"
|
|
26
24
|
},
|
|
27
25
|
"license": "MIT",
|
|
28
26
|
"author": "Kumo",
|
|
@@ -49,8 +47,11 @@
|
|
|
49
47
|
"dependencies": {
|
|
50
48
|
"@xterm/headless": "^6.0.0",
|
|
51
49
|
"bonjour-service": "^1.3.0",
|
|
50
|
+
"claude-adapter": "^2.1.1",
|
|
51
|
+
"fastify": "^5.8.5",
|
|
52
52
|
"http-proxy": "^1.18.1",
|
|
53
53
|
"node-pty": "^1.1.0",
|
|
54
|
+
"openai": "^6.39.1",
|
|
54
55
|
"uuid": "^10.0.0",
|
|
55
56
|
"ws": "^8.18.0"
|
|
56
57
|
},
|