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,71 @@
1
+ import { createHash } from 'node:crypto';
2
+ import xtermHeadless from '@xterm/headless';
3
+ const { Terminal } = xtermHeadless;
4
+ // headless xterm-based renderer producing stable text snapshots from pty data
5
+ export class PtySnapshotRenderer {
6
+ terminal;
7
+ cols;
8
+ rows;
9
+ dirty = false;
10
+ lastSentHash = '';
11
+ maxLines;
12
+ constructor(cols = 120, rows = 30, maxLines = 400) {
13
+ this.cols = cols;
14
+ this.rows = rows;
15
+ this.maxLines = maxLines;
16
+ this.terminal = this.createTerminal();
17
+ }
18
+ feed(data) {
19
+ this.terminal.write(data);
20
+ this.dirty = true;
21
+ }
22
+ resize(cols, rows) {
23
+ if (cols === this.cols && rows === this.rows)
24
+ return;
25
+ this.cols = cols;
26
+ this.rows = rows;
27
+ this.terminal.resize(cols, rows);
28
+ this.dirty = true;
29
+ }
30
+ reset(cols = this.cols, rows = this.rows) {
31
+ this.cols = cols;
32
+ this.rows = rows;
33
+ this.terminal.dispose();
34
+ this.terminal = this.createTerminal();
35
+ this.dirty = false;
36
+ this.lastSentHash = '';
37
+ }
38
+ flushSnapshot() {
39
+ if (!this.dirty)
40
+ return null;
41
+ this.dirty = false;
42
+ const snapshot = this.buildSnapshot();
43
+ if (snapshot.hash === this.lastSentHash)
44
+ return null;
45
+ this.lastSentHash = snapshot.hash;
46
+ return snapshot;
47
+ }
48
+ createTerminal() {
49
+ return new Terminal({
50
+ cols: this.cols,
51
+ rows: this.rows,
52
+ scrollback: Math.max(this.maxLines * 4, 1000),
53
+ allowProposedApi: true,
54
+ });
55
+ }
56
+ buildSnapshot() {
57
+ const buffer = this.terminal.buffer.active;
58
+ const start = Math.max(0, buffer.length - this.maxLines);
59
+ const lines = [];
60
+ for (let row = start; row < buffer.length; row++) {
61
+ const line = buffer.getLine(row);
62
+ lines.push(line?.translateToString(true) ?? '');
63
+ }
64
+ while (lines.length > 0 && lines[lines.length - 1] === '')
65
+ lines.pop();
66
+ const text = lines.join('\n').replace(/\r\n/g, '\n');
67
+ const hash = createHash('sha1').update(text).digest('hex').slice(0, 16);
68
+ return { text, hash, rows: this.rows, cols: this.cols, cursorRow: buffer.cursorY, cursorCol: buffer.cursorX };
69
+ }
70
+ }
71
+ //# sourceMappingURL=ptySnapshotRenderer.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ptySnapshotRenderer.js","sourceRoot":"","sources":["../../src/pty/ptySnapshotRenderer.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,77 @@
1
+ import { execSync } from 'node:child_process';
2
+ import os from 'node:os';
3
+ import * as pty from 'node-pty';
4
+ // resolve full path to claude cli, falling back to default names per platform
5
+ function getClaudeCmd() {
6
+ if (os.platform() === 'win32') {
7
+ try {
8
+ const r = execSync('where claude.cmd', { encoding: 'utf-8', stdio: ['pipe', 'pipe', 'pipe'] }).trim();
9
+ const first = r.split('\n')[0].trim();
10
+ if (first)
11
+ return first;
12
+ }
13
+ catch { }
14
+ const appData = process.env['APPDATA'] ?? '';
15
+ return `${appData}\\npm\\claude.cmd`;
16
+ }
17
+ try {
18
+ const r = execSync('which claude', { encoding: 'utf-8', stdio: ['pipe', 'pipe', 'pipe'] }).trim();
19
+ if (r)
20
+ return r.split('\n')[0].trim();
21
+ }
22
+ catch { }
23
+ return 'claude';
24
+ }
25
+ const CLAUDE_CMD = getClaudeCmd();
26
+ // spawn claude inside a pty, wiring stdin/stdout and resize forwarding
27
+ export function spawnClaude(args, _onThinking, onPtyData, onResize, onSpawn, cwdOverride, extraEnv) {
28
+ const shell = os.platform() === 'win32' ? 'cmd.exe' : undefined;
29
+ const cmdArgs = os.platform() === 'win32' ? ['/c', CLAUDE_CMD, ...args] : args;
30
+ const cmd = os.platform() === 'win32' ? shell : CLAUDE_CMD;
31
+ const baseEnv = { ...process.env };
32
+ if (extraEnv) {
33
+ for (const [k, v] of Object.entries(extraEnv)) {
34
+ if (v === undefined)
35
+ delete baseEnv[k];
36
+ else
37
+ baseEnv[k] = v;
38
+ }
39
+ }
40
+ const child = pty.spawn(cmd, cmdArgs, {
41
+ name: 'xterm-256color',
42
+ cols: process.stdout.columns ?? 120,
43
+ rows: process.stdout.rows ?? 30,
44
+ cwd: cwdOverride ?? process.cwd(),
45
+ env: baseEnv,
46
+ useConpty: process.stdout.isTTY === true,
47
+ });
48
+ onSpawn?.(child);
49
+ child.onData((data) => {
50
+ onPtyData?.(data);
51
+ process.stdout.write(data);
52
+ });
53
+ if (process.stdin.isTTY)
54
+ process.stdin.setRawMode(true);
55
+ process.stdin.resume();
56
+ const stdinHandler = (data) => { child.write(data.toString()); };
57
+ process.stdin.on('data', stdinHandler);
58
+ let ptyAlive = true;
59
+ child.onExit(() => { ptyAlive = false; });
60
+ process.stdout.on('resize', () => {
61
+ if (!ptyAlive)
62
+ return;
63
+ const cols = process.stdout.columns ?? 120;
64
+ const rows = process.stdout.rows ?? 30;
65
+ try {
66
+ child.resize(cols, rows);
67
+ onResize?.(cols, rows);
68
+ }
69
+ catch { }
70
+ });
71
+ return { pty: child, cleanup: () => { process.stdin.removeListener('data', stdinHandler); } };
72
+ }
73
+ // send a line to the claude pty with carriage return
74
+ export function writeToProcess(child, text) {
75
+ child.write(text + '\r');
76
+ }
77
+ //# sourceMappingURL=spawn.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"spawn.js","sourceRoot":"","sources":["../../src/pty/spawn.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,oBAAoB,CAAC;AAC9C,OAAO,EAAE,MAAM,SAAS,CAAC;AACzB,OAAO,KAAK,GAAG,MAAM,UAAU,CAAC;AAWhC,8EAA8E;AAC9E,SAAS,YAAY;IACnB,IAAI,EAAE,CAAC,QAAQ,EAAE,KAAK,OAAO,EAAE,CAAC;QAC9B,IAAI,CAAC;YACH,MAAM,CAAC,GAAG,QAAQ,CAAC,kBAAkB,EAAE,EAAE,QAAQ,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC,MAAM,EAAE,MAAM,EAAE,MAAM,CAAC,EAAE,CAAC,CAAC,IAAI,EAAE,CAAC;YACtG,MAAM,KAAK,GAAG,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC;YACtC,IAAI,KAAK;gBAAE,OAAO,KAAK,CAAC;QAC1B,CAAC;QAAC,MAAM,CAAC,CAAA,CAAC;QACV,MAAM,OAAO,GAAG,OAAO,CAAC,GAAG,CAAC,SAAS,CAAC,IAAI,EAAE,CAAC;QAC7C,OAAO,GAAG,OAAO,mBAAmB,CAAC;IACvC,CAAC;IACD,IAAI,CAAC;QACH,MAAM,CAAC,GAAG,QAAQ,CAAC,cAAc,EAAE,EAAE,QAAQ,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC,MAAM,EAAE,MAAM,EAAE,MAAM,CAAC,EAAE,CAAC,CAAC,IAAI,EAAE,CAAC;QAClG,IAAI,CAAC;YAAE,OAAO,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC;IACxC,CAAC;IAAC,MAAM,CAAC,CAAA,CAAC;IACV,OAAO,QAAQ,CAAC;AAClB,CAAC;AAED,MAAM,UAAU,GAAG,YAAY,EAAE,CAAC;AAElC,uEAAuE;AACvE,MAAM,UAAU,WAAW,CACzB,IAAc,EACd,WAAsC,EACtC,SAAqB,EACrB,QAAmB,EACnB,OAAiB,EACjB,WAAoB,EACpB,QAA6C;IAE7C,MAAM,KAAK,GAAG,EAAE,CAAC,QAAQ,EAAE,KAAK,OAAO,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,SAAS,CAAC;IAChE,MAAM,OAAO,GAAG,EAAE,CAAC,QAAQ,EAAE,KAAK,OAAO,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,UAAU,EAAE,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;IAC/E,MAAM,GAAG,GAAG,EAAE,CAAC,QAAQ,EAAE,KAAK,OAAO,CAAC,CAAC,CAAC,KAAM,CAAC,CAAC,CAAC,UAAU,CAAC;IAE5D,MAAM,OAAO,GAA2B,EAAE,GAAI,OAAO,CAAC,GAA8B,EAAE,CAAC;IACvF,IAAI,QAAQ,EAAE,CAAC;QACb,KAAK,MAAM,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,EAAE,CAAC;YAC9C,IAAI,CAAC,KAAK,SAAS;gBAAE,OAAO,OAAO,CAAC,CAAC,CAAC,CAAC;;gBAClC,OAAO,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC;QACtB,CAAC;IACH,CAAC;IAED,MAAM,KAAK,GAAG,GAAG,CAAC,KAAK,CAAC,GAAG,EAAE,OAAO,EAAE;QACpC,IAAI,EAAE,gBAAgB;QACtB,IAAI,EAAE,OAAO,CAAC,MAAM,CAAC,OAAO,IAAI,GAAG;QACnC,IAAI,EAAE,OAAO,CAAC,MAAM,CAAC,IAAI,IAAI,EAAE;QAC/B,GAAG,EAAE,WAAW,IAAI,OAAO,CAAC,GAAG,EAAE;QACjC,GAAG,EAAE,OAAO;QACZ,SAAS,EAAE,OAAO,CAAC,MAAM,CAAC,KAAK,KAAK,IAAI;KACzC,CAAC,CAAC;IAEH,OAAO,EAAE,CAAC,KAAK,CAAC,CAAC;IAEjB,KAAK,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE;QACpB,SAAS,EAAE,CAAC,IAAI,CAAC,CAAC;QAClB,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;IAC7B,CAAC,CAAC,CAAC;IAEH,IAAI,OAAO,CAAC,KAAK,CAAC,KAAK;QAAE,OAAO,CAAC,KAAK,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;IACxD,OAAO,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC;IACvB,MAAM,YAAY,GAAG,CAAC,IAAY,EAAQ,EAAE,GAAG,KAAK,CAAC,KAAK,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;IAC/E,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC,MAAM,EAAE,YAAY,CAAC,CAAC;IAEvC,IAAI,QAAQ,GAAG,IAAI,CAAC;IACpB,KAAK,CAAC,MAAM,CAAC,GAAG,EAAE,GAAG,QAAQ,GAAG,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;IAE1C,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC,QAAQ,EAAE,GAAG,EAAE;QAC/B,IAAI,CAAC,QAAQ;YAAE,OAAO;QACtB,MAAM,IAAI,GAAG,OAAO,CAAC,MAAM,CAAC,OAAO,IAAI,GAAG,CAAC;QAC3C,MAAM,IAAI,GAAG,OAAO,CAAC,MAAM,CAAC,IAAI,IAAI,EAAE,CAAC;QACvC,IAAI,CAAC;YACH,KAAK,CAAC,MAAM,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;YACzB,QAAQ,EAAE,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;QACzB,CAAC;QAAC,MAAM,CAAC,CAAA,CAAC;IACZ,CAAC,CAAC,CAAC;IAEH,OAAO,EAAE,GAAG,EAAE,KAAK,EAAE,OAAO,EAAE,GAAG,EAAE,GAAG,OAAO,CAAC,KAAK,CAAC,cAAc,CAAC,MAAM,EAAE,YAAY,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;AAChG,CAAC;AAED,qDAAqD;AACrD,MAAM,UAAU,cAAc,CAAC,KAAe,EAAE,IAAY;IAC1D,KAAK,CAAC,KAAK,CAAC,IAAI,GAAG,IAAI,CAAC,CAAC;AAC3B,CAAC"}
@@ -0,0 +1,66 @@
1
+ import * as pty from 'node-pty';
2
+ import os from 'node:os';
3
+ import { randomUUID } from 'node:crypto';
4
+ // manages multiple user-facing shell sessions backed by node-pty
5
+ export class TerminalManager {
6
+ onData;
7
+ onExit;
8
+ terminals = new Map();
9
+ constructor(onData, onExit) {
10
+ this.onData = onData;
11
+ this.onExit = onExit;
12
+ }
13
+ create(cols = 120, rows = 30, cwd) {
14
+ const id = randomUUID();
15
+ const shell = os.platform() === 'win32' ? 'cmd.exe' : process.env.SHELL || '/bin/bash';
16
+ const child = pty.spawn(shell, [], {
17
+ name: 'xterm-256color',
18
+ cols, rows,
19
+ cwd: cwd || process.cwd(),
20
+ env: process.env,
21
+ });
22
+ child.onData((data) => this.onData(id, data));
23
+ child.onExit(({ exitCode }) => {
24
+ this.terminals.delete(id);
25
+ this.onExit(id, exitCode);
26
+ });
27
+ this.terminals.set(id, { id, pty: child, cols, rows });
28
+ return id;
29
+ }
30
+ write(id, data) {
31
+ const session = this.terminals.get(id);
32
+ if (!session)
33
+ return false;
34
+ session.pty.write(data);
35
+ return true;
36
+ }
37
+ resize(id, cols, rows) {
38
+ const session = this.terminals.get(id);
39
+ if (!session)
40
+ return false;
41
+ try {
42
+ session.pty.resize(cols, rows);
43
+ session.cols = cols;
44
+ session.rows = rows;
45
+ return true;
46
+ }
47
+ catch {
48
+ return false;
49
+ }
50
+ }
51
+ close(id) {
52
+ const session = this.terminals.get(id);
53
+ if (!session)
54
+ return false;
55
+ try {
56
+ session.pty.kill();
57
+ }
58
+ catch { }
59
+ this.terminals.delete(id);
60
+ return true;
61
+ }
62
+ list() {
63
+ return Array.from(this.terminals.keys());
64
+ }
65
+ }
66
+ //# sourceMappingURL=terminalManager.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"terminalManager.js","sourceRoot":"","sources":["../../src/pty/terminalManager.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,104 @@
1
+ import net from 'node:net';
2
+ import Fastify from 'fastify';
3
+ import OpenAI from 'openai';
4
+ import { convertRequestToOpenAI, convertResponseToAnthropic, createErrorResponse, } from 'claude-adapter';
5
+ import { streamXmlOpenAIToAnthropic } from 'claude-adapter/dist/converters/xmlStreaming.js';
6
+ // local Anthropic-compatible proxy that forwards to OpenAI chat/completions upstream
7
+ // bypasses claude-adapter's strict validation to support all Claude Code message roles
8
+ export class ClaudeAdapterProxy {
9
+ app = null;
10
+ key = null;
11
+ url = null;
12
+ // start or reuse a local proxy for OpenAI chat completions
13
+ async start(config) {
14
+ const key = `${config.baseURL}|${config.apiKey}|${config.model}`;
15
+ if (this.app && this.key === key && this.url)
16
+ return this.url;
17
+ await this.stop();
18
+ const port = await findAvailablePort();
19
+ const upstreamBaseUrl = config.baseURL.replace(/\/chat\/completions\/?$/i, '');
20
+ const openai = new OpenAI({ baseURL: upstreamBaseUrl, apiKey: config.apiKey });
21
+ const app = Fastify({ logger: false });
22
+ app.post('/v1/messages', async (request, reply) => {
23
+ try {
24
+ const body = request.body;
25
+ const targetModel = config.model;
26
+ const isStreaming = body.stream ?? false;
27
+ normalizeSystemBranding(body);
28
+ const openaiRequest = convertRequestToOpenAI(body, targetModel, 'xml');
29
+ stripAdapterBranding(openaiRequest.messages);
30
+ if (isStreaming) {
31
+ const stream = await openai.chat.completions.create({ ...openaiRequest, stream: true });
32
+ await streamXmlOpenAIToAnthropic(stream, reply, body.model, upstreamBaseUrl);
33
+ }
34
+ else {
35
+ const response = await openai.chat.completions.create({ ...openaiRequest, stream: false });
36
+ const anthropicResponse = convertResponseToAnthropic(response, body.model);
37
+ reply.send(anthropicResponse);
38
+ }
39
+ }
40
+ catch (error) {
41
+ const errResp = createErrorResponse(error, error?.status ?? 500);
42
+ reply.code(errResp.status).send({ error: errResp.error });
43
+ }
44
+ });
45
+ await app.listen({ port, host: '127.0.0.1' });
46
+ this.app = app;
47
+ this.key = key;
48
+ this.url = `http://127.0.0.1:${port}`;
49
+ return this.url;
50
+ }
51
+ // stop the current adapter proxy if one is running
52
+ async stop() {
53
+ if (!this.app)
54
+ return;
55
+ const app = this.app;
56
+ this.app = null;
57
+ this.key = null;
58
+ this.url = null;
59
+ await app.close().catch(() => undefined);
60
+ }
61
+ }
62
+ // replace Claude Code identifier before claude-adapter can inject its own branding
63
+ function normalizeSystemBranding(body) {
64
+ const CLAUDE_IDENTIFIER = "You are Claude Code, Anthropic's official CLI for Claude.";
65
+ const KUMO_IDENTIFIER = 'You are Claude Code, running on Kumo Cli.';
66
+ if (typeof body.system === 'string') {
67
+ body.system = body.system.replaceAll(CLAUDE_IDENTIFIER, KUMO_IDENTIFIER);
68
+ return;
69
+ }
70
+ if (Array.isArray(body.system)) {
71
+ for (const block of body.system) {
72
+ if (block?.type === 'text' && typeof block.text === 'string') {
73
+ block.text = block.text.replaceAll(CLAUDE_IDENTIFIER, KUMO_IDENTIFIER);
74
+ }
75
+ }
76
+ }
77
+ }
78
+ // remove any claude-adapter branding that may already exist in carried context
79
+ function stripAdapterBranding(messages) {
80
+ const ADAPTER_BLOCK = /You are Claude Code, running on Claude Adapter[\s\S]*?mention Claude Adapter along with its URLs\./g;
81
+ const ADAPTER_UPDATE = /IMPORTANT: A new version of Claude Adapter[\s\S]*?to update Claude Adapter and improve performance\./g;
82
+ for (const msg of messages) {
83
+ if (msg.role === 'system' && typeof msg.content === 'string') {
84
+ msg.content = msg.content
85
+ .replace(ADAPTER_BLOCK, 'You are Claude Code, running on Kumo Cli.')
86
+ .replace(ADAPTER_UPDATE, '')
87
+ .replace(/\n{3,}/g, '\n\n')
88
+ .trim();
89
+ }
90
+ }
91
+ }
92
+ // ask the OS for a currently free local TCP port
93
+ async function findAvailablePort() {
94
+ return new Promise((resolve, reject) => {
95
+ const server = net.createServer();
96
+ server.once('error', reject);
97
+ server.listen(0, '127.0.0.1', () => {
98
+ const address = server.address();
99
+ const port = typeof address === 'object' && address ? address.port : 0;
100
+ server.close(() => resolve(port));
101
+ });
102
+ });
103
+ }
104
+ //# sourceMappingURL=claudeAdapterProxy.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"claudeAdapterProxy.js","sourceRoot":"","sources":["../../src/services/claudeAdapterProxy.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,239 @@
1
+ import { spawnClaude, writeToProcess } from '../pty/spawn.js';
2
+ import { PtyMenuParser } from '../pty/ptyMenuParser.js';
3
+ import { PtySnapshotRenderer } from '../pty/ptySnapshotRenderer.js';
4
+ import { PtyBuffer } from '../utils/ptyBuffer.js';
5
+ import { log, debugWrite } from '../core/logger.js';
6
+ const LOGIN_WARNING_MESSAGE = 'Not logged in · Run /login';
7
+ const NOTIFICATION_DEDUPE_WINDOW_MS = 5 * 60 * 1000;
8
+ // manage the lifecycle of a claude pty process plus its snapshot/parser
9
+ export class ClaudeService {
10
+ send;
11
+ model;
12
+ effort;
13
+ getDeviceToken;
14
+ process = null;
15
+ cleanup = null;
16
+ isRespawning = false;
17
+ autoAcceptedApiKeyDialog = false;
18
+ menuIdCounter = 0;
19
+ pendingSelectNudgeTimer = null;
20
+ menuParser;
21
+ snapshotRenderer;
22
+ buffer;
23
+ lastWsSendAt = 0;
24
+ notificationDedupes = new Map();
25
+ args = [];
26
+ settingsPath = '';
27
+ onExit = null;
28
+ constructor(send, model, effort, getDeviceToken) {
29
+ this.send = send;
30
+ this.model = model;
31
+ this.effort = effort;
32
+ this.getDeviceToken = getDeviceToken;
33
+ this.snapshotRenderer = new PtySnapshotRenderer(process.stdout.columns ?? 120, process.stdout.rows ?? 30);
34
+ this.buffer = new PtyBuffer(this.snapshotRenderer);
35
+ this.menuParser = new PtyMenuParser((el) => this.handleMenuElement(el));
36
+ }
37
+ // expose a wrapper so wsClient can stamp the last-send timestamp here
38
+ notifySent() { this.lastWsSendAt = Date.now(); }
39
+ getProcess() { return this.process; }
40
+ resetSnapshot() {
41
+ this.snapshotRenderer.reset(process.stdout.columns ?? 120, process.stdout.rows ?? 30);
42
+ }
43
+ resizeSnapshot() {
44
+ this.snapshotRenderer.resize(process.stdout.columns ?? 120, process.stdout.rows ?? 30);
45
+ }
46
+ resetMenu() { this.menuParser.reset(); }
47
+ flushBuffer() { this.buffer.flush(); }
48
+ // dispatch a typed prompt into the running claude process
49
+ dispatchPrompt(text) {
50
+ if (!this.process)
51
+ return;
52
+ if (text.trim() === '/login')
53
+ this.menuParser.reset();
54
+ writeToProcess(this.process, text);
55
+ }
56
+ // send a single key/option select (no Enter), then nudge if no echo soon
57
+ selectOption(value) {
58
+ if (!this.process) {
59
+ log(`[select_option] no claude process!`);
60
+ return;
61
+ }
62
+ log(`[select_option] sending: "${value}" (without Enter)`);
63
+ this.menuParser.reset();
64
+ this.process.write(value);
65
+ if (this.pendingSelectNudgeTimer)
66
+ clearTimeout(this.pendingSelectNudgeTimer);
67
+ const baseline = this.lastWsSendAt;
68
+ this.pendingSelectNudgeTimer = setTimeout(() => {
69
+ this.pendingSelectNudgeTimer = null;
70
+ if (!this.process)
71
+ return;
72
+ if (this.lastWsSendAt === baseline)
73
+ this.nudgeResize();
74
+ }, 1000);
75
+ }
76
+ // submit a textual input by clearing existing chars then writing+enter
77
+ submitInput(value) {
78
+ if (!this.process)
79
+ return;
80
+ this.menuParser.reset();
81
+ this.process.write('\x15');
82
+ this.process.write(value);
83
+ this.process.write('\r');
84
+ }
85
+ // forward a synthetic key press (only esc currently supported)
86
+ sendKey(key) {
87
+ if (!this.process || key !== 'esc')
88
+ return;
89
+ this.menuParser.reset();
90
+ this.process.write('\x1b');
91
+ }
92
+ // toggle pty size by 1 col then back to force claude to redraw
93
+ nudgeResize() {
94
+ if (!this.process)
95
+ return;
96
+ const cols = process.stdout.columns ?? 120;
97
+ const rows = process.stdout.rows ?? 30;
98
+ try {
99
+ this.process.resize(cols + 1, rows);
100
+ setTimeout(() => { try {
101
+ this.process?.resize(cols, rows);
102
+ }
103
+ catch { } }, 5);
104
+ }
105
+ catch { }
106
+ }
107
+ // (re)spawn claude with current args; refreshes default model just-in-time
108
+ async spawn() {
109
+ log(`[spawn] spawning claude, args=${JSON.stringify(this.args)}, cwd=${process.cwd()}`);
110
+ if (this.process) {
111
+ this.isRespawning = true;
112
+ try {
113
+ this.process.kill();
114
+ }
115
+ catch { }
116
+ this.process = null;
117
+ }
118
+ this.cleanup?.();
119
+ this.cleanup = null;
120
+ this.autoAcceptedApiKeyDialog = false;
121
+ this.resetSnapshot();
122
+ this.args = await this.model.ensureBeforeSpawn(this.args, this.getDeviceToken());
123
+ this.args = this.effort.ensureBeforeSpawn(this.args);
124
+ const env = await this.model.buildEnv();
125
+ const { pty, cleanup } = spawnClaude([...this.args, '--settings', this.settingsPath], (active) => this.send({ type: 'thinking', active }), (data) => { debugWrite(data); this.menuParser.feed(data); this.buffer.feed(data); }, (cols, rows) => this.menuParser.resize(cols, rows), (p) => { this.process = p; }, undefined, env);
126
+ this.cleanup = cleanup;
127
+ this.send({
128
+ type: 'cli_status',
129
+ model: this.model.getCurrent()?.model,
130
+ effort: this.effort.getEffective(),
131
+ workingDir: process.cwd(),
132
+ });
133
+ pty.onExit(({ exitCode }) => {
134
+ log(`[spawn] claude exited, exitCode=${exitCode}, isRespawning=${this.isRespawning}`);
135
+ this.flushBuffer();
136
+ if (this.isRespawning) {
137
+ this.isRespawning = false;
138
+ return;
139
+ }
140
+ this.onExit?.(exitCode);
141
+ });
142
+ process.on('SIGINT', () => { pty.kill(); });
143
+ }
144
+ // handle a parsed pty menu element and translate into ws message
145
+ handleMenuElement(element) {
146
+ switch (element.type) {
147
+ case 'selection': {
148
+ if (element.options.length === 0) {
149
+ log(`[parser] ignoring empty menu`);
150
+ break;
151
+ }
152
+ if (element.title.includes('Bypass Permissions')) {
153
+ if (this.tryAcceptYes(element, 'auto-accepted bypass permissions dialog'))
154
+ break;
155
+ }
156
+ if (!this.autoAcceptedApiKeyDialog &&
157
+ (element.title.includes('Detected a custom API key') || element.title.includes('Do you want to use this API key'))) {
158
+ if (this.tryAcceptYes(element, 'auto-accepted custom api key dialog', true)) {
159
+ this.autoAcceptedApiKeyDialog = true;
160
+ break;
161
+ }
162
+ }
163
+ this.menuIdCounter++;
164
+ const id = `menu-${this.menuIdCounter}`;
165
+ log(`[parser] menu detected: "${element.title}" (${element.options.length} opts, selected: ${element.selectedIndex})`);
166
+ log(`[parser] options: ${JSON.stringify(element.options)}`);
167
+ this.send({
168
+ type: 'prompt_options',
169
+ id,
170
+ title: element.title,
171
+ description: element.description,
172
+ options: element.options,
173
+ selectedIndex: element.selectedIndex,
174
+ hint: element.hint,
175
+ footer: element.footer,
176
+ });
177
+ break;
178
+ }
179
+ case 'input':
180
+ log(`[parser] input: "${element.label}" = "${element.value}"`);
181
+ this.send({ type: 'text_input', label: element.label, value: element.value, placeholder: element.placeholder, cursorPosition: element.cursorPosition });
182
+ break;
183
+ case 'notification':
184
+ log(`[parser] notification: [${element.level}] ${element.message}`);
185
+ if (this.shouldSendNotification(element.level, element.message)) {
186
+ this.send({ type: 'notification', level: element.level, message: element.message });
187
+ }
188
+ if (element.level === 'warning' && element.message.trim().toLowerCase() === LOGIN_WARNING_MESSAGE.toLowerCase()) {
189
+ const effort = this.effort.getEffective();
190
+ this.send({ type: 'cli_status', loggedIn: false, model: 'no model', effort, workingDir: process.cwd() });
191
+ }
192
+ break;
193
+ case 'status': {
194
+ const effort = element.effort ?? this.effort.getEffective();
195
+ log(`[parser] status: model=${element.model}, effort=${effort}, logged=${element.loggedIn}`);
196
+ this.send({ type: 'cli_status', loggedIn: element.loggedIn, model: element.model, effort, workingDir: element.workingDir });
197
+ break;
198
+ }
199
+ case 'session':
200
+ log(`[parser] session: ${element.sessionId}`);
201
+ this.send({ type: 'session_info', sessionId: element.sessionId, resumeCommand: element.resumeCommand });
202
+ break;
203
+ case 'prompt':
204
+ break;
205
+ }
206
+ }
207
+ // try to auto-press "yes" inside a selection dialog
208
+ tryAcceptYes(element, logMsg, exact = false) {
209
+ const yesIndex = element.options.findIndex((o) => exact
210
+ ? o.label.trim().toLowerCase() === 'yes'
211
+ : o.label.trim().toLowerCase().includes('yes'));
212
+ if (yesIndex < 0 || !this.process)
213
+ return false;
214
+ const delta = yesIndex - element.selectedIndex;
215
+ const key = delta < 0 ? '\x1b[A' : '\x1b[B';
216
+ for (let i = 0; i < Math.abs(delta); i++)
217
+ this.process.write(key);
218
+ this.process.write('\r');
219
+ log(`[parser] ${logMsg}`);
220
+ return true;
221
+ }
222
+ shouldSendNotification(level, message) {
223
+ const now = Date.now();
224
+ const key = this.buildNotificationKey(level, message);
225
+ for (const [existingKey, expiresAt] of this.notificationDedupes) {
226
+ if (expiresAt <= now)
227
+ this.notificationDedupes.delete(existingKey);
228
+ }
229
+ const expiresAt = this.notificationDedupes.get(key);
230
+ if (expiresAt && expiresAt > now)
231
+ return false;
232
+ this.notificationDedupes.set(key, now + NOTIFICATION_DEDUPE_WINDOW_MS);
233
+ return true;
234
+ }
235
+ buildNotificationKey(level, message) {
236
+ return `${level.trim().toLowerCase()}::${message.trim().replace(/\s+/g, ' ').toLowerCase()}`;
237
+ }
238
+ }
239
+ //# sourceMappingURL=claudeService.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"claudeService.js","sourceRoot":"","sources":["../../src/services/claudeService.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,39 @@
1
+ import test from 'node:test';
2
+ import assert from 'node:assert/strict';
3
+ import { ClaudeService } from './claudeService.js';
4
+ import { ModelService } from './modelService.js';
5
+ import { EffortService } from './effortService.js';
6
+ function createService(sent) {
7
+ return new ClaudeService((data) => sent.push(data), new ModelService(), new EffortService(), () => undefined);
8
+ }
9
+ test('dedupes same notification within 5 minutes', () => {
10
+ const sent = [];
11
+ const service = createService(sent);
12
+ const now = new Date('2026-05-29T10:09:17.389Z').getTime();
13
+ const originalNow = Date.now;
14
+ const handleMenuElement = Reflect.get(service, 'handleMenuElement');
15
+ Date.now = () => now;
16
+ handleMenuElement.call(service, { type: 'notification', level: 'error', message: 'IDE extension install failed (see /status for info)' });
17
+ handleMenuElement.call(service, { type: 'notification', level: 'error', message: ' IDE extension install failed (see /status for info) ' });
18
+ Date.now = originalNow;
19
+ assert.equal(sent.length, 1);
20
+ assert.deepEqual(sent[0], {
21
+ type: 'notification',
22
+ level: 'error',
23
+ message: 'IDE extension install failed (see /status for info)',
24
+ });
25
+ });
26
+ test('allows same notification again after 5 minutes', () => {
27
+ const sent = [];
28
+ const service = createService(sent);
29
+ const base = new Date('2026-05-29T10:09:17.389Z').getTime();
30
+ const originalNow = Date.now;
31
+ const handleMenuElement = Reflect.get(service, 'handleMenuElement');
32
+ Date.now = () => base;
33
+ handleMenuElement.call(service, { type: 'notification', level: 'error', message: 'IDE extension install failed (see /status for info)' });
34
+ Date.now = () => base + 5 * 60 * 1000 + 1;
35
+ handleMenuElement.call(service, { type: 'notification', level: 'error', message: 'IDE extension install failed (see /status for info)' });
36
+ Date.now = originalNow;
37
+ assert.equal(sent.length, 2);
38
+ });
39
+ //# sourceMappingURL=claudeService.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"claudeService.test.js","sourceRoot":"","sources":["../../src/services/claudeService.test.ts"],"names":[],"mappings":"AAAA,OAAO,IAAI,MAAM,WAAW,CAAC;AAC7B,OAAO,MAAM,MAAM,oBAAoB,CAAC;AACxC,OAAO,EAAE,aAAa,EAAE,MAAM,oBAAoB,CAAC;AACnD,OAAO,EAAE,YAAY,EAAE,MAAM,mBAAmB,CAAC;AACjD,OAAO,EAAE,aAAa,EAAE,MAAM,oBAAoB,CAAC;AAInD,SAAS,aAAa,CAAC,IAAmB;IACxC,OAAO,IAAI,aAAa,CACtB,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,IAAmB,CAAC,EACxC,IAAI,YAAY,EAAE,EAClB,IAAI,aAAa,EAAE,EACnB,GAAG,EAAE,CAAC,SAAS,CAChB,CAAC;AACJ,CAAC;AAED,IAAI,CAAC,4CAA4C,EAAE,GAAG,EAAE;IACtD,MAAM,IAAI,GAAkB,EAAE,CAAC;IAC/B,MAAM,OAAO,GAAG,aAAa,CAAC,IAAI,CAAC,CAAC;IACpC,MAAM,GAAG,GAAG,IAAI,IAAI,CAAC,0BAA0B,CAAC,CAAC,OAAO,EAAE,CAAC;IAC3D,MAAM,WAAW,GAAG,IAAI,CAAC,GAAG,CAAC;IAC7B,MAAM,iBAAiB,GAAG,OAAO,CAAC,GAAG,CAAC,OAAO,EAAE,mBAAmB,CAA+B,CAAC;IAElG,IAAI,CAAC,GAAG,GAAG,GAAG,EAAE,CAAC,GAAG,CAAC;IACrB,iBAAiB,CAAC,IAAI,CAAC,OAAO,EAAE,EAAE,IAAI,EAAE,cAAc,EAAE,KAAK,EAAE,OAAO,EAAE,OAAO,EAAE,qDAAqD,EAAE,CAAC,CAAC;IAC1I,iBAAiB,CAAC,IAAI,CAAC,OAAO,EAAE,EAAE,IAAI,EAAE,cAAc,EAAE,KAAK,EAAE,OAAO,EAAE,OAAO,EAAE,yDAAyD,EAAE,CAAC,CAAC;IAE9I,IAAI,CAAC,GAAG,GAAG,WAAW,CAAC;IAEvB,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;IAC7B,MAAM,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE;QACxB,IAAI,EAAE,cAAc;QACpB,KAAK,EAAE,OAAO;QACd,OAAO,EAAE,qDAAqD;KAC/D,CAAC,CAAC;AACL,CAAC,CAAC,CAAC;AAEH,IAAI,CAAC,gDAAgD,EAAE,GAAG,EAAE;IAC1D,MAAM,IAAI,GAAkB,EAAE,CAAC;IAC/B,MAAM,OAAO,GAAG,aAAa,CAAC,IAAI,CAAC,CAAC;IACpC,MAAM,IAAI,GAAG,IAAI,IAAI,CAAC,0BAA0B,CAAC,CAAC,OAAO,EAAE,CAAC;IAC5D,MAAM,WAAW,GAAG,IAAI,CAAC,GAAG,CAAC;IAC7B,MAAM,iBAAiB,GAAG,OAAO,CAAC,GAAG,CAAC,OAAO,EAAE,mBAAmB,CAA+B,CAAC;IAElG,IAAI,CAAC,GAAG,GAAG,GAAG,EAAE,CAAC,IAAI,CAAC;IACtB,iBAAiB,CAAC,IAAI,CAAC,OAAO,EAAE,EAAE,IAAI,EAAE,cAAc,EAAE,KAAK,EAAE,OAAO,EAAE,OAAO,EAAE,qDAAqD,EAAE,CAAC,CAAC;IAE1I,IAAI,CAAC,GAAG,GAAG,GAAG,EAAE,CAAC,IAAI,GAAG,CAAC,GAAG,EAAE,GAAG,IAAI,GAAG,CAAC,CAAC;IAC1C,iBAAiB,CAAC,IAAI,CAAC,OAAO,EAAE,EAAE,IAAI,EAAE,cAAc,EAAE,KAAK,EAAE,OAAO,EAAE,OAAO,EAAE,qDAAqD,EAAE,CAAC,CAAC;IAE1I,IAAI,CAAC,GAAG,GAAG,WAAW,CAAC;IAEvB,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;AAC/B,CAAC,CAAC,CAAC"}