@cleocode/adapters 2026.4.10 → 2026.4.11
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/providers/claude-code/adapter.js +185 -0
- package/dist/providers/claude-code/adapter.js.map +1 -0
- package/dist/providers/claude-code/context-monitor.js +159 -0
- package/dist/providers/claude-code/context-monitor.js.map +1 -0
- package/dist/providers/claude-code/hooks.js +286 -0
- package/dist/providers/claude-code/hooks.js.map +1 -0
- package/dist/providers/claude-code/index.js +41 -0
- package/dist/providers/claude-code/index.js.map +1 -0
- package/dist/providers/claude-code/install.js +161 -0
- package/dist/providers/claude-code/install.js.map +1 -0
- package/dist/providers/claude-code/paths.js +41 -0
- package/dist/providers/claude-code/paths.js.map +1 -0
- package/dist/providers/claude-code/spawn.js +171 -0
- package/dist/providers/claude-code/spawn.js.map +1 -0
- package/dist/providers/claude-code/statusline.js +130 -0
- package/dist/providers/claude-code/statusline.js.map +1 -0
- package/dist/providers/claude-code/task-sync.js +119 -0
- package/dist/providers/claude-code/task-sync.js.map +1 -0
- package/dist/providers/claude-code/transport.js +29 -0
- package/dist/providers/claude-code/transport.js.map +1 -0
- package/dist/providers/codex/adapter.js +147 -0
- package/dist/providers/codex/adapter.js.map +1 -0
- package/dist/providers/codex/hooks.js +113 -0
- package/dist/providers/codex/hooks.js.map +1 -0
- package/dist/providers/codex/index.js +39 -0
- package/dist/providers/codex/index.js.map +1 -0
- package/dist/providers/codex/install.js +125 -0
- package/dist/providers/codex/install.js.map +1 -0
- package/dist/providers/cursor/adapter.js +152 -0
- package/dist/providers/cursor/adapter.js.map +1 -0
- package/dist/providers/cursor/hooks.js +208 -0
- package/dist/providers/cursor/hooks.js.map +1 -0
- package/dist/providers/cursor/index.js +36 -0
- package/dist/providers/cursor/index.js.map +1 -0
- package/dist/providers/cursor/install.js +181 -0
- package/dist/providers/cursor/install.js.map +1 -0
- package/dist/providers/cursor/spawn.js +59 -0
- package/dist/providers/cursor/spawn.js.map +1 -0
- package/dist/providers/gemini-cli/adapter.js +159 -0
- package/dist/providers/gemini-cli/adapter.js.map +1 -0
- package/dist/providers/gemini-cli/hooks.js +128 -0
- package/dist/providers/gemini-cli/hooks.js.map +1 -0
- package/dist/providers/gemini-cli/index.js +39 -0
- package/dist/providers/gemini-cli/index.js.map +1 -0
- package/dist/providers/gemini-cli/install.js +125 -0
- package/dist/providers/gemini-cli/install.js.map +1 -0
- package/dist/providers/kimi/adapter.js +146 -0
- package/dist/providers/kimi/adapter.js.map +1 -0
- package/dist/providers/kimi/hooks.js +79 -0
- package/dist/providers/kimi/hooks.js.map +1 -0
- package/dist/providers/kimi/index.js +39 -0
- package/dist/providers/kimi/index.js.map +1 -0
- package/dist/providers/kimi/install.js +125 -0
- package/dist/providers/kimi/install.js.map +1 -0
- package/dist/providers/opencode/adapter.js +167 -0
- package/dist/providers/opencode/adapter.js.map +1 -0
- package/dist/providers/opencode/hooks.js +206 -0
- package/dist/providers/opencode/hooks.js.map +1 -0
- package/dist/providers/opencode/index.js +37 -0
- package/dist/providers/opencode/index.js.map +1 -0
- package/dist/providers/opencode/install.js +116 -0
- package/dist/providers/opencode/install.js.map +1 -0
- package/dist/providers/opencode/spawn.js +241 -0
- package/dist/providers/opencode/spawn.js.map +1 -0
- package/dist/providers/shared/transcript-reader.js +124 -0
- package/dist/providers/shared/transcript-reader.js.map +1 -0
- package/dist/registry.js +88 -0
- package/dist/registry.js.map +1 -0
- package/package.json +3 -3
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/providers/opencode/index.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AAEH,OAAO,EAAE,eAAe,EAAE,MAAM,cAAc,CAAC;AAE/C,OAAO,EAAE,eAAe,EAAE,MAAM,cAAc,CAAC;AAC/C,OAAO,EAAE,oBAAoB,EAAE,MAAM,YAAY,CAAC;AAClD,OAAO,EAAE,uBAAuB,EAAE,MAAM,cAAc,CAAC;AACvD,OAAO,EAAE,qBAAqB,EAAE,MAAM,YAAY,CAAC;AAEnD,eAAe,eAAe,CAAC;AAE/B;;;;;;;;;;;;;;;;;;GAkBG;AACH,MAAM,UAAU,aAAa;IAC3B,OAAO,IAAI,eAAe,EAAE,CAAC;AAC/B,CAAC"}
|
|
@@ -0,0 +1,116 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* OpenCode Install Provider
|
|
3
|
+
*
|
|
4
|
+
* Handles CLEO installation into OpenCode environments:
|
|
5
|
+
* - Ensures AGENTS.md has CLEO @-references
|
|
6
|
+
*
|
|
7
|
+
* @task T5240
|
|
8
|
+
*/
|
|
9
|
+
import { existsSync, readFileSync, writeFileSync } from 'node:fs';
|
|
10
|
+
import { join } from 'node:path';
|
|
11
|
+
/** Lines that should appear in AGENTS.md to reference CLEO. */
|
|
12
|
+
const INSTRUCTION_REFERENCES = ['@~/.cleo/templates/CLEO-INJECTION.md', '@.cleo/memory-bridge.md'];
|
|
13
|
+
/**
|
|
14
|
+
* Install provider for OpenCode.
|
|
15
|
+
*
|
|
16
|
+
* Manages CLEO's integration with OpenCode by:
|
|
17
|
+
* 1. Ensuring AGENTS.md contains @-references to CLEO instruction files
|
|
18
|
+
*
|
|
19
|
+
* @remarks
|
|
20
|
+
* Installation is idempotent -- running install multiple times on the same
|
|
21
|
+
* project produces the same result. Only AGENTS.md is managed; OpenCode's
|
|
22
|
+
* plugin system is handled separately by the hook provider.
|
|
23
|
+
*/
|
|
24
|
+
export class OpenCodeInstallProvider {
|
|
25
|
+
/**
|
|
26
|
+
* Install CLEO into an OpenCode project.
|
|
27
|
+
*
|
|
28
|
+
* @param options - Installation options including project directory
|
|
29
|
+
* @returns Result describing what was installed
|
|
30
|
+
*/
|
|
31
|
+
async install(options) {
|
|
32
|
+
const { projectDir } = options;
|
|
33
|
+
const installedAt = new Date().toISOString();
|
|
34
|
+
let instructionFileUpdated = false;
|
|
35
|
+
const details = {};
|
|
36
|
+
// Step 1: Ensure AGENTS.md has @-references
|
|
37
|
+
instructionFileUpdated = this.updateInstructionFile(projectDir);
|
|
38
|
+
if (instructionFileUpdated) {
|
|
39
|
+
details.instructionFile = join(projectDir, 'AGENTS.md');
|
|
40
|
+
}
|
|
41
|
+
return {
|
|
42
|
+
success: true,
|
|
43
|
+
installedAt,
|
|
44
|
+
instructionFileUpdated,
|
|
45
|
+
mcpRegistered: false,
|
|
46
|
+
details,
|
|
47
|
+
};
|
|
48
|
+
}
|
|
49
|
+
/**
|
|
50
|
+
* Uninstall CLEO from the current OpenCode project.
|
|
51
|
+
*
|
|
52
|
+
* Does not remove AGENTS.md references (they are harmless if CLEO is not present).
|
|
53
|
+
*/
|
|
54
|
+
async uninstall() { }
|
|
55
|
+
/**
|
|
56
|
+
* Check whether CLEO is installed in the current environment.
|
|
57
|
+
*
|
|
58
|
+
* Checks for CLEO references in AGENTS.md.
|
|
59
|
+
*/
|
|
60
|
+
async isInstalled() {
|
|
61
|
+
const agentsMdPath = join(process.cwd(), 'AGENTS.md');
|
|
62
|
+
if (existsSync(agentsMdPath)) {
|
|
63
|
+
try {
|
|
64
|
+
const content = readFileSync(agentsMdPath, 'utf-8');
|
|
65
|
+
if (INSTRUCTION_REFERENCES.some((ref) => content.includes(ref))) {
|
|
66
|
+
return true;
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
catch {
|
|
70
|
+
// Fall through
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
return false;
|
|
74
|
+
}
|
|
75
|
+
/**
|
|
76
|
+
* Ensure AGENTS.md contains @-references to CLEO instruction files.
|
|
77
|
+
*
|
|
78
|
+
* Creates AGENTS.md if it does not exist. Appends any missing references.
|
|
79
|
+
*
|
|
80
|
+
* @param projectDir - Project root directory
|
|
81
|
+
*/
|
|
82
|
+
async ensureInstructionReferences(projectDir) {
|
|
83
|
+
this.updateInstructionFile(projectDir);
|
|
84
|
+
}
|
|
85
|
+
/**
|
|
86
|
+
* Update AGENTS.md with CLEO @-references.
|
|
87
|
+
*
|
|
88
|
+
* @returns true if the file was created or modified
|
|
89
|
+
*/
|
|
90
|
+
updateInstructionFile(projectDir) {
|
|
91
|
+
const agentsMdPath = join(projectDir, 'AGENTS.md');
|
|
92
|
+
let content = '';
|
|
93
|
+
let existed = false;
|
|
94
|
+
if (existsSync(agentsMdPath)) {
|
|
95
|
+
content = readFileSync(agentsMdPath, 'utf-8');
|
|
96
|
+
existed = true;
|
|
97
|
+
}
|
|
98
|
+
const missingRefs = INSTRUCTION_REFERENCES.filter((ref) => !content.includes(ref));
|
|
99
|
+
if (missingRefs.length === 0) {
|
|
100
|
+
return false;
|
|
101
|
+
}
|
|
102
|
+
const refsBlock = missingRefs.join('\n');
|
|
103
|
+
if (existed) {
|
|
104
|
+
// Append missing references
|
|
105
|
+
const separator = content.endsWith('\n') ? '' : '\n';
|
|
106
|
+
content = content + separator + refsBlock + '\n';
|
|
107
|
+
}
|
|
108
|
+
else {
|
|
109
|
+
// Create new AGENTS.md with references
|
|
110
|
+
content = refsBlock + '\n';
|
|
111
|
+
}
|
|
112
|
+
writeFileSync(agentsMdPath, content, 'utf-8');
|
|
113
|
+
return true;
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
//# sourceMappingURL=install.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"install.js","sourceRoot":"","sources":["../../../src/providers/opencode/install.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AAEH,OAAO,EAAE,UAAU,EAAE,YAAY,EAAE,aAAa,EAAE,MAAM,SAAS,CAAC;AAClE,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AAGjC,+DAA+D;AAC/D,MAAM,sBAAsB,GAAG,CAAC,sCAAsC,EAAE,yBAAyB,CAAC,CAAC;AAEnG;;;;;;;;;;GAUG;AACH,MAAM,OAAO,uBAAuB;IAClC;;;;;OAKG;IACH,KAAK,CAAC,OAAO,CAAC,OAAuB;QACnC,MAAM,EAAE,UAAU,EAAE,GAAG,OAAO,CAAC;QAC/B,MAAM,WAAW,GAAG,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC;QAC7C,IAAI,sBAAsB,GAAG,KAAK,CAAC;QACnC,MAAM,OAAO,GAA4B,EAAE,CAAC;QAE5C,4CAA4C;QAC5C,sBAAsB,GAAG,IAAI,CAAC,qBAAqB,CAAC,UAAU,CAAC,CAAC;QAChE,IAAI,sBAAsB,EAAE,CAAC;YAC3B,OAAO,CAAC,eAAe,GAAG,IAAI,CAAC,UAAU,EAAE,WAAW,CAAC,CAAC;QAC1D,CAAC;QAED,OAAO;YACL,OAAO,EAAE,IAAI;YACb,WAAW;YACX,sBAAsB;YACtB,aAAa,EAAE,KAAK;YACpB,OAAO;SACR,CAAC;IACJ,CAAC;IAED;;;;OAIG;IACH,KAAK,CAAC,SAAS,KAAmB,CAAC;IAEnC;;;;OAIG;IACH,KAAK,CAAC,WAAW;QACf,MAAM,YAAY,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,WAAW,CAAC,CAAC;QACtD,IAAI,UAAU,CAAC,YAAY,CAAC,EAAE,CAAC;YAC7B,IAAI,CAAC;gBACH,MAAM,OAAO,GAAG,YAAY,CAAC,YAAY,EAAE,OAAO,CAAC,CAAC;gBACpD,IAAI,sBAAsB,CAAC,IAAI,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC;oBAChE,OAAO,IAAI,CAAC;gBACd,CAAC;YACH,CAAC;YAAC,MAAM,CAAC;gBACP,eAAe;YACjB,CAAC;QACH,CAAC;QAED,OAAO,KAAK,CAAC;IACf,CAAC;IAED;;;;;;OAMG;IACH,KAAK,CAAC,2BAA2B,CAAC,UAAkB;QAClD,IAAI,CAAC,qBAAqB,CAAC,UAAU,CAAC,CAAC;IACzC,CAAC;IAED;;;;OAIG;IACK,qBAAqB,CAAC,UAAkB;QAC9C,MAAM,YAAY,GAAG,IAAI,CAAC,UAAU,EAAE,WAAW,CAAC,CAAC;QACnD,IAAI,OAAO,GAAG,EAAE,CAAC;QACjB,IAAI,OAAO,GAAG,KAAK,CAAC;QAEpB,IAAI,UAAU,CAAC,YAAY,CAAC,EAAE,CAAC;YAC7B,OAAO,GAAG,YAAY,CAAC,YAAY,EAAE,OAAO,CAAC,CAAC;YAC9C,OAAO,GAAG,IAAI,CAAC;QACjB,CAAC;QAED,MAAM,WAAW,GAAG,sBAAsB,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC;QAEnF,IAAI,WAAW,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YAC7B,OAAO,KAAK,CAAC;QACf,CAAC;QAED,MAAM,SAAS,GAAG,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAEzC,IAAI,OAAO,EAAE,CAAC;YACZ,4BAA4B;YAC5B,MAAM,SAAS,GAAG,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC;YACrD,OAAO,GAAG,OAAO,GAAG,SAAS,GAAG,SAAS,GAAG,IAAI,CAAC;QACnD,CAAC;aAAM,CAAC;YACN,uCAAuC;YACvC,OAAO,GAAG,SAAS,GAAG,IAAI,CAAC;QAC7B,CAAC;QAED,aAAa,CAAC,YAAY,EAAE,OAAO,EAAE,OAAO,CAAC,CAAC;QAC9C,OAAO,IAAI,CAAC;IACd,CAAC;CACF"}
|
|
@@ -0,0 +1,241 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* OpenCode Spawn Provider
|
|
3
|
+
*
|
|
4
|
+
* Implements AdapterSpawnProvider for OpenCode CLI.
|
|
5
|
+
* Migrated from src/core/spawn/adapters/opencode-adapter.ts
|
|
6
|
+
*
|
|
7
|
+
* Uses `opencode run --agent ... --format json` to spawn subagent
|
|
8
|
+
* processes. Processes run detached and are tracked by PID for
|
|
9
|
+
* listing and termination.
|
|
10
|
+
*
|
|
11
|
+
* @task T5240
|
|
12
|
+
*/
|
|
13
|
+
import { exec, spawn as nodeSpawn } from 'node:child_process';
|
|
14
|
+
import { mkdir, readFile, writeFile } from 'node:fs/promises';
|
|
15
|
+
import { join } from 'node:path';
|
|
16
|
+
import { promisify } from 'node:util';
|
|
17
|
+
const execAsync = promisify(exec);
|
|
18
|
+
/** Name used for the CLEO subagent definition in OpenCode's agent directory. */
|
|
19
|
+
const OPENCODE_SUBAGENT_NAME = 'cleo-subagent';
|
|
20
|
+
/** Fallback agent name when custom agent definition cannot be created. */
|
|
21
|
+
const OPENCODE_FALLBACK_AGENT = 'general';
|
|
22
|
+
/**
|
|
23
|
+
* Build the markdown content for an OpenCode agent definition file.
|
|
24
|
+
*
|
|
25
|
+
* OpenCode agents are defined as markdown files with YAML frontmatter
|
|
26
|
+
* in the .opencode/agent/ directory.
|
|
27
|
+
*
|
|
28
|
+
* @remarks
|
|
29
|
+
* The generated markdown uses YAML frontmatter with `mode: subagent`
|
|
30
|
+
* and `hidden: true` so the agent does not appear in OpenCode's
|
|
31
|
+
* interactive agent selection menu.
|
|
32
|
+
*
|
|
33
|
+
* @param description - Agent description for frontmatter
|
|
34
|
+
* @param instructions - Markdown instructions body
|
|
35
|
+
* @returns Complete agent definition markdown with YAML frontmatter
|
|
36
|
+
*
|
|
37
|
+
* @example
|
|
38
|
+
* ```typescript
|
|
39
|
+
* import { buildOpenCodeAgentMarkdown } from '@cleocode/adapters/providers/opencode/spawn';
|
|
40
|
+
*
|
|
41
|
+
* const md = buildOpenCodeAgentMarkdown(
|
|
42
|
+
* 'CLEO task executor',
|
|
43
|
+
* '# Subagent\n\nExecute the delegated task.',
|
|
44
|
+
* );
|
|
45
|
+
* ```
|
|
46
|
+
*/
|
|
47
|
+
export function buildOpenCodeAgentMarkdown(description, instructions) {
|
|
48
|
+
const normalizedDesc = description.replace(/\s+/g, ' ').trim();
|
|
49
|
+
return [
|
|
50
|
+
'---',
|
|
51
|
+
`description: ${JSON.stringify(normalizedDesc)}`,
|
|
52
|
+
'mode: subagent',
|
|
53
|
+
'hidden: true',
|
|
54
|
+
'---',
|
|
55
|
+
'',
|
|
56
|
+
instructions.trim(),
|
|
57
|
+
'',
|
|
58
|
+
].join('\n');
|
|
59
|
+
}
|
|
60
|
+
/**
|
|
61
|
+
* Ensure the CLEO subagent definition exists in the project's
|
|
62
|
+
* .opencode/agent/ directory.
|
|
63
|
+
*
|
|
64
|
+
* Creates or updates the agent definition file if the content has changed.
|
|
65
|
+
*
|
|
66
|
+
* @param workingDirectory - Project root directory
|
|
67
|
+
* @returns The agent name to use for spawning
|
|
68
|
+
*/
|
|
69
|
+
async function ensureSubagentDefinition(workingDirectory) {
|
|
70
|
+
const agentDir = join(workingDirectory, '.opencode', 'agent');
|
|
71
|
+
const agentPath = join(agentDir, `${OPENCODE_SUBAGENT_NAME}.md`);
|
|
72
|
+
const description = 'CLEO task executor with protocol compliance.';
|
|
73
|
+
const instructions = [
|
|
74
|
+
'# CLEO Subagent',
|
|
75
|
+
'',
|
|
76
|
+
'You are a CLEO subagent executing a delegated task.',
|
|
77
|
+
'Follow the CLEO protocol and complete the assigned work.',
|
|
78
|
+
'',
|
|
79
|
+
'@~/.cleo/templates/CLEO-INJECTION.md',
|
|
80
|
+
].join('\n');
|
|
81
|
+
const content = buildOpenCodeAgentMarkdown(description, instructions);
|
|
82
|
+
await mkdir(agentDir, { recursive: true });
|
|
83
|
+
let existing = null;
|
|
84
|
+
try {
|
|
85
|
+
existing = await readFile(agentPath, 'utf-8');
|
|
86
|
+
}
|
|
87
|
+
catch {
|
|
88
|
+
existing = null;
|
|
89
|
+
}
|
|
90
|
+
if (existing !== content) {
|
|
91
|
+
await writeFile(agentPath, content, 'utf-8');
|
|
92
|
+
}
|
|
93
|
+
return OPENCODE_SUBAGENT_NAME;
|
|
94
|
+
}
|
|
95
|
+
/**
|
|
96
|
+
* Spawn provider for OpenCode.
|
|
97
|
+
*
|
|
98
|
+
* Spawns detached OpenCode CLI processes for subagent execution.
|
|
99
|
+
* Each spawn ensures a CLEO subagent definition exists, then runs
|
|
100
|
+
* `opencode run --format json --agent <name> --title <title> <prompt>`
|
|
101
|
+
* as a detached, unref'd child process.
|
|
102
|
+
*
|
|
103
|
+
* @remarks
|
|
104
|
+
* Before spawning, the provider ensures a `cleo-subagent` agent definition
|
|
105
|
+
* exists in `.opencode/agent/`. If the definition cannot be created, it
|
|
106
|
+
* falls back to the built-in `general` agent. Processes are tracked by
|
|
107
|
+
* instance ID in an in-memory map and verified via `kill(pid, 0)` liveness
|
|
108
|
+
* checks.
|
|
109
|
+
*/
|
|
110
|
+
export class OpenCodeSpawnProvider {
|
|
111
|
+
/** Map of instance IDs to tracked process info. */
|
|
112
|
+
processMap = new Map();
|
|
113
|
+
/**
|
|
114
|
+
* Check if the OpenCode CLI is available in PATH.
|
|
115
|
+
*
|
|
116
|
+
* @returns true if `opencode` is found via `which`
|
|
117
|
+
*/
|
|
118
|
+
async canSpawn() {
|
|
119
|
+
try {
|
|
120
|
+
await execAsync('which opencode');
|
|
121
|
+
return true;
|
|
122
|
+
}
|
|
123
|
+
catch {
|
|
124
|
+
return false;
|
|
125
|
+
}
|
|
126
|
+
}
|
|
127
|
+
/**
|
|
128
|
+
* Spawn a subagent via OpenCode CLI.
|
|
129
|
+
*
|
|
130
|
+
* Ensures the CLEO subagent definition exists in the project's
|
|
131
|
+
* .opencode/agent/ directory, then spawns a detached OpenCode
|
|
132
|
+
* process. The process runs independently of the parent.
|
|
133
|
+
*
|
|
134
|
+
* @param context - Spawn context with taskId, prompt, and options
|
|
135
|
+
* @returns Spawn result with instance ID and status
|
|
136
|
+
*/
|
|
137
|
+
async spawn(context) {
|
|
138
|
+
const instanceId = `opencode-${Date.now()}-${Math.random().toString(36).substring(2, 9)}`;
|
|
139
|
+
const startTime = new Date().toISOString();
|
|
140
|
+
const workingDirectory = context.workingDirectory ?? process.cwd();
|
|
141
|
+
try {
|
|
142
|
+
let agentName;
|
|
143
|
+
try {
|
|
144
|
+
agentName = await ensureSubagentDefinition(workingDirectory);
|
|
145
|
+
}
|
|
146
|
+
catch {
|
|
147
|
+
agentName = OPENCODE_FALLBACK_AGENT;
|
|
148
|
+
}
|
|
149
|
+
const child = nodeSpawn('opencode', [
|
|
150
|
+
'run',
|
|
151
|
+
'--format',
|
|
152
|
+
'json',
|
|
153
|
+
'--agent',
|
|
154
|
+
agentName,
|
|
155
|
+
'--title',
|
|
156
|
+
`CLEO ${context.taskId}`,
|
|
157
|
+
context.prompt,
|
|
158
|
+
], {
|
|
159
|
+
cwd: workingDirectory,
|
|
160
|
+
detached: true,
|
|
161
|
+
stdio: 'ignore',
|
|
162
|
+
});
|
|
163
|
+
child.unref();
|
|
164
|
+
if (child.pid) {
|
|
165
|
+
this.processMap.set(instanceId, {
|
|
166
|
+
pid: child.pid,
|
|
167
|
+
taskId: context.taskId,
|
|
168
|
+
startTime,
|
|
169
|
+
});
|
|
170
|
+
}
|
|
171
|
+
child.on('exit', () => {
|
|
172
|
+
this.processMap.delete(instanceId);
|
|
173
|
+
});
|
|
174
|
+
return {
|
|
175
|
+
instanceId,
|
|
176
|
+
taskId: context.taskId,
|
|
177
|
+
providerId: 'opencode',
|
|
178
|
+
status: 'running',
|
|
179
|
+
startTime,
|
|
180
|
+
};
|
|
181
|
+
}
|
|
182
|
+
catch {
|
|
183
|
+
return {
|
|
184
|
+
instanceId,
|
|
185
|
+
taskId: context.taskId,
|
|
186
|
+
providerId: 'opencode',
|
|
187
|
+
status: 'failed',
|
|
188
|
+
startTime,
|
|
189
|
+
endTime: new Date().toISOString(),
|
|
190
|
+
};
|
|
191
|
+
}
|
|
192
|
+
}
|
|
193
|
+
/**
|
|
194
|
+
* List currently running OpenCode subagent processes.
|
|
195
|
+
*
|
|
196
|
+
* Checks each tracked process via kill(pid, 0) to verify it is still alive.
|
|
197
|
+
* Dead processes are automatically cleaned from the tracking map.
|
|
198
|
+
*
|
|
199
|
+
* @returns Array of spawn results for running processes
|
|
200
|
+
*/
|
|
201
|
+
async listRunning() {
|
|
202
|
+
const running = [];
|
|
203
|
+
for (const [instanceId, tracked] of this.processMap.entries()) {
|
|
204
|
+
try {
|
|
205
|
+
process.kill(tracked.pid, 0);
|
|
206
|
+
running.push({
|
|
207
|
+
instanceId,
|
|
208
|
+
taskId: tracked.taskId,
|
|
209
|
+
providerId: 'opencode',
|
|
210
|
+
status: 'running',
|
|
211
|
+
startTime: tracked.startTime,
|
|
212
|
+
});
|
|
213
|
+
}
|
|
214
|
+
catch {
|
|
215
|
+
this.processMap.delete(instanceId);
|
|
216
|
+
}
|
|
217
|
+
}
|
|
218
|
+
return running;
|
|
219
|
+
}
|
|
220
|
+
/**
|
|
221
|
+
* Terminate a running spawn by instance ID.
|
|
222
|
+
*
|
|
223
|
+
* Sends SIGTERM to the tracked process. If the process is not found
|
|
224
|
+
* or has already exited, this is a no-op.
|
|
225
|
+
*
|
|
226
|
+
* @param instanceId - ID of the spawn instance to terminate
|
|
227
|
+
*/
|
|
228
|
+
async terminate(instanceId) {
|
|
229
|
+
const tracked = this.processMap.get(instanceId);
|
|
230
|
+
if (!tracked)
|
|
231
|
+
return;
|
|
232
|
+
try {
|
|
233
|
+
process.kill(tracked.pid, 'SIGTERM');
|
|
234
|
+
}
|
|
235
|
+
catch {
|
|
236
|
+
// Process may have already exited
|
|
237
|
+
}
|
|
238
|
+
this.processMap.delete(instanceId);
|
|
239
|
+
}
|
|
240
|
+
}
|
|
241
|
+
//# sourceMappingURL=spawn.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"spawn.js","sourceRoot":"","sources":["../../../src/providers/opencode/spawn.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,124 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Shared transcript-reading utility for provider hook adapters.
|
|
3
|
+
*
|
|
4
|
+
* Several providers (Gemini CLI, Codex CLI) store session data in a
|
|
5
|
+
* flat directory of JSON/JSONL files using the same role/content schema.
|
|
6
|
+
* This module centralises the "find most-recent file, parse turns"
|
|
7
|
+
* logic to avoid duplicating it in each hook provider.
|
|
8
|
+
*
|
|
9
|
+
* Usage:
|
|
10
|
+
* ```ts
|
|
11
|
+
* import { readLatestTranscript } from '../shared/transcript-reader.js';
|
|
12
|
+
*
|
|
13
|
+
* async getTranscript(_sessionId: string, _projectDir: string) {
|
|
14
|
+
* return readLatestTranscript(join(homedir(), '.gemini'));
|
|
15
|
+
* }
|
|
16
|
+
* ```
|
|
17
|
+
*
|
|
18
|
+
* @task T161
|
|
19
|
+
* @epic T134
|
|
20
|
+
*/
|
|
21
|
+
import { readdir, readFile } from 'node:fs/promises';
|
|
22
|
+
import { join } from 'node:path';
|
|
23
|
+
// ---------------------------------------------------------------------------
|
|
24
|
+
// Helpers
|
|
25
|
+
// ---------------------------------------------------------------------------
|
|
26
|
+
/**
|
|
27
|
+
* Parse a raw JSONL or JSON session file into an array of transcript turns.
|
|
28
|
+
*
|
|
29
|
+
* Lines that are not valid JSON, or that lack a string `role` and string
|
|
30
|
+
* `content`, are silently skipped.
|
|
31
|
+
*
|
|
32
|
+
* @param raw - Raw file contents (UTF-8 string).
|
|
33
|
+
* @returns Array of `{ role, content }` pairs, in file order.
|
|
34
|
+
*/
|
|
35
|
+
function parseTranscriptLines(raw) {
|
|
36
|
+
const turns = [];
|
|
37
|
+
const lines = raw.split('\n').filter((l) => l.trim());
|
|
38
|
+
for (const line of lines) {
|
|
39
|
+
try {
|
|
40
|
+
const entry = JSON.parse(line);
|
|
41
|
+
const role = entry.role;
|
|
42
|
+
const content = entry.content;
|
|
43
|
+
if (typeof role === 'string' && typeof content === 'string') {
|
|
44
|
+
turns.push({ role, content });
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
catch {
|
|
48
|
+
// Skip malformed lines
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
return turns;
|
|
52
|
+
}
|
|
53
|
+
// ---------------------------------------------------------------------------
|
|
54
|
+
// Public API
|
|
55
|
+
// ---------------------------------------------------------------------------
|
|
56
|
+
/**
|
|
57
|
+
* Read the most recent JSON or JSONL session file from `providerDir` and
|
|
58
|
+
* return its contents as a flat transcript string.
|
|
59
|
+
*
|
|
60
|
+
* Files are sorted in descending order by filename — this works naturally
|
|
61
|
+
* for providers that embed timestamps in filenames. The most recently named
|
|
62
|
+
* file is read first.
|
|
63
|
+
*
|
|
64
|
+
* Returns `null` when:
|
|
65
|
+
* - `providerDir` does not exist or cannot be read
|
|
66
|
+
* - No JSON/JSONL files are present
|
|
67
|
+
* - The most recent file contains no parseable turns
|
|
68
|
+
*
|
|
69
|
+
* @remarks
|
|
70
|
+
* This utility is shared by Gemini CLI and Codex CLI hook providers which
|
|
71
|
+
* both store session data in the same flat JSON/JSONL format. Only the
|
|
72
|
+
* most recent file is read to keep memory usage bounded.
|
|
73
|
+
*
|
|
74
|
+
* @param providerDir - Absolute path to the provider's session directory
|
|
75
|
+
* (e.g. `~/.gemini` or `~/.codex`).
|
|
76
|
+
* @returns A plain-text transcript with lines of the form `role: content`,
|
|
77
|
+
* or `null` if no transcript could be extracted.
|
|
78
|
+
*
|
|
79
|
+
* @example
|
|
80
|
+
* ```typescript
|
|
81
|
+
* import { readLatestTranscript } from '../shared/transcript-reader.js';
|
|
82
|
+
*
|
|
83
|
+
* const transcript = await readLatestTranscript('/home/user/.gemini');
|
|
84
|
+
* if (transcript) {
|
|
85
|
+
* console.log(transcript);
|
|
86
|
+
* }
|
|
87
|
+
* ```
|
|
88
|
+
*
|
|
89
|
+
* @task T161
|
|
90
|
+
* @epic T134
|
|
91
|
+
*/
|
|
92
|
+
export async function readLatestTranscript(providerDir) {
|
|
93
|
+
let allFiles = [];
|
|
94
|
+
try {
|
|
95
|
+
const entries = await readdir(providerDir, { withFileTypes: true });
|
|
96
|
+
for (const entry of entries) {
|
|
97
|
+
if (!entry.isFile())
|
|
98
|
+
continue;
|
|
99
|
+
const name = entry.name;
|
|
100
|
+
if (name.endsWith('.json') || name.endsWith('.jsonl')) {
|
|
101
|
+
allFiles.push(join(providerDir, name));
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
catch {
|
|
106
|
+
return null;
|
|
107
|
+
}
|
|
108
|
+
if (allFiles.length === 0)
|
|
109
|
+
return null;
|
|
110
|
+
// Sort descending — timestamps in filenames sort naturally
|
|
111
|
+
allFiles = allFiles.sort((a, b) => b.localeCompare(a));
|
|
112
|
+
const mostRecent = allFiles[0];
|
|
113
|
+
if (!mostRecent)
|
|
114
|
+
return null;
|
|
115
|
+
try {
|
|
116
|
+
const raw = await readFile(mostRecent, 'utf-8');
|
|
117
|
+
const turns = parseTranscriptLines(raw);
|
|
118
|
+
return turns.length > 0 ? turns.map((t) => `${t.role}: ${t.content}`).join('\n') : null;
|
|
119
|
+
}
|
|
120
|
+
catch {
|
|
121
|
+
return null;
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
//# sourceMappingURL=transcript-reader.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"transcript-reader.js","sourceRoot":"","sources":["../../../src/providers/shared/transcript-reader.ts"],"names":[],"mappings":"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"}
|
package/dist/registry.js
ADDED
|
@@ -0,0 +1,88 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Adapter registry — discovers and provides access to provider manifests.
|
|
3
|
+
*
|
|
4
|
+
* Scans the providers/ directory for manifest.json files and returns
|
|
5
|
+
* the discovered adapter manifests for use by AdapterManager.
|
|
6
|
+
*
|
|
7
|
+
* @task T5240
|
|
8
|
+
*/
|
|
9
|
+
import { readFileSync } from 'node:fs';
|
|
10
|
+
import { dirname, join, resolve } from 'node:path';
|
|
11
|
+
import { fileURLToPath } from 'node:url';
|
|
12
|
+
/** Known provider IDs bundled with @cleocode/adapters. */
|
|
13
|
+
const PROVIDER_IDS = ['claude-code', 'opencode', 'cursor'];
|
|
14
|
+
/**
|
|
15
|
+
* Get the manifests for all bundled provider adapters.
|
|
16
|
+
*
|
|
17
|
+
* @remarks
|
|
18
|
+
* Scans the known provider directories for `manifest.json` files.
|
|
19
|
+
* Providers whose manifests cannot be loaded (missing or malformed)
|
|
20
|
+
* are silently skipped.
|
|
21
|
+
*
|
|
22
|
+
* @returns Array of adapter manifests for successfully loaded providers
|
|
23
|
+
*
|
|
24
|
+
* @example
|
|
25
|
+
* ```typescript
|
|
26
|
+
* import { getProviderManifests } from '@cleocode/adapters';
|
|
27
|
+
*
|
|
28
|
+
* const manifests = getProviderManifests();
|
|
29
|
+
* for (const m of manifests) {
|
|
30
|
+
* console.log(`${m.id}: ${m.name} v${m.version}`);
|
|
31
|
+
* }
|
|
32
|
+
* ```
|
|
33
|
+
*/
|
|
34
|
+
export function getProviderManifests() {
|
|
35
|
+
const manifests = [];
|
|
36
|
+
const baseDir = resolve(dirname(fileURLToPath(import.meta.url)), 'providers');
|
|
37
|
+
for (const providerId of PROVIDER_IDS) {
|
|
38
|
+
try {
|
|
39
|
+
const manifestPath = join(baseDir, providerId, 'manifest.json');
|
|
40
|
+
const raw = readFileSync(manifestPath, 'utf-8');
|
|
41
|
+
manifests.push(JSON.parse(raw));
|
|
42
|
+
}
|
|
43
|
+
catch {
|
|
44
|
+
// Skip providers whose manifests cannot be loaded
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
return manifests;
|
|
48
|
+
}
|
|
49
|
+
/**
|
|
50
|
+
* Discover all available provider adapters.
|
|
51
|
+
*
|
|
52
|
+
* Returns a map of provider ID to adapter factory function.
|
|
53
|
+
*
|
|
54
|
+
* @remarks
|
|
55
|
+
* Each factory lazily imports the provider module and constructs a new
|
|
56
|
+
* adapter instance. This avoids loading all provider code upfront and
|
|
57
|
+
* keeps startup fast.
|
|
58
|
+
*
|
|
59
|
+
* @returns Map of provider ID to async factory function that creates an adapter instance
|
|
60
|
+
*
|
|
61
|
+
* @example
|
|
62
|
+
* ```typescript
|
|
63
|
+
* import { discoverProviders } from '@cleocode/adapters';
|
|
64
|
+
*
|
|
65
|
+
* const providers = await discoverProviders();
|
|
66
|
+
* const factory = providers.get('claude-code');
|
|
67
|
+
* if (factory) {
|
|
68
|
+
* const adapter = await factory();
|
|
69
|
+
* }
|
|
70
|
+
* ```
|
|
71
|
+
*/
|
|
72
|
+
export async function discoverProviders() {
|
|
73
|
+
const providers = new Map();
|
|
74
|
+
providers.set('claude-code', async () => {
|
|
75
|
+
const { ClaudeCodeAdapter } = await import('./providers/claude-code/index.js');
|
|
76
|
+
return new ClaudeCodeAdapter();
|
|
77
|
+
});
|
|
78
|
+
providers.set('opencode', async () => {
|
|
79
|
+
const { OpenCodeAdapter } = await import('./providers/opencode/index.js');
|
|
80
|
+
return new OpenCodeAdapter();
|
|
81
|
+
});
|
|
82
|
+
providers.set('cursor', async () => {
|
|
83
|
+
const { CursorAdapter } = await import('./providers/cursor/index.js');
|
|
84
|
+
return new CursorAdapter();
|
|
85
|
+
});
|
|
86
|
+
return providers;
|
|
87
|
+
}
|
|
88
|
+
//# sourceMappingURL=registry.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"registry.js","sourceRoot":"","sources":["../src/registry.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AAEH,OAAO,EAAE,YAAY,EAAE,MAAM,SAAS,CAAC;AACvC,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACnD,OAAO,EAAE,aAAa,EAAE,MAAM,UAAU,CAAC;AAqCzC,0DAA0D;AAC1D,MAAM,YAAY,GAAG,CAAC,aAAa,EAAE,UAAU,EAAE,QAAQ,CAAU,CAAC;AAEpE;;;;;;;;;;;;;;;;;;;GAmBG;AACH,MAAM,UAAU,oBAAoB;IAClC,MAAM,SAAS,GAAsB,EAAE,CAAC;IACxC,MAAM,OAAO,GAAG,OAAO,CAAC,OAAO,CAAC,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,WAAW,CAAC,CAAC;IAE9E,KAAK,MAAM,UAAU,IAAI,YAAY,EAAE,CAAC;QACtC,IAAI,CAAC;YACH,MAAM,YAAY,GAAG,IAAI,CAAC,OAAO,EAAE,UAAU,EAAE,eAAe,CAAC,CAAC;YAChE,MAAM,GAAG,GAAG,YAAY,CAAC,YAAY,EAAE,OAAO,CAAC,CAAC;YAChD,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAoB,CAAC,CAAC;QACrD,CAAC;QAAC,MAAM,CAAC;YACP,kDAAkD;QACpD,CAAC;IACH,CAAC;IAED,OAAO,SAAS,CAAC;AACnB,CAAC;AAED;;;;;;;;;;;;;;;;;;;;;;GAsBG;AACH,MAAM,CAAC,KAAK,UAAU,iBAAiB;IACrC,MAAM,SAAS,GAAG,IAAI,GAAG,EAAkC,CAAC;IAE5D,SAAS,CAAC,GAAG,CAAC,aAAa,EAAE,KAAK,IAAI,EAAE;QACtC,MAAM,EAAE,iBAAiB,EAAE,GAAG,MAAM,MAAM,CAAC,kCAAkC,CAAC,CAAC;QAC/E,OAAO,IAAI,iBAAiB,EAAE,CAAC;IACjC,CAAC,CAAC,CAAC;IAEH,SAAS,CAAC,GAAG,CAAC,UAAU,EAAE,KAAK,IAAI,EAAE;QACnC,MAAM,EAAE,eAAe,EAAE,GAAG,MAAM,MAAM,CAAC,+BAA+B,CAAC,CAAC;QAC1E,OAAO,IAAI,eAAe,EAAE,CAAC;IAC/B,CAAC,CAAC,CAAC;IAEH,SAAS,CAAC,GAAG,CAAC,QAAQ,EAAE,KAAK,IAAI,EAAE;QACjC,MAAM,EAAE,aAAa,EAAE,GAAG,MAAM,MAAM,CAAC,6BAA6B,CAAC,CAAC;QACtE,OAAO,IAAI,aAAa,EAAE,CAAC;IAC7B,CAAC,CAAC,CAAC;IAEH,OAAO,SAAS,CAAC;AACnB,CAAC"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@cleocode/adapters",
|
|
3
|
-
"version": "2026.4.
|
|
3
|
+
"version": "2026.4.11",
|
|
4
4
|
"description": "Unified provider adapters for CLEO (Claude Code, OpenCode, Cursor)",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"main": "dist/index.js",
|
|
@@ -12,8 +12,8 @@
|
|
|
12
12
|
}
|
|
13
13
|
},
|
|
14
14
|
"dependencies": {
|
|
15
|
-
"@cleocode/caamp": "2026.4.
|
|
16
|
-
"@cleocode/contracts": "2026.4.
|
|
15
|
+
"@cleocode/caamp": "2026.4.11",
|
|
16
|
+
"@cleocode/contracts": "2026.4.11"
|
|
17
17
|
},
|
|
18
18
|
"license": "MIT",
|
|
19
19
|
"engines": {
|