@cleocode/adapters 2.0.0

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 (217) hide show
  1. package/dist/index.d.ts +27 -0
  2. package/dist/index.d.ts.map +1 -0
  3. package/dist/index.js +1927 -0
  4. package/dist/index.js.map +7 -0
  5. package/dist/providers/claude-code/adapter.d.ts +75 -0
  6. package/dist/providers/claude-code/adapter.d.ts.map +1 -0
  7. package/dist/providers/claude-code/adapter.js +154 -0
  8. package/dist/providers/claude-code/adapter.js.map +1 -0
  9. package/dist/providers/claude-code/context-monitor.d.ts +24 -0
  10. package/dist/providers/claude-code/context-monitor.d.ts.map +1 -0
  11. package/dist/providers/claude-code/context-monitor.js +148 -0
  12. package/dist/providers/claude-code/context-monitor.js.map +1 -0
  13. package/dist/providers/claude-code/hooks.d.ts +59 -0
  14. package/dist/providers/claude-code/hooks.d.ts.map +1 -0
  15. package/dist/providers/claude-code/hooks.js +77 -0
  16. package/dist/providers/claude-code/hooks.js.map +1 -0
  17. package/dist/providers/claude-code/index.d.ts +24 -0
  18. package/dist/providers/claude-code/index.d.ts.map +1 -0
  19. package/dist/providers/claude-code/index.js +26 -0
  20. package/dist/providers/claude-code/index.js.map +1 -0
  21. package/dist/providers/claude-code/install.d.ts +75 -0
  22. package/dist/providers/claude-code/install.d.ts.map +1 -0
  23. package/dist/providers/claude-code/install.js +237 -0
  24. package/dist/providers/claude-code/install.js.map +1 -0
  25. package/dist/providers/claude-code/paths.d.ts +24 -0
  26. package/dist/providers/claude-code/paths.d.ts.map +1 -0
  27. package/dist/providers/claude-code/paths.js +33 -0
  28. package/dist/providers/claude-code/paths.js.map +1 -0
  29. package/dist/providers/claude-code/spawn.d.ts +60 -0
  30. package/dist/providers/claude-code/spawn.d.ts.map +1 -0
  31. package/dist/providers/claude-code/spawn.js +160 -0
  32. package/dist/providers/claude-code/spawn.js.map +1 -0
  33. package/dist/providers/claude-code/statusline.d.ts +24 -0
  34. package/dist/providers/claude-code/statusline.d.ts.map +1 -0
  35. package/dist/providers/claude-code/statusline.js +85 -0
  36. package/dist/providers/claude-code/statusline.js.map +1 -0
  37. package/dist/providers/claude-code/task-sync.d.ts +27 -0
  38. package/dist/providers/claude-code/task-sync.d.ts.map +1 -0
  39. package/dist/providers/claude-code/task-sync.js +124 -0
  40. package/dist/providers/claude-code/task-sync.js.map +1 -0
  41. package/dist/providers/claude-code/transport.d.ts +14 -0
  42. package/dist/providers/claude-code/transport.d.ts.map +1 -0
  43. package/dist/providers/claude-code/transport.js +18 -0
  44. package/dist/providers/claude-code/transport.js.map +1 -0
  45. package/dist/providers/cursor/adapter.d.ts +62 -0
  46. package/dist/providers/cursor/adapter.d.ts.map +1 -0
  47. package/dist/providers/cursor/adapter.js +124 -0
  48. package/dist/providers/cursor/adapter.js.map +1 -0
  49. package/dist/providers/cursor/hooks.d.ts +48 -0
  50. package/dist/providers/cursor/hooks.d.ts.map +1 -0
  51. package/dist/providers/cursor/hooks.js +55 -0
  52. package/dist/providers/cursor/hooks.js.map +1 -0
  53. package/dist/providers/cursor/index.d.ts +19 -0
  54. package/dist/providers/cursor/index.d.ts.map +1 -0
  55. package/dist/providers/cursor/index.js +21 -0
  56. package/dist/providers/cursor/index.js.map +1 -0
  57. package/dist/providers/cursor/install.d.ts +94 -0
  58. package/dist/providers/cursor/install.d.ts.map +1 -0
  59. package/dist/providers/cursor/install.js +241 -0
  60. package/dist/providers/cursor/install.js.map +1 -0
  61. package/dist/providers/cursor/spawn.d.ts +50 -0
  62. package/dist/providers/cursor/spawn.d.ts.map +1 -0
  63. package/dist/providers/cursor/spawn.js +59 -0
  64. package/dist/providers/cursor/spawn.js.map +1 -0
  65. package/dist/providers/opencode/adapter.d.ts +67 -0
  66. package/dist/providers/opencode/adapter.d.ts.map +1 -0
  67. package/dist/providers/opencode/adapter.js +144 -0
  68. package/dist/providers/opencode/adapter.js.map +1 -0
  69. package/dist/providers/opencode/hooks.d.ts +66 -0
  70. package/dist/providers/opencode/hooks.d.ts.map +1 -0
  71. package/dist/providers/opencode/hooks.js +89 -0
  72. package/dist/providers/opencode/hooks.js.map +1 -0
  73. package/dist/providers/opencode/index.d.ts +20 -0
  74. package/dist/providers/opencode/index.d.ts.map +1 -0
  75. package/dist/providers/opencode/index.js +22 -0
  76. package/dist/providers/opencode/index.js.map +1 -0
  77. package/dist/providers/opencode/install.d.ts +65 -0
  78. package/dist/providers/opencode/install.d.ts.map +1 -0
  79. package/dist/providers/opencode/install.js +183 -0
  80. package/dist/providers/opencode/install.js.map +1 -0
  81. package/dist/providers/opencode/spawn.d.ts +72 -0
  82. package/dist/providers/opencode/spawn.d.ts.map +1 -0
  83. package/dist/providers/opencode/spawn.js +219 -0
  84. package/dist/providers/opencode/spawn.js.map +1 -0
  85. package/dist/registry.d.ts +36 -0
  86. package/dist/registry.d.ts.map +1 -0
  87. package/dist/registry.js +55 -0
  88. package/dist/registry.js.map +1 -0
  89. package/package.json +32 -0
  90. package/src/index.d.ts +27 -0
  91. package/src/index.d.ts.map +1 -0
  92. package/src/index.js +28 -0
  93. package/src/index.js.map +1 -0
  94. package/src/index.ts +37 -0
  95. package/src/providers/claude-code/__tests__/adapter.test.d.ts +7 -0
  96. package/src/providers/claude-code/__tests__/adapter.test.d.ts.map +1 -0
  97. package/src/providers/claude-code/__tests__/adapter.test.js +249 -0
  98. package/src/providers/claude-code/__tests__/adapter.test.js.map +1 -0
  99. package/src/providers/claude-code/__tests__/adapter.test.ts +291 -0
  100. package/src/providers/claude-code/adapter.d.ts +75 -0
  101. package/src/providers/claude-code/adapter.d.ts.map +1 -0
  102. package/src/providers/claude-code/adapter.js +154 -0
  103. package/src/providers/claude-code/adapter.js.map +1 -0
  104. package/src/providers/claude-code/adapter.ts +175 -0
  105. package/src/providers/claude-code/context-monitor.d.ts +24 -0
  106. package/src/providers/claude-code/context-monitor.d.ts.map +1 -0
  107. package/src/providers/claude-code/context-monitor.js +148 -0
  108. package/src/providers/claude-code/context-monitor.js.map +1 -0
  109. package/src/providers/claude-code/context-monitor.ts +175 -0
  110. package/src/providers/claude-code/hooks.d.ts +59 -0
  111. package/src/providers/claude-code/hooks.d.ts.map +1 -0
  112. package/src/providers/claude-code/hooks.js +77 -0
  113. package/src/providers/claude-code/hooks.js.map +1 -0
  114. package/src/providers/claude-code/hooks.ts +85 -0
  115. package/src/providers/claude-code/index.d.ts +24 -0
  116. package/src/providers/claude-code/index.d.ts.map +1 -0
  117. package/src/providers/claude-code/index.js +26 -0
  118. package/src/providers/claude-code/index.js.map +1 -0
  119. package/src/providers/claude-code/index.ts +33 -0
  120. package/src/providers/claude-code/install.d.ts +75 -0
  121. package/src/providers/claude-code/install.d.ts.map +1 -0
  122. package/src/providers/claude-code/install.js +237 -0
  123. package/src/providers/claude-code/install.js.map +1 -0
  124. package/src/providers/claude-code/install.ts +267 -0
  125. package/src/providers/claude-code/manifest.json +26 -0
  126. package/src/providers/claude-code/paths.d.ts +24 -0
  127. package/src/providers/claude-code/paths.d.ts.map +1 -0
  128. package/src/providers/claude-code/paths.js +33 -0
  129. package/src/providers/claude-code/paths.js.map +1 -0
  130. package/src/providers/claude-code/paths.ts +38 -0
  131. package/src/providers/claude-code/spawn.d.ts +60 -0
  132. package/src/providers/claude-code/spawn.d.ts.map +1 -0
  133. package/src/providers/claude-code/spawn.js +160 -0
  134. package/src/providers/claude-code/spawn.js.map +1 -0
  135. package/src/providers/claude-code/spawn.ts +178 -0
  136. package/src/providers/claude-code/statusline.d.ts +24 -0
  137. package/src/providers/claude-code/statusline.d.ts.map +1 -0
  138. package/src/providers/claude-code/statusline.js +85 -0
  139. package/src/providers/claude-code/statusline.js.map +1 -0
  140. package/src/providers/claude-code/statusline.ts +99 -0
  141. package/src/providers/claude-code/task-sync.d.ts +27 -0
  142. package/src/providers/claude-code/task-sync.d.ts.map +1 -0
  143. package/src/providers/claude-code/task-sync.js +124 -0
  144. package/src/providers/claude-code/task-sync.js.map +1 -0
  145. package/src/providers/claude-code/task-sync.ts +158 -0
  146. package/src/providers/claude-code/transport.d.ts +14 -0
  147. package/src/providers/claude-code/transport.d.ts.map +1 -0
  148. package/src/providers/claude-code/transport.js +18 -0
  149. package/src/providers/claude-code/transport.js.map +1 -0
  150. package/src/providers/claude-code/transport.ts +21 -0
  151. package/src/providers/cursor/__tests__/adapter.test.d.ts +7 -0
  152. package/src/providers/cursor/__tests__/adapter.test.d.ts.map +1 -0
  153. package/src/providers/cursor/__tests__/adapter.test.js +246 -0
  154. package/src/providers/cursor/__tests__/adapter.test.js.map +1 -0
  155. package/src/providers/cursor/__tests__/adapter.test.ts +291 -0
  156. package/src/providers/cursor/adapter.d.ts +62 -0
  157. package/src/providers/cursor/adapter.d.ts.map +1 -0
  158. package/src/providers/cursor/adapter.js +124 -0
  159. package/src/providers/cursor/adapter.js.map +1 -0
  160. package/src/providers/cursor/adapter.ts +145 -0
  161. package/src/providers/cursor/hooks.d.ts +48 -0
  162. package/src/providers/cursor/hooks.d.ts.map +1 -0
  163. package/src/providers/cursor/hooks.js +55 -0
  164. package/src/providers/cursor/hooks.js.map +1 -0
  165. package/src/providers/cursor/hooks.ts +61 -0
  166. package/src/providers/cursor/index.d.ts +19 -0
  167. package/src/providers/cursor/index.d.ts.map +1 -0
  168. package/src/providers/cursor/index.js +21 -0
  169. package/src/providers/cursor/index.js.map +1 -0
  170. package/src/providers/cursor/index.ts +24 -0
  171. package/src/providers/cursor/install.d.ts +94 -0
  172. package/src/providers/cursor/install.d.ts.map +1 -0
  173. package/src/providers/cursor/install.js +241 -0
  174. package/src/providers/cursor/install.js.map +1 -0
  175. package/src/providers/cursor/install.ts +271 -0
  176. package/src/providers/cursor/manifest.json +26 -0
  177. package/src/providers/cursor/spawn.d.ts +50 -0
  178. package/src/providers/cursor/spawn.d.ts.map +1 -0
  179. package/src/providers/cursor/spawn.js +59 -0
  180. package/src/providers/cursor/spawn.js.map +1 -0
  181. package/src/providers/cursor/spawn.ts +66 -0
  182. package/src/providers/opencode/__tests__/adapter.test.d.ts +7 -0
  183. package/src/providers/opencode/__tests__/adapter.test.d.ts.map +1 -0
  184. package/src/providers/opencode/__tests__/adapter.test.js +263 -0
  185. package/src/providers/opencode/__tests__/adapter.test.js.map +1 -0
  186. package/src/providers/opencode/__tests__/adapter.test.ts +309 -0
  187. package/src/providers/opencode/adapter.d.ts +67 -0
  188. package/src/providers/opencode/adapter.d.ts.map +1 -0
  189. package/src/providers/opencode/adapter.js +144 -0
  190. package/src/providers/opencode/adapter.js.map +1 -0
  191. package/src/providers/opencode/adapter.ts +165 -0
  192. package/src/providers/opencode/hooks.d.ts +66 -0
  193. package/src/providers/opencode/hooks.d.ts.map +1 -0
  194. package/src/providers/opencode/hooks.js +89 -0
  195. package/src/providers/opencode/hooks.js.map +1 -0
  196. package/src/providers/opencode/hooks.ts +97 -0
  197. package/src/providers/opencode/index.d.ts +20 -0
  198. package/src/providers/opencode/index.d.ts.map +1 -0
  199. package/src/providers/opencode/index.js +22 -0
  200. package/src/providers/opencode/index.js.map +1 -0
  201. package/src/providers/opencode/index.ts +25 -0
  202. package/src/providers/opencode/install.d.ts +65 -0
  203. package/src/providers/opencode/install.d.ts.map +1 -0
  204. package/src/providers/opencode/install.js +183 -0
  205. package/src/providers/opencode/install.js.map +1 -0
  206. package/src/providers/opencode/install.ts +206 -0
  207. package/src/providers/opencode/manifest.json +26 -0
  208. package/src/providers/opencode/spawn.d.ts +72 -0
  209. package/src/providers/opencode/spawn.d.ts.map +1 -0
  210. package/src/providers/opencode/spawn.js +219 -0
  211. package/src/providers/opencode/spawn.js.map +1 -0
  212. package/src/providers/opencode/spawn.ts +253 -0
  213. package/src/registry.d.ts +36 -0
  214. package/src/registry.d.ts.map +1 -0
  215. package/src/registry.js +55 -0
  216. package/src/registry.js.map +1 -0
  217. package/src/registry.ts +81 -0
