@cleocode/adapters 2026.5.3 → 2026.5.5
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/dist/index.js +40 -0
- package/dist/index.js.map +3 -3
- package/package.json +4 -4
- package/dist/cant-context.js +0 -711
- package/dist/cant-context.js.map +0 -1
- package/dist/providers/claude-code/adapter.js +0 -222
- package/dist/providers/claude-code/adapter.js.map +0 -1
- package/dist/providers/claude-code/context-monitor.js +0 -159
- package/dist/providers/claude-code/context-monitor.js.map +0 -1
- package/dist/providers/claude-code/hooks.js +0 -428
- package/dist/providers/claude-code/hooks.js.map +0 -1
- package/dist/providers/claude-code/index.js +0 -41
- package/dist/providers/claude-code/index.js.map +0 -1
- package/dist/providers/claude-code/install.js +0 -310
- package/dist/providers/claude-code/install.js.map +0 -1
- package/dist/providers/claude-code/paths.js +0 -41
- package/dist/providers/claude-code/paths.js.map +0 -1
- package/dist/providers/claude-code/spawn.js +0 -198
- package/dist/providers/claude-code/spawn.js.map +0 -1
- package/dist/providers/claude-code/statusline.js +0 -130
- package/dist/providers/claude-code/statusline.js.map +0 -1
- package/dist/providers/claude-code/task-sync.js +0 -119
- package/dist/providers/claude-code/task-sync.js.map +0 -1
- package/dist/providers/claude-code/transport.js +0 -29
- package/dist/providers/claude-code/transport.js.map +0 -1
- package/dist/providers/claude-sdk/adapter.js +0 -88
- package/dist/providers/claude-sdk/adapter.js.map +0 -1
- package/dist/providers/claude-sdk/index.js +0 -34
- package/dist/providers/claude-sdk/index.js.map +0 -1
- package/dist/providers/claude-sdk/install.js +0 -61
- package/dist/providers/claude-sdk/install.js.map +0 -1
- package/dist/providers/claude-sdk/mcp-registry.js +0 -66
- package/dist/providers/claude-sdk/mcp-registry.js.map +0 -1
- package/dist/providers/claude-sdk/session-store.js +0 -84
- package/dist/providers/claude-sdk/session-store.js.map +0 -1
- package/dist/providers/claude-sdk/spawn.js +0 -251
- package/dist/providers/claude-sdk/spawn.js.map +0 -1
- package/dist/providers/claude-sdk/tool-bridge.js +0 -50
- package/dist/providers/claude-sdk/tool-bridge.js.map +0 -1
- package/dist/providers/codex/adapter.js +0 -146
- package/dist/providers/codex/adapter.js.map +0 -1
- package/dist/providers/codex/hooks.js +0 -113
- package/dist/providers/codex/hooks.js.map +0 -1
- package/dist/providers/codex/index.js +0 -40
- package/dist/providers/codex/index.js.map +0 -1
- package/dist/providers/codex/install.js +0 -132
- package/dist/providers/codex/install.js.map +0 -1
- package/dist/providers/codex/spawn.js +0 -203
- package/dist/providers/codex/spawn.js.map +0 -1
- package/dist/providers/cursor/adapter.js +0 -151
- package/dist/providers/cursor/adapter.js.map +0 -1
- package/dist/providers/cursor/hooks.js +0 -208
- package/dist/providers/cursor/hooks.js.map +0 -1
- package/dist/providers/cursor/index.js +0 -36
- package/dist/providers/cursor/index.js.map +0 -1
- package/dist/providers/cursor/install.js +0 -281
- package/dist/providers/cursor/install.js.map +0 -1
- package/dist/providers/cursor/spawn.js +0 -59
- package/dist/providers/cursor/spawn.js.map +0 -1
- package/dist/providers/gemini-cli/adapter.js +0 -158
- package/dist/providers/gemini-cli/adapter.js.map +0 -1
- package/dist/providers/gemini-cli/hooks.js +0 -128
- package/dist/providers/gemini-cli/hooks.js.map +0 -1
- package/dist/providers/gemini-cli/index.js +0 -40
- package/dist/providers/gemini-cli/index.js.map +0 -1
- package/dist/providers/gemini-cli/install.js +0 -124
- package/dist/providers/gemini-cli/install.js.map +0 -1
- package/dist/providers/gemini-cli/spawn.js +0 -195
- package/dist/providers/gemini-cli/spawn.js.map +0 -1
- package/dist/providers/kimi/adapter.js +0 -145
- package/dist/providers/kimi/adapter.js.map +0 -1
- package/dist/providers/kimi/hooks.js +0 -79
- package/dist/providers/kimi/hooks.js.map +0 -1
- package/dist/providers/kimi/index.js +0 -40
- package/dist/providers/kimi/index.js.map +0 -1
- package/dist/providers/kimi/install.js +0 -124
- package/dist/providers/kimi/install.js.map +0 -1
- package/dist/providers/kimi/spawn.js +0 -225
- package/dist/providers/kimi/spawn.js.map +0 -1
- package/dist/providers/openai-sdk/adapter.js +0 -121
- package/dist/providers/openai-sdk/adapter.js.map +0 -1
- package/dist/providers/openai-sdk/guardrails.js +0 -174
- package/dist/providers/openai-sdk/guardrails.js.map +0 -1
- package/dist/providers/openai-sdk/handoff.js +0 -128
- package/dist/providers/openai-sdk/handoff.js.map +0 -1
- package/dist/providers/openai-sdk/index.js +0 -40
- package/dist/providers/openai-sdk/index.js.map +0 -1
- package/dist/providers/openai-sdk/install.js +0 -120
- package/dist/providers/openai-sdk/install.js.map +0 -1
- package/dist/providers/openai-sdk/spawn.js +0 -361
- package/dist/providers/openai-sdk/spawn.js.map +0 -1
- package/dist/providers/openai-sdk/tracing.js +0 -159
- package/dist/providers/openai-sdk/tracing.js.map +0 -1
- package/dist/providers/opencode/adapter.js +0 -166
- package/dist/providers/opencode/adapter.js.map +0 -1
- package/dist/providers/opencode/hooks.js +0 -206
- package/dist/providers/opencode/hooks.js.map +0 -1
- package/dist/providers/opencode/index.js +0 -37
- package/dist/providers/opencode/index.js.map +0 -1
- package/dist/providers/opencode/install.js +0 -242
- package/dist/providers/opencode/install.js.map +0 -1
- package/dist/providers/opencode/spawn.js +0 -257
- package/dist/providers/opencode/spawn.js.map +0 -1
- package/dist/providers/pi/adapter.js +0 -220
- package/dist/providers/pi/adapter.js.map +0 -1
- package/dist/providers/pi/hooks.js +0 -223
- package/dist/providers/pi/hooks.js.map +0 -1
- package/dist/providers/pi/index.js +0 -38
- package/dist/providers/pi/index.js.map +0 -1
- package/dist/providers/pi/install.js +0 -183
- package/dist/providers/pi/install.js.map +0 -1
- package/dist/providers/pi/spawn.js +0 -187
- package/dist/providers/pi/spawn.js.map +0 -1
- package/dist/providers/shared/conduit-trace-writer.js +0 -65
- package/dist/providers/shared/conduit-trace-writer.js.map +0 -1
- package/dist/providers/shared/hook-template-installer.js +0 -209
- package/dist/providers/shared/hook-template-installer.js.map +0 -1
- package/dist/providers/shared/paths.js +0 -77
- package/dist/providers/shared/paths.js.map +0 -1
- package/dist/providers/shared/sdk-result-mapper.js +0 -55
- package/dist/providers/shared/sdk-result-mapper.js.map +0 -1
- package/dist/providers/shared/transcript-reader.js +0 -124
- package/dist/providers/shared/transcript-reader.js.map +0 -1
- package/dist/registry.js +0 -92
- package/dist/registry.js.map +0 -1
|
@@ -1,158 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Gemini CLI Adapter
|
|
3
|
-
*
|
|
4
|
-
* Main CLEOProviderAdapter implementation for Google Gemini CLI.
|
|
5
|
-
* Provides hooks and install capabilities for CLEO integration.
|
|
6
|
-
*
|
|
7
|
-
* @task T161
|
|
8
|
-
* @epic T134
|
|
9
|
-
*/
|
|
10
|
-
import { exec } from 'node:child_process';
|
|
11
|
-
import { existsSync } from 'node:fs';
|
|
12
|
-
import { homedir } from 'node:os';
|
|
13
|
-
import { join } from 'node:path';
|
|
14
|
-
import { promisify } from 'node:util';
|
|
15
|
-
import { GeminiCliHookProvider } from './hooks.js';
|
|
16
|
-
import { GeminiCliInstallProvider } from './install.js';
|
|
17
|
-
const execAsync = promisify(exec);
|
|
18
|
-
/**
|
|
19
|
-
* CLEO provider adapter for Google Gemini CLI.
|
|
20
|
-
*
|
|
21
|
-
* Bridges CLEO's adapter system with Gemini CLI's native capabilities:
|
|
22
|
-
* - Hooks: Maps Gemini CLI events (SessionStart, PreToolUse, etc.) to CAAMP events
|
|
23
|
-
* - Install: Ensures AGENTS.md references for CLEO instruction files
|
|
24
|
-
*
|
|
25
|
-
* @remarks
|
|
26
|
-
* Gemini CLI supports 10 canonical CAAMP events through its hook system,
|
|
27
|
-
* including PreModel and PostModel which most other providers lack. It has
|
|
28
|
-
* no spawn or transport capabilities. Integration is through AGENTS.md
|
|
29
|
-
* instruction files and the Gemini CLI's configuration at `~/.gemini/`.
|
|
30
|
-
*
|
|
31
|
-
* @task T161
|
|
32
|
-
* @epic T134
|
|
33
|
-
*/
|
|
34
|
-
export class GeminiCliAdapter {
|
|
35
|
-
/** Unique provider identifier. */
|
|
36
|
-
id = 'gemini-cli';
|
|
37
|
-
/** Human-readable provider name. */
|
|
38
|
-
name = 'Gemini CLI';
|
|
39
|
-
/** Adapter version string. */
|
|
40
|
-
version = '1.0.0';
|
|
41
|
-
/** Declared capabilities for this provider. */
|
|
42
|
-
capabilities = {
|
|
43
|
-
supportsHooks: true,
|
|
44
|
-
supportedHookEvents: [
|
|
45
|
-
'SessionStart',
|
|
46
|
-
'SessionEnd',
|
|
47
|
-
'BeforeAgent',
|
|
48
|
-
'AfterAgent',
|
|
49
|
-
'BeforeTool',
|
|
50
|
-
'AfterTool',
|
|
51
|
-
'BeforeModel',
|
|
52
|
-
'AfterModel',
|
|
53
|
-
'PreCompress',
|
|
54
|
-
'Notification',
|
|
55
|
-
],
|
|
56
|
-
supportsSpawn: false,
|
|
57
|
-
supportsInstall: true,
|
|
58
|
-
supportsInstructionFiles: false,
|
|
59
|
-
supportsContextMonitor: false,
|
|
60
|
-
supportsStatusline: false,
|
|
61
|
-
supportsProviderPaths: false,
|
|
62
|
-
supportsTransport: false,
|
|
63
|
-
supportsTaskSync: false,
|
|
64
|
-
};
|
|
65
|
-
/** Hook provider for CAAMP event mapping. */
|
|
66
|
-
hooks;
|
|
67
|
-
/** Install provider for managing instruction files. */
|
|
68
|
-
install;
|
|
69
|
-
/** Project directory this adapter was initialized with, or null. */
|
|
70
|
-
projectDir = null;
|
|
71
|
-
/** Whether {@link initialize} has been called. */
|
|
72
|
-
initialized = false;
|
|
73
|
-
constructor() {
|
|
74
|
-
this.hooks = new GeminiCliHookProvider();
|
|
75
|
-
this.install = new GeminiCliInstallProvider();
|
|
76
|
-
}
|
|
77
|
-
/**
|
|
78
|
-
* Initialize the adapter for a given project directory.
|
|
79
|
-
*
|
|
80
|
-
* @param projectDir - Root directory of the project
|
|
81
|
-
* @task T161
|
|
82
|
-
*/
|
|
83
|
-
async initialize(projectDir) {
|
|
84
|
-
this.projectDir = projectDir;
|
|
85
|
-
this.initialized = true;
|
|
86
|
-
}
|
|
87
|
-
/**
|
|
88
|
-
* Dispose the adapter and clean up resources.
|
|
89
|
-
*
|
|
90
|
-
* Unregisters hooks and releases any tracked state.
|
|
91
|
-
* @task T161
|
|
92
|
-
*/
|
|
93
|
-
async dispose() {
|
|
94
|
-
if (this.hooks.isRegistered()) {
|
|
95
|
-
await this.hooks.unregisterNativeHooks();
|
|
96
|
-
}
|
|
97
|
-
this.initialized = false;
|
|
98
|
-
this.projectDir = null;
|
|
99
|
-
}
|
|
100
|
-
/**
|
|
101
|
-
* Run a health check to verify Gemini CLI is accessible.
|
|
102
|
-
*
|
|
103
|
-
* Checks:
|
|
104
|
-
* 1. Adapter has been initialized
|
|
105
|
-
* 2. Gemini CLI binary is available in PATH
|
|
106
|
-
* 3. ~/.gemini/ configuration directory exists
|
|
107
|
-
*
|
|
108
|
-
* @returns Health status with details about each check
|
|
109
|
-
* @task T161
|
|
110
|
-
*/
|
|
111
|
-
async healthCheck() {
|
|
112
|
-
const details = {};
|
|
113
|
-
if (!this.initialized) {
|
|
114
|
-
return {
|
|
115
|
-
healthy: false,
|
|
116
|
-
provider: this.id,
|
|
117
|
-
details: { error: 'Adapter not initialized' },
|
|
118
|
-
};
|
|
119
|
-
}
|
|
120
|
-
// Check Gemini CLI availability
|
|
121
|
-
let cliAvailable = false;
|
|
122
|
-
try {
|
|
123
|
-
const { stdout } = await execAsync('which gemini');
|
|
124
|
-
cliAvailable = stdout.trim().length > 0;
|
|
125
|
-
details.cliPath = stdout.trim();
|
|
126
|
-
}
|
|
127
|
-
catch {
|
|
128
|
-
details.cliAvailable = false;
|
|
129
|
-
}
|
|
130
|
-
// Check for Gemini CLI config directory
|
|
131
|
-
const geminiConfigDir = join(homedir(), '.gemini');
|
|
132
|
-
const configExists = existsSync(geminiConfigDir);
|
|
133
|
-
details.configDirExists = configExists;
|
|
134
|
-
// Healthy if CLI is available (primary requirement)
|
|
135
|
-
const healthy = cliAvailable;
|
|
136
|
-
details.cliAvailable = cliAvailable;
|
|
137
|
-
return {
|
|
138
|
-
healthy,
|
|
139
|
-
provider: this.id,
|
|
140
|
-
details,
|
|
141
|
-
};
|
|
142
|
-
}
|
|
143
|
-
/**
|
|
144
|
-
* Check whether the adapter has been initialized.
|
|
145
|
-
* @task T161
|
|
146
|
-
*/
|
|
147
|
-
isInitialized() {
|
|
148
|
-
return this.initialized;
|
|
149
|
-
}
|
|
150
|
-
/**
|
|
151
|
-
* Get the project directory this adapter was initialized with.
|
|
152
|
-
* @task T161
|
|
153
|
-
*/
|
|
154
|
-
getProjectDir() {
|
|
155
|
-
return this.projectDir;
|
|
156
|
-
}
|
|
157
|
-
}
|
|
158
|
-
//# sourceMappingURL=adapter.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"adapter.js","sourceRoot":"","sources":["../../../src/providers/gemini-cli/adapter.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAEH,OAAO,EAAE,IAAI,EAAE,MAAM,oBAAoB,CAAC;AAC1C,OAAO,EAAE,UAAU,EAAE,MAAM,SAAS,CAAC;AACrC,OAAO,EAAE,OAAO,EAAE,MAAM,SAAS,CAAC;AAClC,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AACjC,OAAO,EAAE,SAAS,EAAE,MAAM,WAAW,CAAC;AAMtC,OAAO,EAAE,qBAAqB,EAAE,MAAM,YAAY,CAAC;AACnD,OAAO,EAAE,wBAAwB,EAAE,MAAM,cAAc,CAAC;AAExD,MAAM,SAAS,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC;AAElC;;;;;;;;;;;;;;;GAeG;AACH,MAAM,OAAO,gBAAgB;IAC3B,kCAAkC;IACzB,EAAE,GAAG,YAAY,CAAC;IAC3B,oCAAoC;IAC3B,IAAI,GAAG,YAAY,CAAC;IAC7B,8BAA8B;IACrB,OAAO,GAAG,OAAO,CAAC;IAE3B,+CAA+C;IAC/C,YAAY,GAAwB;QAClC,aAAa,EAAE,IAAI;QACnB,mBAAmB,EAAE;YACnB,cAAc;YACd,YAAY;YACZ,aAAa;YACb,YAAY;YACZ,YAAY;YACZ,WAAW;YACX,aAAa;YACb,YAAY;YACZ,aAAa;YACb,cAAc;SACf;QACD,aAAa,EAAE,KAAK;QACpB,eAAe,EAAE,IAAI;QACrB,wBAAwB,EAAE,KAAK;QAC/B,sBAAsB,EAAE,KAAK;QAC7B,kBAAkB,EAAE,KAAK;QACzB,qBAAqB,EAAE,KAAK;QAC5B,iBAAiB,EAAE,KAAK;QACxB,gBAAgB,EAAE,KAAK;KACxB,CAAC;IAEF,6CAA6C;IAC7C,KAAK,CAAwB;IAC7B,uDAAuD;IACvD,OAAO,CAA2B;IAElC,oEAAoE;IAC5D,UAAU,GAAkB,IAAI,CAAC;IACzC,kDAAkD;IAC1C,WAAW,GAAG,KAAK,CAAC;IAE5B;QACE,IAAI,CAAC,KAAK,GAAG,IAAI,qBAAqB,EAAE,CAAC;QACzC,IAAI,CAAC,OAAO,GAAG,IAAI,wBAAwB,EAAE,CAAC;IAChD,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,UAAU,CAAC,UAAkB;QACjC,IAAI,CAAC,UAAU,GAAG,UAAU,CAAC;QAC7B,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC;IAC1B,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,OAAO;QACX,IAAI,IAAI,CAAC,KAAK,CAAC,YAAY,EAAE,EAAE,CAAC;YAC9B,MAAM,IAAI,CAAC,KAAK,CAAC,qBAAqB,EAAE,CAAC;QAC3C,CAAC;QACD,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC;QACzB,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC;IACzB,CAAC;IAED;;;;;;;;;;OAUG;IACH,KAAK,CAAC,WAAW;QACf,MAAM,OAAO,GAA4B,EAAE,CAAC;QAE5C,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC;YACtB,OAAO;gBACL,OAAO,EAAE,KAAK;gBACd,QAAQ,EAAE,IAAI,CAAC,EAAE;gBACjB,OAAO,EAAE,EAAE,KAAK,EAAE,yBAAyB,EAAE;aAC9C,CAAC;QACJ,CAAC;QAED,gCAAgC;QAChC,IAAI,YAAY,GAAG,KAAK,CAAC;QACzB,IAAI,CAAC;YACH,MAAM,EAAE,MAAM,EAAE,GAAG,MAAM,SAAS,CAAC,cAAc,CAAC,CAAC;YACnD,YAAY,GAAG,MAAM,CAAC,IAAI,EAAE,CAAC,MAAM,GAAG,CAAC,CAAC;YACxC,OAAO,CAAC,OAAO,GAAG,MAAM,CAAC,IAAI,EAAE,CAAC;QAClC,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,CAAC,YAAY,GAAG,KAAK,CAAC;QAC/B,CAAC;QAED,wCAAwC;QACxC,MAAM,eAAe,GAAG,IAAI,CAAC,OAAO,EAAE,EAAE,SAAS,CAAC,CAAC;QACnD,MAAM,YAAY,GAAG,UAAU,CAAC,eAAe,CAAC,CAAC;QACjD,OAAO,CAAC,eAAe,GAAG,YAAY,CAAC;QAEvC,oDAAoD;QACpD,MAAM,OAAO,GAAG,YAAY,CAAC;QAC7B,OAAO,CAAC,YAAY,GAAG,YAAY,CAAC;QAEpC,OAAO;YACL,OAAO;YACP,QAAQ,EAAE,IAAI,CAAC,EAAE;YACjB,OAAO;SACR,CAAC;IACJ,CAAC;IAED;;;OAGG;IACH,aAAa;QACX,OAAO,IAAI,CAAC,WAAW,CAAC;IAC1B,CAAC;IAED;;;OAGG;IACH,aAAa;QACX,OAAO,IAAI,CAAC,UAAU,CAAC;IACzB,CAAC;CACF"}
|
|
@@ -1,128 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Gemini CLI Hook Provider
|
|
3
|
-
*
|
|
4
|
-
* Maps Gemini CLI's native hook events to CAAMP canonical hook events.
|
|
5
|
-
* Gemini CLI supports 11 canonical events through its hook system.
|
|
6
|
-
*
|
|
7
|
-
* Gemini CLI event mapping:
|
|
8
|
-
* - SessionStart -> SessionStart
|
|
9
|
-
* - SessionEnd -> SessionEnd
|
|
10
|
-
* - PromptSubmit -> BeforeAgent
|
|
11
|
-
* - ResponseComplete -> AfterAgent
|
|
12
|
-
* - PreToolUse -> BeforeTool
|
|
13
|
-
* - PostToolUse -> AfterTool
|
|
14
|
-
* - PreModel -> BeforeModel
|
|
15
|
-
* - PostModel -> AfterModel
|
|
16
|
-
* - PreCompact -> PreCompress
|
|
17
|
-
* - Notification -> Notification
|
|
18
|
-
*
|
|
19
|
-
* @task T161
|
|
20
|
-
* @epic T134
|
|
21
|
-
*/
|
|
22
|
-
import { homedir } from 'node:os';
|
|
23
|
-
import { join } from 'node:path';
|
|
24
|
-
import { readLatestTranscript } from '../shared/transcript-reader.js';
|
|
25
|
-
/**
|
|
26
|
-
* Mapping from Gemini CLI native event names to CAAMP canonical event names.
|
|
27
|
-
*/
|
|
28
|
-
const GEMINI_CLI_EVENT_MAP = {
|
|
29
|
-
SessionStart: 'SessionStart',
|
|
30
|
-
SessionEnd: 'SessionEnd',
|
|
31
|
-
PromptSubmit: 'BeforeAgent',
|
|
32
|
-
ResponseComplete: 'AfterAgent',
|
|
33
|
-
PreToolUse: 'BeforeTool',
|
|
34
|
-
PostToolUse: 'AfterTool',
|
|
35
|
-
PreModel: 'BeforeModel',
|
|
36
|
-
PostModel: 'AfterModel',
|
|
37
|
-
PreCompact: 'PreCompress',
|
|
38
|
-
Notification: 'Notification',
|
|
39
|
-
};
|
|
40
|
-
/**
|
|
41
|
-
* Hook provider for Gemini CLI.
|
|
42
|
-
*
|
|
43
|
-
* Gemini CLI registers hooks via its configuration system at
|
|
44
|
-
* ~/.gemini/. Hook handlers are shell scripts or commands that
|
|
45
|
-
* execute when the corresponding event fires.
|
|
46
|
-
*
|
|
47
|
-
* Since hooks are registered through the config system (managed by
|
|
48
|
-
* the install provider), registerNativeHooks and unregisterNativeHooks
|
|
49
|
-
* track registration state without performing filesystem operations.
|
|
50
|
-
*
|
|
51
|
-
* @remarks
|
|
52
|
-
* Gemini CLI uses its own event naming convention (e.g. BeforeAgent,
|
|
53
|
-
* AfterTool, PreCompress) which differs from both the PascalCase CAAMP
|
|
54
|
-
* canonical names and other providers' conventions. The static event map
|
|
55
|
-
* covers all 10 supported canonical events.
|
|
56
|
-
*
|
|
57
|
-
* @task T161
|
|
58
|
-
* @epic T134
|
|
59
|
-
*/
|
|
60
|
-
export class GeminiCliHookProvider {
|
|
61
|
-
/** Whether hooks have been registered for the current session. */
|
|
62
|
-
registered = false;
|
|
63
|
-
/**
|
|
64
|
-
* Map a Gemini CLI native event name to a CAAMP hook event name.
|
|
65
|
-
*
|
|
66
|
-
* @param providerEvent - Gemini CLI event name (e.g. "SessionStart", "PreToolUse")
|
|
67
|
-
* @returns CAAMP event name or null if unmapped
|
|
68
|
-
* @task T161
|
|
69
|
-
*/
|
|
70
|
-
mapProviderEvent(providerEvent) {
|
|
71
|
-
return GEMINI_CLI_EVENT_MAP[providerEvent] ?? null;
|
|
72
|
-
}
|
|
73
|
-
/**
|
|
74
|
-
* Register native hooks for a project.
|
|
75
|
-
*
|
|
76
|
-
* For Gemini CLI, hooks are registered via the config system
|
|
77
|
-
* (~/.gemini/), which is handled by the install provider.
|
|
78
|
-
* This method marks hooks as registered without performing
|
|
79
|
-
* filesystem operations.
|
|
80
|
-
*
|
|
81
|
-
* @param _projectDir - Project directory (unused; hooks are global)
|
|
82
|
-
* @task T161
|
|
83
|
-
*/
|
|
84
|
-
async registerNativeHooks(_projectDir) {
|
|
85
|
-
this.registered = true;
|
|
86
|
-
}
|
|
87
|
-
/**
|
|
88
|
-
* Unregister native hooks.
|
|
89
|
-
*
|
|
90
|
-
* For Gemini CLI, this is a no-op since hooks are managed through
|
|
91
|
-
* the config system. Unregistration happens via the install
|
|
92
|
-
* provider's uninstall method.
|
|
93
|
-
* @task T161
|
|
94
|
-
*/
|
|
95
|
-
async unregisterNativeHooks() {
|
|
96
|
-
this.registered = false;
|
|
97
|
-
}
|
|
98
|
-
/**
|
|
99
|
-
* Check whether hooks have been registered via registerNativeHooks.
|
|
100
|
-
* @task T161
|
|
101
|
-
*/
|
|
102
|
-
isRegistered() {
|
|
103
|
-
return this.registered;
|
|
104
|
-
}
|
|
105
|
-
/**
|
|
106
|
-
* Get the full event mapping for introspection/debugging.
|
|
107
|
-
* @task T161
|
|
108
|
-
*/
|
|
109
|
-
getEventMap() {
|
|
110
|
-
return { ...GEMINI_CLI_EVENT_MAP };
|
|
111
|
-
}
|
|
112
|
-
/**
|
|
113
|
-
* Extract a plain-text transcript from Gemini CLI session data.
|
|
114
|
-
*
|
|
115
|
-
* Reads the most recent JSON/JSONL session file under `~/.gemini/`
|
|
116
|
-
* and returns its turns as a flat string for brain observation extraction.
|
|
117
|
-
*
|
|
118
|
-
* Returns null when no session data is found or on any read error.
|
|
119
|
-
*
|
|
120
|
-
* @param _sessionId - CLEO session ID (unused; reads the most recent file)
|
|
121
|
-
* @param _projectDir - Project directory (unused; Gemini CLI uses global paths)
|
|
122
|
-
* @task T161 @epic T134
|
|
123
|
-
*/
|
|
124
|
-
async getTranscript(_sessionId, _projectDir) {
|
|
125
|
-
return readLatestTranscript(join(homedir(), '.gemini'));
|
|
126
|
-
}
|
|
127
|
-
}
|
|
128
|
-
//# sourceMappingURL=hooks.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"hooks.js","sourceRoot":"","sources":["../../../src/providers/gemini-cli/hooks.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;;GAoBG;AAEH,OAAO,EAAE,OAAO,EAAE,MAAM,SAAS,CAAC;AAClC,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AAEjC,OAAO,EAAE,oBAAoB,EAAE,MAAM,gCAAgC,CAAC;AAEtE;;GAEG;AACH,MAAM,oBAAoB,GAA2B;IACnD,YAAY,EAAE,cAAc;IAC5B,UAAU,EAAE,YAAY;IACxB,YAAY,EAAE,aAAa;IAC3B,gBAAgB,EAAE,YAAY;IAC9B,UAAU,EAAE,YAAY;IACxB,WAAW,EAAE,WAAW;IACxB,QAAQ,EAAE,aAAa;IACvB,SAAS,EAAE,YAAY;IACvB,UAAU,EAAE,aAAa;IACzB,YAAY,EAAE,cAAc;CAC7B,CAAC;AAEF;;;;;;;;;;;;;;;;;;;GAmBG;AACH,MAAM,OAAO,qBAAqB;IAChC,kEAAkE;IAC1D,UAAU,GAAG,KAAK,CAAC;IAE3B;;;;;;OAMG;IACH,gBAAgB,CAAC,aAAqB;QACpC,OAAO,oBAAoB,CAAC,aAAa,CAAC,IAAI,IAAI,CAAC;IACrD,CAAC;IAED;;;;;;;;;;OAUG;IACH,KAAK,CAAC,mBAAmB,CAAC,WAAmB;QAC3C,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC;IACzB,CAAC;IAED;;;;;;;OAOG;IACH,KAAK,CAAC,qBAAqB;QACzB,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC;IAC1B,CAAC;IAED;;;OAGG;IACH,YAAY;QACV,OAAO,IAAI,CAAC,UAAU,CAAC;IACzB,CAAC;IAED;;;OAGG;IACH,WAAW;QACT,OAAO,EAAE,GAAG,oBAAoB,EAAE,CAAC;IACrC,CAAC;IAED;;;;;;;;;;;OAWG;IACH,KAAK,CAAC,aAAa,CAAC,UAAkB,EAAE,WAAmB;QACzD,OAAO,oBAAoB,CAAC,IAAI,CAAC,OAAO,EAAE,EAAE,SAAS,CAAC,CAAC,CAAC;IAC1D,CAAC;CACF"}
|
|
@@ -1,40 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @packageDocumentation
|
|
3
|
-
*
|
|
4
|
-
* CLEO provider adapter for Google Gemini CLI.
|
|
5
|
-
* Default export is the adapter class for dynamic loading by AdapterManager.
|
|
6
|
-
*
|
|
7
|
-
* @task T161
|
|
8
|
-
* @epic T134
|
|
9
|
-
*/
|
|
10
|
-
import { GeminiCliAdapter } from './adapter.js';
|
|
11
|
-
export { GeminiCliAdapter } from './adapter.js';
|
|
12
|
-
export { GeminiCliHookProvider } from './hooks.js';
|
|
13
|
-
export { GeminiCliInstallProvider } from './install.js';
|
|
14
|
-
export { GeminiCliSpawnProvider } from './spawn.js';
|
|
15
|
-
export default GeminiCliAdapter;
|
|
16
|
-
/**
|
|
17
|
-
* Factory function for creating adapter instances.
|
|
18
|
-
* Used by AdapterManager's dynamic import fallback.
|
|
19
|
-
*
|
|
20
|
-
* @remarks
|
|
21
|
-
* This is the primary entry point for dynamic adapter loading.
|
|
22
|
-
* AdapterManager calls this function when it resolves the gemini-cli
|
|
23
|
-
* provider via its import-based discovery mechanism.
|
|
24
|
-
*
|
|
25
|
-
* @returns A new {@link GeminiCliAdapter} instance ready for initialization
|
|
26
|
-
*
|
|
27
|
-
* @example
|
|
28
|
-
* ```typescript
|
|
29
|
-
* import { createAdapter } from '@cleocode/adapters/providers/gemini-cli';
|
|
30
|
-
*
|
|
31
|
-
* const adapter = createAdapter();
|
|
32
|
-
* await adapter.initialize('/path/to/project');
|
|
33
|
-
* ```
|
|
34
|
-
*
|
|
35
|
-
* @task T161
|
|
36
|
-
*/
|
|
37
|
-
export function createAdapter() {
|
|
38
|
-
return new GeminiCliAdapter();
|
|
39
|
-
}
|
|
40
|
-
//# sourceMappingURL=index.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/providers/gemini-cli/index.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAEH,OAAO,EAAE,gBAAgB,EAAE,MAAM,cAAc,CAAC;AAEhD,OAAO,EAAE,gBAAgB,EAAE,MAAM,cAAc,CAAC;AAChD,OAAO,EAAE,qBAAqB,EAAE,MAAM,YAAY,CAAC;AACnD,OAAO,EAAE,wBAAwB,EAAE,MAAM,cAAc,CAAC;AACxD,OAAO,EAAE,sBAAsB,EAAE,MAAM,YAAY,CAAC;AAEpD,eAAe,gBAAgB,CAAC;AAEhC;;;;;;;;;;;;;;;;;;;;GAoBG;AACH,MAAM,UAAU,aAAa;IAC3B,OAAO,IAAI,gBAAgB,EAAE,CAAC;AAChC,CAAC"}
|
|
@@ -1,124 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Gemini CLI Install Provider
|
|
3
|
-
*
|
|
4
|
-
* Handles CLEO installation into Gemini CLI environments:
|
|
5
|
-
* - Ensures AGENTS.md has CLEO @-references
|
|
6
|
-
*
|
|
7
|
-
* @task T161
|
|
8
|
-
* @epic T134
|
|
9
|
-
*/
|
|
10
|
-
import { existsSync, readFileSync, writeFileSync } from 'node:fs';
|
|
11
|
-
import { join } from 'node:path';
|
|
12
|
-
/** Lines that should appear in AGENTS.md to reference CLEO. */
|
|
13
|
-
const INSTRUCTION_REFERENCES = ['@~/.cleo/templates/CLEO-INJECTION.md', '@.cleo/memory-bridge.md'];
|
|
14
|
-
/**
|
|
15
|
-
* Install provider for Gemini CLI.
|
|
16
|
-
*
|
|
17
|
-
* Manages CLEO's integration with Gemini CLI by:
|
|
18
|
-
* 1. Ensuring AGENTS.md contains @-references to CLEO instruction files
|
|
19
|
-
*
|
|
20
|
-
* @remarks
|
|
21
|
-
* Installation is idempotent -- running install multiple times on the same
|
|
22
|
-
* project produces the same result. Only AGENTS.md is managed; Gemini CLI
|
|
23
|
-
* does not have an MCP or plugin registration mechanism.
|
|
24
|
-
*
|
|
25
|
-
* @task T161
|
|
26
|
-
* @epic T134
|
|
27
|
-
*/
|
|
28
|
-
export class GeminiCliInstallProvider {
|
|
29
|
-
/**
|
|
30
|
-
* Install CLEO into a Gemini CLI environment.
|
|
31
|
-
*
|
|
32
|
-
* @param options - Installation options including project directory
|
|
33
|
-
* @returns Result describing what was installed
|
|
34
|
-
* @task T161
|
|
35
|
-
*/
|
|
36
|
-
async install(options) {
|
|
37
|
-
const { projectDir } = options;
|
|
38
|
-
const installedAt = new Date().toISOString();
|
|
39
|
-
let instructionFileUpdated = false;
|
|
40
|
-
const details = {};
|
|
41
|
-
// Step 1: Ensure AGENTS.md has @-references
|
|
42
|
-
instructionFileUpdated = this.updateInstructionFile(projectDir);
|
|
43
|
-
if (instructionFileUpdated) {
|
|
44
|
-
details.instructionFile = join(projectDir, 'AGENTS.md');
|
|
45
|
-
}
|
|
46
|
-
return {
|
|
47
|
-
success: true,
|
|
48
|
-
installedAt,
|
|
49
|
-
instructionFileUpdated,
|
|
50
|
-
details,
|
|
51
|
-
};
|
|
52
|
-
}
|
|
53
|
-
/**
|
|
54
|
-
* Uninstall CLEO from the Gemini CLI environment.
|
|
55
|
-
*
|
|
56
|
-
* Does not remove AGENTS.md references (they are harmless if CLEO is not present).
|
|
57
|
-
* @task T161
|
|
58
|
-
*/
|
|
59
|
-
async uninstall() {
|
|
60
|
-
// No-op: no MCP registration to remove
|
|
61
|
-
}
|
|
62
|
-
/**
|
|
63
|
-
* Check whether CLEO is installed in the Gemini CLI environment.
|
|
64
|
-
*
|
|
65
|
-
* Checks for CLEO references in AGENTS.md.
|
|
66
|
-
* @task T161
|
|
67
|
-
*/
|
|
68
|
-
async isInstalled() {
|
|
69
|
-
const agentsMdPath = join(process.cwd(), 'AGENTS.md');
|
|
70
|
-
if (existsSync(agentsMdPath)) {
|
|
71
|
-
try {
|
|
72
|
-
const content = readFileSync(agentsMdPath, 'utf-8');
|
|
73
|
-
if (INSTRUCTION_REFERENCES.some((ref) => content.includes(ref))) {
|
|
74
|
-
return true;
|
|
75
|
-
}
|
|
76
|
-
}
|
|
77
|
-
catch {
|
|
78
|
-
// Fall through
|
|
79
|
-
}
|
|
80
|
-
}
|
|
81
|
-
return false;
|
|
82
|
-
}
|
|
83
|
-
/**
|
|
84
|
-
* Ensure AGENTS.md contains @-references to CLEO instruction files.
|
|
85
|
-
*
|
|
86
|
-
* Creates AGENTS.md if it does not exist. Appends any missing references.
|
|
87
|
-
*
|
|
88
|
-
* @param projectDir - Project root directory
|
|
89
|
-
* @task T161
|
|
90
|
-
*/
|
|
91
|
-
async ensureInstructionReferences(projectDir) {
|
|
92
|
-
this.updateInstructionFile(projectDir);
|
|
93
|
-
}
|
|
94
|
-
/**
|
|
95
|
-
* Update AGENTS.md with CLEO @-references.
|
|
96
|
-
*
|
|
97
|
-
* @param projectDir - Project root directory
|
|
98
|
-
* @returns true if the file was created or modified
|
|
99
|
-
*/
|
|
100
|
-
updateInstructionFile(projectDir) {
|
|
101
|
-
const agentsMdPath = join(projectDir, 'AGENTS.md');
|
|
102
|
-
let content = '';
|
|
103
|
-
let existed = false;
|
|
104
|
-
if (existsSync(agentsMdPath)) {
|
|
105
|
-
content = readFileSync(agentsMdPath, 'utf-8');
|
|
106
|
-
existed = true;
|
|
107
|
-
}
|
|
108
|
-
const missingRefs = INSTRUCTION_REFERENCES.filter((ref) => !content.includes(ref));
|
|
109
|
-
if (missingRefs.length === 0) {
|
|
110
|
-
return false;
|
|
111
|
-
}
|
|
112
|
-
const refsBlock = missingRefs.join('\n');
|
|
113
|
-
if (existed) {
|
|
114
|
-
const separator = content.endsWith('\n') ? '' : '\n';
|
|
115
|
-
content = content + separator + refsBlock + '\n';
|
|
116
|
-
}
|
|
117
|
-
else {
|
|
118
|
-
content = refsBlock + '\n';
|
|
119
|
-
}
|
|
120
|
-
writeFileSync(agentsMdPath, content, 'utf-8');
|
|
121
|
-
return true;
|
|
122
|
-
}
|
|
123
|
-
}
|
|
124
|
-
//# sourceMappingURL=install.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"install.js","sourceRoot":"","sources":["../../../src/providers/gemini-cli/install.ts"],"names":[],"mappings":"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"}
|
|
@@ -1,195 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Gemini CLI Spawn Provider
|
|
3
|
-
*
|
|
4
|
-
* Implements `AdapterSpawnProvider` for the Google Gemini CLI (`gemini` binary).
|
|
5
|
-
*
|
|
6
|
-
* The `gemini` binary is the Google Gemini CLI agent, available at:
|
|
7
|
-
* https://github.com/google-gemini/gemini-cli
|
|
8
|
-
*
|
|
9
|
-
* Invocation: `gemini --yolo < <prompt-file>`
|
|
10
|
-
*
|
|
11
|
-
* The provider pipes the prompt via stdin using the `--yolo` flag, which
|
|
12
|
-
* enables non-interactive mode (auto-approve all actions). Processes run
|
|
13
|
-
* detached and are tracked by PID for listing and termination.
|
|
14
|
-
*
|
|
15
|
-
* If the `gemini` binary is not found, `canSpawn()` returns `false` with a
|
|
16
|
-
* graceful error — no crash.
|
|
17
|
-
*
|
|
18
|
-
* @remarks
|
|
19
|
-
* The Gemini CLI supports a `--model` flag to select the model family and a
|
|
20
|
-
* `--yolo` flag for non-interactive headless execution (equivalent to Claude
|
|
21
|
-
* Code's `--dangerously-skip-permissions`). Prompts are supplied via stdin
|
|
22
|
-
* when run in `--yolo` mode. Install: `npm install -g @google/gemini-cli`
|
|
23
|
-
* or see the GitHub repo above.
|
|
24
|
-
*
|
|
25
|
-
* @task T648
|
|
26
|
-
*/
|
|
27
|
-
import { exec, spawn as nodeSpawn } from 'node:child_process';
|
|
28
|
-
import { promisify } from 'node:util';
|
|
29
|
-
import { getErrorMessage } from '@cleocode/contracts';
|
|
30
|
-
const execAsync = promisify(exec);
|
|
31
|
-
/** Default Gemini model for subagent spawns. */
|
|
32
|
-
const DEFAULT_MODEL = 'gemini-2.5-pro';
|
|
33
|
-
/**
|
|
34
|
-
* Spawn provider for the Google Gemini CLI.
|
|
35
|
-
*
|
|
36
|
-
* Spawns detached Gemini CLI processes for subagent execution. Each spawn
|
|
37
|
-
* pipes its prompt via stdin, then runs
|
|
38
|
-
* `gemini --yolo --model <model>` as a detached, unref'd child process.
|
|
39
|
-
*
|
|
40
|
-
* @remarks
|
|
41
|
-
* `canSpawn()` returns `false` (with no crash) when the `gemini` binary is
|
|
42
|
-
* not found in PATH. Install instructions are emitted via `console.warn`
|
|
43
|
-
* once to help operators discover the binary is missing.
|
|
44
|
-
*
|
|
45
|
-
* Processes are tracked by instance ID in an in-memory map and verified
|
|
46
|
-
* via `kill(pid, 0)` liveness checks.
|
|
47
|
-
*
|
|
48
|
-
* @task T648
|
|
49
|
-
*/
|
|
50
|
-
export class GeminiCliSpawnProvider {
|
|
51
|
-
/** Map of instance IDs to tracked process info. */
|
|
52
|
-
processMap = new Map();
|
|
53
|
-
/**
|
|
54
|
-
* Check if the Gemini CLI is available in PATH.
|
|
55
|
-
*
|
|
56
|
-
* @returns `true` if `gemini` is found via `which`
|
|
57
|
-
*/
|
|
58
|
-
async canSpawn() {
|
|
59
|
-
try {
|
|
60
|
-
await execAsync('which gemini');
|
|
61
|
-
return true;
|
|
62
|
-
}
|
|
63
|
-
catch {
|
|
64
|
-
console.warn('[GeminiCliSpawnProvider] gemini CLI not found. ' +
|
|
65
|
-
'Install: npm install -g @google/gemini-cli ' +
|
|
66
|
-
'Docs: https://github.com/google-gemini/gemini-cli');
|
|
67
|
-
return false;
|
|
68
|
-
}
|
|
69
|
-
}
|
|
70
|
-
/**
|
|
71
|
-
* Spawn a subagent via the Gemini CLI.
|
|
72
|
-
*
|
|
73
|
-
* Pipes the enriched prompt to stdin and spawns a detached Gemini
|
|
74
|
-
* process. The process runs independently of the parent.
|
|
75
|
-
*
|
|
76
|
-
* @param context - Spawn context with taskId, prompt, and options
|
|
77
|
-
* @returns Spawn result with instance ID and status
|
|
78
|
-
*/
|
|
79
|
-
async spawn(context) {
|
|
80
|
-
const instanceId = `gemini-${Date.now()}-${Math.random().toString(36).substring(2, 9)}`;
|
|
81
|
-
const startTime = new Date().toISOString();
|
|
82
|
-
try {
|
|
83
|
-
// Enrich prompt with CANT bundle, memory bridge, and mental model.
|
|
84
|
-
// Best-effort: if CANT context is unavailable, the raw prompt is used.
|
|
85
|
-
let enrichedPrompt = context.prompt;
|
|
86
|
-
try {
|
|
87
|
-
const { buildCantEnrichedPrompt } = await import('../../cant-context.js');
|
|
88
|
-
enrichedPrompt = await buildCantEnrichedPrompt({
|
|
89
|
-
projectDir: context.workingDirectory ?? process.cwd(),
|
|
90
|
-
basePrompt: context.prompt,
|
|
91
|
-
agentName: context.options?.agentName ?? undefined,
|
|
92
|
-
});
|
|
93
|
-
}
|
|
94
|
-
catch {
|
|
95
|
-
// CANT enrichment unavailable — use raw prompt
|
|
96
|
-
}
|
|
97
|
-
const model = context.options?.model ?? DEFAULT_MODEL;
|
|
98
|
-
// --yolo: non-interactive batch mode (auto-approve all actions)
|
|
99
|
-
// --model: select the Gemini model variant
|
|
100
|
-
// Prompt is supplied via stdin (pipe)
|
|
101
|
-
const args = ['--yolo', '--model', model];
|
|
102
|
-
const spawnOpts = {
|
|
103
|
-
detached: true,
|
|
104
|
-
stdio: ['pipe', 'ignore', 'ignore'],
|
|
105
|
-
};
|
|
106
|
-
if (context.workingDirectory) {
|
|
107
|
-
spawnOpts.cwd = context.workingDirectory;
|
|
108
|
-
}
|
|
109
|
-
const child = nodeSpawn('gemini', args, spawnOpts);
|
|
110
|
-
// Write the prompt to stdin then close so the CLI receives it.
|
|
111
|
-
if (child.stdin) {
|
|
112
|
-
child.stdin.write(enrichedPrompt, 'utf-8');
|
|
113
|
-
child.stdin.end();
|
|
114
|
-
}
|
|
115
|
-
child.unref();
|
|
116
|
-
if (child.pid) {
|
|
117
|
-
this.processMap.set(instanceId, {
|
|
118
|
-
pid: child.pid,
|
|
119
|
-
taskId: context.taskId,
|
|
120
|
-
startTime,
|
|
121
|
-
});
|
|
122
|
-
}
|
|
123
|
-
child.on('exit', () => {
|
|
124
|
-
this.processMap.delete(instanceId);
|
|
125
|
-
});
|
|
126
|
-
return {
|
|
127
|
-
instanceId,
|
|
128
|
-
taskId: context.taskId,
|
|
129
|
-
providerId: 'gemini-cli',
|
|
130
|
-
status: 'running',
|
|
131
|
-
startTime,
|
|
132
|
-
};
|
|
133
|
-
}
|
|
134
|
-
catch (error) {
|
|
135
|
-
console.error(`[GeminiCliSpawnProvider] Failed to spawn: ${getErrorMessage(error)}`);
|
|
136
|
-
return {
|
|
137
|
-
instanceId,
|
|
138
|
-
taskId: context.taskId,
|
|
139
|
-
providerId: 'gemini-cli',
|
|
140
|
-
status: 'failed',
|
|
141
|
-
startTime,
|
|
142
|
-
endTime: new Date().toISOString(),
|
|
143
|
-
error: getErrorMessage(error),
|
|
144
|
-
};
|
|
145
|
-
}
|
|
146
|
-
}
|
|
147
|
-
/**
|
|
148
|
-
* List currently running Gemini CLI subagent processes.
|
|
149
|
-
*
|
|
150
|
-
* Checks each tracked process via kill(pid, 0) to verify it is still alive.
|
|
151
|
-
* Dead processes are automatically cleaned from the tracking map.
|
|
152
|
-
*
|
|
153
|
-
* @returns Array of spawn results for running processes
|
|
154
|
-
*/
|
|
155
|
-
async listRunning() {
|
|
156
|
-
const running = [];
|
|
157
|
-
for (const [instanceId, tracked] of this.processMap.entries()) {
|
|
158
|
-
try {
|
|
159
|
-
process.kill(tracked.pid, 0);
|
|
160
|
-
running.push({
|
|
161
|
-
instanceId,
|
|
162
|
-
taskId: tracked.taskId,
|
|
163
|
-
providerId: 'gemini-cli',
|
|
164
|
-
status: 'running',
|
|
165
|
-
startTime: tracked.startTime,
|
|
166
|
-
});
|
|
167
|
-
}
|
|
168
|
-
catch {
|
|
169
|
-
this.processMap.delete(instanceId);
|
|
170
|
-
}
|
|
171
|
-
}
|
|
172
|
-
return running;
|
|
173
|
-
}
|
|
174
|
-
/**
|
|
175
|
-
* Terminate a running spawn by instance ID.
|
|
176
|
-
*
|
|
177
|
-
* Sends SIGTERM to the tracked process. If the process is not found
|
|
178
|
-
* or has already exited, this is a no-op.
|
|
179
|
-
*
|
|
180
|
-
* @param instanceId - ID of the spawn instance to terminate
|
|
181
|
-
*/
|
|
182
|
-
async terminate(instanceId) {
|
|
183
|
-
const tracked = this.processMap.get(instanceId);
|
|
184
|
-
if (!tracked)
|
|
185
|
-
return;
|
|
186
|
-
try {
|
|
187
|
-
process.kill(tracked.pid, 'SIGTERM');
|
|
188
|
-
}
|
|
189
|
-
catch {
|
|
190
|
-
// Process may have already exited
|
|
191
|
-
}
|
|
192
|
-
this.processMap.delete(instanceId);
|
|
193
|
-
}
|
|
194
|
-
}
|
|
195
|
-
//# sourceMappingURL=spawn.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"spawn.js","sourceRoot":"","sources":["../../../src/providers/gemini-cli/spawn.ts"],"names":[],"mappings":"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"}
|