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.
Files changed (70) hide show
  1. package/README.md +20 -31
  2. package/dist/auth/credentialsStore.js +41 -0
  3. package/dist/auth/credentialsStore.js.map +1 -0
  4. package/dist/claude/generateHookSettings.js +23 -0
  5. package/dist/claude/generateHookSettings.js.map +1 -0
  6. package/dist/claude/hookServer.js +33 -0
  7. package/dist/claude/hookServer.js.map +1 -0
  8. package/dist/claude/sessionScanner.js +250 -0
  9. package/dist/claude/sessionScanner.js.map +1 -0
  10. package/dist/config.js +40 -0
  11. package/dist/config.js.map +1 -0
  12. package/dist/core/config.js +48 -0
  13. package/dist/core/config.js.map +1 -0
  14. package/dist/core/loadDotEnv.js +46 -0
  15. package/dist/core/loadDotEnv.js.map +1 -0
  16. package/dist/core/logger.js +29 -0
  17. package/dist/core/logger.js.map +1 -0
  18. package/dist/handlers/messageRouter.js +225 -0
  19. package/dist/handlers/messageRouter.js.map +1 -0
  20. package/dist/http/httpClient.js +41 -0
  21. package/dist/http/httpClient.js.map +1 -0
  22. package/dist/index.js +120 -1364
  23. package/dist/index.js.map +1 -1
  24. package/dist/logger.js +29 -0
  25. package/dist/logger.js.map +1 -0
  26. package/dist/pty/ansiCodes.js +14 -0
  27. package/dist/pty/ansiCodes.js.map +1 -0
  28. package/dist/pty/ptyMenuParser.js +357 -0
  29. package/dist/pty/ptyMenuParser.js.map +1 -0
  30. package/dist/pty/ptySnapshotRenderer.js +71 -0
  31. package/dist/pty/ptySnapshotRenderer.js.map +1 -0
  32. package/dist/pty/spawn.js +77 -0
  33. package/dist/pty/spawn.js.map +1 -0
  34. package/dist/pty/terminalManager.js +66 -0
  35. package/dist/pty/terminalManager.js.map +1 -0
  36. package/dist/services/claudeAdapterProxy.js +104 -0
  37. package/dist/services/claudeAdapterProxy.js.map +1 -0
  38. package/dist/services/claudeService.js +239 -0
  39. package/dist/services/claudeService.js.map +1 -0
  40. package/dist/services/claudeService.test.js +39 -0
  41. package/dist/services/claudeService.test.js.map +1 -0
  42. package/dist/services/effortService.js +89 -0
  43. package/dist/services/effortService.js.map +1 -0
  44. package/dist/services/fileService.js +129 -0
  45. package/dist/services/fileService.js.map +1 -0
  46. package/dist/services/modelService.js +115 -0
  47. package/dist/services/modelService.js.map +1 -0
  48. package/dist/services/modelService.test.js +76 -0
  49. package/dist/services/modelService.test.js.map +1 -0
  50. package/dist/services/pairingService.js +129 -0
  51. package/dist/services/pairingService.js.map +1 -0
  52. package/dist/services/sessionService.js +168 -0
  53. package/dist/services/sessionService.js.map +1 -0
  54. package/dist/services/tunnelService.js +47 -0
  55. package/dist/services/tunnelService.js.map +1 -0
  56. package/dist/sessionScanner.js +130 -4
  57. package/dist/sessionScanner.js.map +1 -1
  58. package/dist/snapshotScanner.js +3 -1
  59. package/dist/snapshotScanner.js.map +1 -1
  60. package/dist/transport/directWsServer.js +135 -0
  61. package/dist/transport/directWsServer.js.map +1 -0
  62. package/dist/transport/wsClient.js +87 -0
  63. package/dist/transport/wsClient.js.map +1 -0
  64. package/dist/utils/ignorePaths.js +54 -0
  65. package/dist/utils/ignorePaths.js.map +1 -0
  66. package/dist/utils/ptyBuffer.js +46 -0
  67. package/dist/utils/ptyBuffer.js.map +1 -0
  68. package/dist/utils/safePath.js +43 -0
  69. package/dist/utils/safePath.js.map +1 -0
  70. 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.1",
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": ">=18"
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
  },