@@ -0,0 +1,144 @@
1
+ /**
2
+ * OpenCode Adapter
3
+ *
4
+ * Main CLEOProviderAdapter implementation for OpenCode AI coding assistant.
5
+ * Provides spawn, hooks, and install capabilities for CLEO integration.
6
+ *
7
+ * @task T5240
8
+ */
9
+ import { exec } from 'node:child_process';
10
+ import { existsSync } from 'node:fs';
11
+ import { join } from 'node:path';
12
+ import { promisify } from 'node:util';
13
+ import { OpenCodeHookProvider } from './hooks.js';
14
+ import { OpenCodeInstallProvider } from './install.js';
15
+ import { OpenCodeSpawnProvider } from './spawn.js';
16
+ const execAsync = promisify(exec);
17
+ /**
18
+ * CLEO provider adapter for OpenCode AI coding assistant.
19
+ *
20
+ * Bridges CLEO's adapter system with OpenCode's native capabilities:
21
+ * - Hooks: Maps OpenCode events (session.start, tool.complete, etc.) to CAAMP events
22
+ * - Spawn: Launches subagent processes via the `opencode` CLI
23
+ * - Install: Registers MCP server in .opencode/config.json and ensures AGENTS.md references
24
+ */
25
+ export class OpenCodeAdapter {
26
+ id = 'opencode';
27
+ name = 'OpenCode';
28
+ version = '1.0.0';
29
+ capabilities = {
30
+ supportsHooks: true,
31
+ supportedHookEvents: [
32
+ 'onSessionStart',
33
+ 'onSessionEnd',
34
+ 'onToolStart',
35
+ 'onToolComplete',
36
+ 'onError',
37
+ 'onPromptSubmit',
38
+ ],
39
+ supportsSpawn: true,
40
+ supportsInstall: true,
41
+ supportsMcp: true,
42
+ supportsInstructionFiles: true,
43
+ instructionFilePattern: 'AGENTS.md',
44
+ supportsContextMonitor: false,
45
+ supportsStatusline: false,
46
+ supportsProviderPaths: true,
47
+ supportsTransport: false,
48
+ supportsTaskSync: false,
49
+ };
50
+ hooks;
51
+ spawn;
52
+ install;
53
+ projectDir = null;
54
+ initialized = false;
55
+ constructor() {
56
+ this.hooks = new OpenCodeHookProvider();
57
+ this.spawn = new OpenCodeSpawnProvider();
58
+ this.install = new OpenCodeInstallProvider();
59
+ }
60
+ /**
61
+ * Initialize the adapter for a given project directory.
62
+ *
63
+ * Validates the environment by checking for the OpenCode CLI
64
+ * and OpenCode configuration directory.
65
+ *
66
+ * @param projectDir - Root directory of the project
67
+ */
68
+ async initialize(projectDir) {
69
+ this.projectDir = projectDir;
70
+ this.initialized = true;
71
+ }
72
+ /**
73
+ * Dispose the adapter and clean up resources.
74
+ *
75
+ * Unregisters hooks and releases any tracked state.
76
+ */
77
+ async dispose() {
78
+ if (this.hooks.isRegistered()) {
79
+ await this.hooks.unregisterNativeHooks();
80
+ }
81
+ this.initialized = false;
82
+ this.projectDir = null;
83
+ }
84
+ /**
85
+ * Run a health check to verify OpenCode is accessible.
86
+ *
87
+ * Checks:
88
+ * 1. Adapter has been initialized
89
+ * 2. OpenCode CLI is available in PATH
90
+ * 3. .opencode/ configuration directory exists in the project
91
+ *
92
+ * @returns Health status with details about each check
93
+ */
94
+ async healthCheck() {
95
+ const details = {};
96
+ if (!this.initialized) {
97
+ return {
98
+ healthy: false,
99
+ provider: this.id,
100
+ details: { error: 'Adapter not initialized' },
101
+ };
102
+ }
103
+ // Check OpenCode CLI availability
104
+ let cliAvailable = false;
105
+ try {
106
+ const { stdout } = await execAsync('which opencode');
107
+ cliAvailable = stdout.trim().length > 0;
108
+ details.cliPath = stdout.trim();
109
+ }
110
+ catch {
111
+ details.cliAvailable = false;
112
+ }
113
+ // Check for OpenCode config directory in the project
114
+ if (this.projectDir) {
115
+ const openCodeConfigDir = join(this.projectDir, '.opencode');
116
+ const configExists = existsSync(openCodeConfigDir);
117
+ details.configDirExists = configExists;
118
+ }
119
+ // Check for OPENCODE_VERSION env var
120
+ const versionEnvSet = process.env.OPENCODE_VERSION !== undefined;
121
+ details.versionEnvSet = versionEnvSet;
122
+ // Healthy if CLI is available (primary requirement)
123
+ const healthy = cliAvailable;
124
+ details.cliAvailable = cliAvailable;
125
+ return {
126
+ healthy,
127
+ provider: this.id,
128
+ details,
129
+ };
130
+ }
131
+ /**
132
+ * Check whether the adapter has been initialized.
133
+ */
134
+ isInitialized() {
135
+ return this.initialized;
136
+ }
137
+ /**
138
+ * Get the project directory this adapter was initialized with.
139
+ */
140
+ getProjectDir() {
141
+ return this.projectDir;
142
+ }
143
+ }
144
+ //# sourceMappingURL=adapter.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"adapter.js","sourceRoot":"","sources":["adapter.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,165 @@
1
+ /**
2
+ * OpenCode Adapter
3
+ *
4
+ * Main CLEOProviderAdapter implementation for OpenCode AI coding assistant.
5
+ * Provides spawn, hooks, and install capabilities for CLEO integration.
6
+ *
7
+ * @task T5240
8
+ */
9
+
10
+ import { exec } from 'node:child_process';
11
+ import { existsSync } from 'node:fs';
12
+ import { join } from 'node:path';
13
+ import { promisify } from 'node:util';
14
+ import type {
15
+ AdapterCapabilities,
16
+ AdapterHealthStatus,
17
+ CLEOProviderAdapter,
18
+ } from '@cleocode/contracts';
19
+ import { OpenCodeHookProvider } from './hooks.js';
20
+ import { OpenCodeInstallProvider } from './install.js';
21
+ import { OpenCodeSpawnProvider } from './spawn.js';
22
+
23
+ const execAsync = promisify(exec);
24
+
25
+ /**
26
+ * CLEO provider adapter for OpenCode AI coding assistant.
27
+ *
28
+ * Bridges CLEO's adapter system with OpenCode's native capabilities:
29
+ * - Hooks: Maps OpenCode events (session.start, tool.complete, etc.) to CAAMP events
30
+ * - Spawn: Launches subagent processes via the `opencode` CLI
31
+ * - Install: Registers MCP server in .opencode/config.json and ensures AGENTS.md references
32
+ */
33
+ export class OpenCodeAdapter implements CLEOProviderAdapter {
34
+ readonly id = 'opencode';
35
+ readonly name = 'OpenCode';
36
+ readonly version = '1.0.0';
37
+
38
+ capabilities: AdapterCapabilities = {
39
+ supportsHooks: true,
40
+ supportedHookEvents: [
41
+ 'onSessionStart',
42
+ 'onSessionEnd',
43
+ 'onToolStart',
44
+ 'onToolComplete',
45
+ 'onError',
46
+ 'onPromptSubmit',
47
+ ],
48
+ supportsSpawn: true,
49
+ supportsInstall: true,
50
+ supportsMcp: true,
51
+ supportsInstructionFiles: true,
52
+ instructionFilePattern: 'AGENTS.md',
53
+ supportsContextMonitor: false,
54
+ supportsStatusline: false,
55
+ supportsProviderPaths: true,
56
+ supportsTransport: false,
57
+ supportsTaskSync: false,
58
+ };
59
+
60
+ hooks: OpenCodeHookProvider;
61
+ spawn: OpenCodeSpawnProvider;
62
+ install: OpenCodeInstallProvider;
63
+
64
+ private projectDir: string | null = null;
65
+ private initialized = false;
66
+
67
+ constructor() {
68
+ this.hooks = new OpenCodeHookProvider();
69
+ this.spawn = new OpenCodeSpawnProvider();
70
+ this.install = new OpenCodeInstallProvider();
71
+ }
72
+
73
+ /**
74
+ * Initialize the adapter for a given project directory.
75
+ *
76
+ * Validates the environment by checking for the OpenCode CLI
77
+ * and OpenCode configuration directory.
78
+ *
79
+ * @param projectDir - Root directory of the project
80
+ */
81
+ async initialize(projectDir: string): Promise<void> {
82
+ this.projectDir = projectDir;
83
+ this.initialized = true;
84
+ }
85
+
86
+ /**
87
+ * Dispose the adapter and clean up resources.
88
+ *
89
+ * Unregisters hooks and releases any tracked state.
90
+ */
91
+ async dispose(): Promise<void> {
92
+ if (this.hooks.isRegistered()) {
93
+ await this.hooks.unregisterNativeHooks();
94
+ }
95
+ this.initialized = false;
96
+ this.projectDir = null;
97
+ }
98
+
99
+ /**
100
+ * Run a health check to verify OpenCode is accessible.
101
+ *
102
+ * Checks:
103
+ * 1. Adapter has been initialized
104
+ * 2. OpenCode CLI is available in PATH
105
+ * 3. .opencode/ configuration directory exists in the project
106
+ *
107
+ * @returns Health status with details about each check
108
+ */
109
+ async healthCheck(): Promise<AdapterHealthStatus> {
110
+ const details: Record<string, unknown> = {};
111
+
112
+ if (!this.initialized) {
113
+ return {
114
+ healthy: false,
115
+ provider: this.id,
116
+ details: { error: 'Adapter not initialized' },
117
+ };
118
+ }
119
+
120
+ // Check OpenCode CLI availability
121
+ let cliAvailable = false;
122
+ try {
123
+ const { stdout } = await execAsync('which opencode');
124
+ cliAvailable = stdout.trim().length > 0;
125
+ details.cliPath = stdout.trim();
126
+ } catch {
127
+ details.cliAvailable = false;
128
+ }
129
+
130
+ // Check for OpenCode config directory in the project
131
+ if (this.projectDir) {
132
+ const openCodeConfigDir = join(this.projectDir, '.opencode');
133
+ const configExists = existsSync(openCodeConfigDir);
134
+ details.configDirExists = configExists;
135
+ }
136
+
137
+ // Check for OPENCODE_VERSION env var
138
+ const versionEnvSet = process.env.OPENCODE_VERSION !== undefined;
139
+ details.versionEnvSet = versionEnvSet;
140
+
141
+ // Healthy if CLI is available (primary requirement)
142
+ const healthy = cliAvailable;
143
+ details.cliAvailable = cliAvailable;
144
+
145
+ return {
146
+ healthy,
147
+ provider: this.id,
148
+ details,
149
+ };
150
+ }
151
+
152
+ /**
153
+ * Check whether the adapter has been initialized.
154
+ */
155
+ isInitialized(): boolean {
156
+ return this.initialized;
157
+ }
158
+
159
+ /**
160
+ * Get the project directory this adapter was initialized with.
161
+ */
162
+ getProjectDir(): string | null {
163
+ return this.projectDir;
164
+ }
165
+ }
@@ -0,0 +1,66 @@
1
+ /**
2
+ * OpenCode Hook Provider
3
+ *
4
+ * Maps OpenCode's native hook events to CAAMP hook events.
5
+ * OpenCode supports 6 of 8 CAAMP events through its agent/hook system.
6
+ *
7
+ * OpenCode event mapping:
8
+ * - session.start -> onSessionStart
9
+ * - session.end -> onSessionEnd
10
+ * - tool.start -> onToolStart
11
+ * - tool.complete -> onToolComplete
12
+ * - error -> onError
13
+ * - prompt.submit -> onPromptSubmit
14
+ *
15
+ * @task T5240
16
+ */
17
+ import type { AdapterHookProvider } from '@cleocode/contracts';
18
+ /**
19
+ * Hook provider for OpenCode.
20
+ *
21
+ * OpenCode registers hooks via its configuration system at
22
+ * .opencode/config.json. Hook handlers are defined as shell commands
23
+ * or script paths that execute when the corresponding event fires.
24
+ *
25
+ * Since hooks are registered through the config system (managed by
26
+ * the install provider), registerNativeHooks and unregisterNativeHooks
27
+ * track registration state without performing filesystem operations.
28
+ */
29
+ export declare class OpenCodeHookProvider implements AdapterHookProvider {
30
+ private registered;
31
+ /**
32
+ * Map an OpenCode native event name to a CAAMP hook event name.
33
+ *
34
+ * @param providerEvent - OpenCode event name (e.g. "session.start", "tool.complete")
35
+ * @returns CAAMP event name or null if unmapped
36
+ */
37
+ mapProviderEvent(providerEvent: string): string | null;
38
+ /**
39
+ * Register native hooks for a project.
40
+ *
41
+ * For OpenCode, hooks are registered via the config system
42
+ * (.opencode/config.json), which is handled by the install provider.
43
+ * This method marks hooks as registered without performing
44
+ * filesystem operations.
45
+ *
46
+ * @param _projectDir - Project directory (unused; config manages registration)
47
+ */
48
+ registerNativeHooks(_projectDir: string): Promise<void>;
49
+ /**
50
+ * Unregister native hooks.
51
+ *
52
+ * For OpenCode, this is a no-op since hooks are managed through
53
+ * the config system. Unregistration happens via the install
54
+ * provider's uninstall method.
55
+ */
56
+ unregisterNativeHooks(): Promise<void>;
57
+ /**
58
+ * Check whether hooks have been registered via registerNativeHooks.
59
+ */
60
+ isRegistered(): boolean;
61
+ /**
62
+ * Get the full event mapping for introspection/debugging.
63
+ */
64
+ getEventMap(): Readonly<Record<string, string>>;
65
+ }
66
+ //# sourceMappingURL=hooks.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"hooks.d.ts","sourceRoot":"","sources":["hooks.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;GAeG;AAEH,OAAO,KAAK,EAAE,mBAAmB,EAAE,MAAM,qBAAqB,CAAC;AAiB/D;;;;;;;;;;GAUG;AACH,qBAAa,oBAAqB,YAAW,mBAAmB;IAC9D,OAAO,CAAC,UAAU,CAAS;IAE3B;;;;;OAKG;IACH,gBAAgB,CAAC,aAAa,EAAE,MAAM,GAAG,MAAM,GAAG,IAAI;IAItD;;;;;;;;;OASG;IACG,mBAAmB,CAAC,WAAW,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAI7D;;;;;;OAMG;IACG,qBAAqB,IAAI,OAAO,CAAC,IAAI,CAAC;IAI5C;;OAEG;IACH,YAAY,IAAI,OAAO;IAIvB;;OAEG;IACH,WAAW,IAAI,QAAQ,CAAC,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;CAGhD"}
@@ -0,0 +1,89 @@
1
+ /**
2
+ * OpenCode Hook Provider
3
+ *
4
+ * Maps OpenCode's native hook events to CAAMP hook events.
5
+ * OpenCode supports 6 of 8 CAAMP events through its agent/hook system.
6
+ *
7
+ * OpenCode event mapping:
8
+ * - session.start -> onSessionStart
9
+ * - session.end -> onSessionEnd
10
+ * - tool.start -> onToolStart
11
+ * - tool.complete -> onToolComplete
12
+ * - error -> onError
13
+ * - prompt.submit -> onPromptSubmit
14
+ *
15
+ * @task T5240
16
+ */
17
+ /**
18
+ * Mapping from OpenCode native event names to CAAMP event names.
19
+ *
20
+ * OpenCode uses dot-delimited event names (e.g. "session.start")
21
+ * while CAAMP uses camelCase (e.g. "onSessionStart").
22
+ */
23
+ const OPENCODE_EVENT_MAP = {
24
+ 'session.start': 'onSessionStart',
25
+ 'session.end': 'onSessionEnd',
26
+ 'tool.start': 'onToolStart',
27
+ 'tool.complete': 'onToolComplete',
28
+ 'error': 'onError',
29
+ 'prompt.submit': 'onPromptSubmit',
30
+ };
31
+ /**
32
+ * Hook provider for OpenCode.
33
+ *
34
+ * OpenCode registers hooks via its configuration system at
35
+ * .opencode/config.json. Hook handlers are defined as shell commands
36
+ * or script paths that execute when the corresponding event fires.
37
+ *
38
+ * Since hooks are registered through the config system (managed by
39
+ * the install provider), registerNativeHooks and unregisterNativeHooks
40
+ * track registration state without performing filesystem operations.
41
+ */
42
+ export class OpenCodeHookProvider {
43
+ registered = false;
44
+ /**
45
+ * Map an OpenCode native event name to a CAAMP hook event name.
46
+ *
47
+ * @param providerEvent - OpenCode event name (e.g. "session.start", "tool.complete")
48
+ * @returns CAAMP event name or null if unmapped
49
+ */
50
+ mapProviderEvent(providerEvent) {
51
+ return OPENCODE_EVENT_MAP[providerEvent] ?? null;
52
+ }
53
+ /**
54
+ * Register native hooks for a project.
55
+ *
56
+ * For OpenCode, hooks are registered via the config system
57
+ * (.opencode/config.json), which is handled by the install provider.
58
+ * This method marks hooks as registered without performing
59
+ * filesystem operations.
60
+ *
61
+ * @param _projectDir - Project directory (unused; config manages registration)
62
+ */
63
+ async registerNativeHooks(_projectDir) {
64
+ this.registered = true;
65
+ }
66
+ /**
67
+ * Unregister native hooks.
68
+ *
69
+ * For OpenCode, this is a no-op since hooks are managed through
70
+ * the config system. Unregistration happens via the install
71
+ * provider's uninstall method.
72
+ */
73
+ async unregisterNativeHooks() {
74
+ this.registered = false;
75
+ }
76
+ /**
77
+ * Check whether hooks have been registered via registerNativeHooks.
78
+ */
79
+ isRegistered() {
80
+ return this.registered;
81
+ }
82
+ /**
83
+ * Get the full event mapping for introspection/debugging.
84
+ */
85
+ getEventMap() {
86
+ return { ...OPENCODE_EVENT_MAP };
87
+ }
88
+ }
89
+ //# sourceMappingURL=hooks.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"hooks.js","sourceRoot":"","sources":["hooks.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;GAeG;AAIH;;;;;GAKG;AACH,MAAM,kBAAkB,GAA2B;IACjD,eAAe,EAAE,gBAAgB;IACjC,aAAa,EAAE,cAAc;IAC7B,YAAY,EAAE,aAAa;IAC3B,eAAe,EAAE,gBAAgB;IACjC,OAAO,EAAE,SAAS;IAClB,eAAe,EAAE,gBAAgB;CAClC,CAAC;AAEF;;;;;;;;;;GAUG;AACH,MAAM,OAAO,oBAAoB;IACvB,UAAU,GAAG,KAAK,CAAC;IAE3B;;;;;OAKG;IACH,gBAAgB,CAAC,aAAqB;QACpC,OAAO,kBAAkB,CAAC,aAAa,CAAC,IAAI,IAAI,CAAC;IACnD,CAAC;IAED;;;;;;;;;OASG;IACH,KAAK,CAAC,mBAAmB,CAAC,WAAmB;QAC3C,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC;IACzB,CAAC;IAED;;;;;;OAMG;IACH,KAAK,CAAC,qBAAqB;QACzB,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC;IAC1B,CAAC;IAED;;OAEG;IACH,YAAY;QACV,OAAO,IAAI,CAAC,UAAU,CAAC;IACzB,CAAC;IAED;;OAEG;IACH,WAAW;QACT,OAAO,EAAE,GAAG,kBAAkB,EAAE,CAAC;IACnC,CAAC;CACF"}
@@ -0,0 +1,97 @@
1
+ /**
2
+ * OpenCode Hook Provider
3
+ *
4
+ * Maps OpenCode's native hook events to CAAMP hook events.
5
+ * OpenCode supports 6 of 8 CAAMP events through its agent/hook system.
6
+ *
7
+ * OpenCode event mapping:
8
+ * - session.start -> onSessionStart
9
+ * - session.end -> onSessionEnd
10
+ * - tool.start -> onToolStart
11
+ * - tool.complete -> onToolComplete
12
+ * - error -> onError
13
+ * - prompt.submit -> onPromptSubmit
14
+ *
15
+ * @task T5240
16
+ */
17
+
18
+ import type { AdapterHookProvider } from '@cleocode/contracts';
19
+
20
+ /**
21
+ * Mapping from OpenCode native event names to CAAMP event names.
22
+ *
23
+ * OpenCode uses dot-delimited event names (e.g. "session.start")
24
+ * while CAAMP uses camelCase (e.g. "onSessionStart").
25
+ */
26
+ const OPENCODE_EVENT_MAP: Record<string, string> = {
27
+ 'session.start': 'onSessionStart',
28
+ 'session.end': 'onSessionEnd',
29
+ 'tool.start': 'onToolStart',
30
+ 'tool.complete': 'onToolComplete',
31
+ 'error': 'onError',
32
+ 'prompt.submit': 'onPromptSubmit',
33
+ };
34
+
35
+ /**
36
+ * Hook provider for OpenCode.
37
+ *
38
+ * OpenCode registers hooks via its configuration system at
39
+ * .opencode/config.json. Hook handlers are defined as shell commands
40
+ * or script paths that execute when the corresponding event fires.
41
+ *
42
+ * Since hooks are registered through the config system (managed by
43
+ * the install provider), registerNativeHooks and unregisterNativeHooks
44
+ * track registration state without performing filesystem operations.
45
+ */
46
+ export class OpenCodeHookProvider implements AdapterHookProvider {
47
+ private registered = false;
48
+
49
+ /**
50
+ * Map an OpenCode native event name to a CAAMP hook event name.
51
+ *
52
+ * @param providerEvent - OpenCode event name (e.g. "session.start", "tool.complete")
53
+ * @returns CAAMP event name or null if unmapped
54
+ */
55
+ mapProviderEvent(providerEvent: string): string | null {
56
+ return OPENCODE_EVENT_MAP[providerEvent] ?? null;
57
+ }
58
+
59
+ /**
60
+ * Register native hooks for a project.
61
+ *
62
+ * For OpenCode, hooks are registered via the config system
63
+ * (.opencode/config.json), which is handled by the install provider.
64
+ * This method marks hooks as registered without performing
65
+ * filesystem operations.
66
+ *
67
+ * @param _projectDir - Project directory (unused; config manages registration)
68
+ */
69
+ async registerNativeHooks(_projectDir: string): Promise<void> {
70
+ this.registered = true;
71
+ }
72
+
73
+ /**
74
+ * Unregister native hooks.
75
+ *
76
+ * For OpenCode, this is a no-op since hooks are managed through
77
+ * the config system. Unregistration happens via the install
78
+ * provider's uninstall method.
79
+ */
80
+ async unregisterNativeHooks(): Promise<void> {
81
+ this.registered = false;
82
+ }
83
+
84
+ /**
85
+ * Check whether hooks have been registered via registerNativeHooks.
86
+ */
87
+ isRegistered(): boolean {
88
+ return this.registered;
89
+ }
90
+
91
+ /**
92
+ * Get the full event mapping for introspection/debugging.
93
+ */
94
+ getEventMap(): Readonly<Record<string, string>> {
95
+ return { ...OPENCODE_EVENT_MAP };
96
+ }
97
+ }
@@ -0,0 +1,20 @@
1
+ /**
2
+ * OpenCode provider adapter.
3
+ *
4
+ * CLEO provider adapter for OpenCode AI coding assistant.
5
+ * Default export is the adapter class for dynamic loading by AdapterManager.
6
+ *
7
+ * @task T5240
8
+ */
9
+ import { OpenCodeAdapter } from './adapter.js';
10
+ export { OpenCodeAdapter } from './adapter.js';
11
+ export { OpenCodeHookProvider } from './hooks.js';
12
+ export { OpenCodeSpawnProvider } from './spawn.js';
13
+ export { OpenCodeInstallProvider } from './install.js';
14
+ export default OpenCodeAdapter;
15
+ /**
16
+ * Factory function for creating adapter instances.
17
+ * Used by AdapterManager's dynamic import fallback.
18
+ */
19
+ export declare function createAdapter(): OpenCodeAdapter;
20
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["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,qBAAqB,EAAE,MAAM,YAAY,CAAC;AACnD,OAAO,EAAE,uBAAuB,EAAE,MAAM,cAAc,CAAC;AAEvD,eAAe,eAAe,CAAC;AAE/B;;;GAGG;AACH,wBAAgB,aAAa,IAAI,eAAe,CAE/C"}
@@ -0,0 +1,22 @@
1
+ /**
2
+ * OpenCode provider adapter.
3
+ *
4
+ * CLEO provider adapter for OpenCode AI coding assistant.
5
+ * Default export is the adapter class for dynamic loading by AdapterManager.
6
+ *
7
+ * @task T5240
8
+ */
9
+ import { OpenCodeAdapter } from './adapter.js';
10
+ export { OpenCodeAdapter } from './adapter.js';
11
+ export { OpenCodeHookProvider } from './hooks.js';
12
+ export { OpenCodeSpawnProvider } from './spawn.js';
13
+ export { OpenCodeInstallProvider } from './install.js';
14
+ export default OpenCodeAdapter;
15
+ /**
16
+ * Factory function for creating adapter instances.
17
+ * Used by AdapterManager's dynamic import fallback.
18
+ */
19
+ export function createAdapter() {
20
+ return new OpenCodeAdapter();
21
+ }
22
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["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,qBAAqB,EAAE,MAAM,YAAY,CAAC;AACnD,OAAO,EAAE,uBAAuB,EAAE,MAAM,cAAc,CAAC;AAEvD,eAAe,eAAe,CAAC;AAE/B;;;GAGG;AACH,MAAM,UAAU,aAAa;IAC3B,OAAO,IAAI,eAAe,EAAE,CAAC;AAC/B,CAAC"}
@@ -0,0 +1,25 @@
1
+ /**
2
+ * OpenCode provider adapter.
3
+ *
4
+ * CLEO provider adapter for OpenCode AI coding assistant.
5
+ * Default export is the adapter class for dynamic loading by AdapterManager.
6
+ *
7
+ * @task T5240
8
+ */
9
+
10
+ import { OpenCodeAdapter } from './adapter.js';
11
+
12
+ export { OpenCodeAdapter } from './adapter.js';
13
+ export { OpenCodeHookProvider } from './hooks.js';
14
+ export { OpenCodeSpawnProvider } from './spawn.js';
15
+ export { OpenCodeInstallProvider } from './install.js';
16
+
17
+ export default OpenCodeAdapter;
18
+
19
+ /**
20
+ * Factory function for creating adapter instances.
21
+ * Used by AdapterManager's dynamic import fallback.
22
+ */
23
+ export function createAdapter(): OpenCodeAdapter {
24
+ return new OpenCodeAdapter();
25
+ }