@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.
Files changed (69) hide show
  1. package/dist/providers/claude-code/adapter.js +185 -0
  2. package/dist/providers/claude-code/adapter.js.map +1 -0
  3. package/dist/providers/claude-code/context-monitor.js +159 -0
  4. package/dist/providers/claude-code/context-monitor.js.map +1 -0
  5. package/dist/providers/claude-code/hooks.js +286 -0
  6. package/dist/providers/claude-code/hooks.js.map +1 -0
  7. package/dist/providers/claude-code/index.js +41 -0
  8. package/dist/providers/claude-code/index.js.map +1 -0
  9. package/dist/providers/claude-code/install.js +161 -0
  10. package/dist/providers/claude-code/install.js.map +1 -0
  11. package/dist/providers/claude-code/paths.js +41 -0
  12. package/dist/providers/claude-code/paths.js.map +1 -0
  13. package/dist/providers/claude-code/spawn.js +171 -0
  14. package/dist/providers/claude-code/spawn.js.map +1 -0
  15. package/dist/providers/claude-code/statusline.js +130 -0
  16. package/dist/providers/claude-code/statusline.js.map +1 -0
  17. package/dist/providers/claude-code/task-sync.js +119 -0
  18. package/dist/providers/claude-code/task-sync.js.map +1 -0
  19. package/dist/providers/claude-code/transport.js +29 -0
  20. package/dist/providers/claude-code/transport.js.map +1 -0
  21. package/dist/providers/codex/adapter.js +147 -0
  22. package/dist/providers/codex/adapter.js.map +1 -0
  23. package/dist/providers/codex/hooks.js +113 -0
  24. package/dist/providers/codex/hooks.js.map +1 -0
  25. package/dist/providers/codex/index.js +39 -0
  26. package/dist/providers/codex/index.js.map +1 -0
  27. package/dist/providers/codex/install.js +125 -0
  28. package/dist/providers/codex/install.js.map +1 -0
  29. package/dist/providers/cursor/adapter.js +152 -0
  30. package/dist/providers/cursor/adapter.js.map +1 -0
  31. package/dist/providers/cursor/hooks.js +208 -0
  32. package/dist/providers/cursor/hooks.js.map +1 -0
  33. package/dist/providers/cursor/index.js +36 -0
  34. package/dist/providers/cursor/index.js.map +1 -0
  35. package/dist/providers/cursor/install.js +181 -0
  36. package/dist/providers/cursor/install.js.map +1 -0
  37. package/dist/providers/cursor/spawn.js +59 -0
  38. package/dist/providers/cursor/spawn.js.map +1 -0
  39. package/dist/providers/gemini-cli/adapter.js +159 -0
  40. package/dist/providers/gemini-cli/adapter.js.map +1 -0
  41. package/dist/providers/gemini-cli/hooks.js +128 -0
  42. package/dist/providers/gemini-cli/hooks.js.map +1 -0
  43. package/dist/providers/gemini-cli/index.js +39 -0
  44. package/dist/providers/gemini-cli/index.js.map +1 -0
  45. package/dist/providers/gemini-cli/install.js +125 -0
  46. package/dist/providers/gemini-cli/install.js.map +1 -0
  47. package/dist/providers/kimi/adapter.js +146 -0
  48. package/dist/providers/kimi/adapter.js.map +1 -0
  49. package/dist/providers/kimi/hooks.js +79 -0
  50. package/dist/providers/kimi/hooks.js.map +1 -0
  51. package/dist/providers/kimi/index.js +39 -0
  52. package/dist/providers/kimi/index.js.map +1 -0
  53. package/dist/providers/kimi/install.js +125 -0
  54. package/dist/providers/kimi/install.js.map +1 -0
  55. package/dist/providers/opencode/adapter.js +167 -0
  56. package/dist/providers/opencode/adapter.js.map +1 -0
  57. package/dist/providers/opencode/hooks.js +206 -0
  58. package/dist/providers/opencode/hooks.js.map +1 -0
  59. package/dist/providers/opencode/index.js +37 -0
  60. package/dist/providers/opencode/index.js.map +1 -0
  61. package/dist/providers/opencode/install.js +116 -0
  62. package/dist/providers/opencode/install.js.map +1 -0
  63. package/dist/providers/opencode/spawn.js +241 -0
  64. package/dist/providers/opencode/spawn.js.map +1 -0
  65. package/dist/providers/shared/transcript-reader.js +124 -0
  66. package/dist/providers/shared/transcript-reader.js.map +1 -0
  67. package/dist/registry.js +88 -0
  68. package/dist/registry.js.map +1 -0
  69. 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":"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"}
@@ -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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"}
@@ -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.10",
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.10",
16
- "@cleocode/contracts": "2026.4.10"
15
+ "@cleocode/caamp": "2026.4.11",
16
+ "@cleocode/contracts": "2026.4.11"
17
17
  },
18
18
  "license": "MIT",
19
19
  "engines": {