@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,65 @@
1
+ /**
2
+ * OpenCode Install Provider
3
+ *
4
+ * Handles CLEO installation into OpenCode environments:
5
+ * - Registers CLEO MCP server in .opencode/config.json
6
+ * - Ensures AGENTS.md has CLEO @-references
7
+ *
8
+ * @task T5240
9
+ */
10
+ import type { AdapterInstallProvider, InstallOptions, InstallResult } from '@cleocode/contracts';
11
+ /**
12
+ * Install provider for OpenCode.
13
+ *
14
+ * Manages CLEO's integration with OpenCode by:
15
+ * 1. Registering the CLEO MCP server in .opencode/config.json
16
+ * 2. Ensuring AGENTS.md contains @-references to CLEO instruction files
17
+ */
18
+ export declare class OpenCodeInstallProvider implements AdapterInstallProvider {
19
+ private installedProjectDir;
20
+ /**
21
+ * Install CLEO into an OpenCode project.
22
+ *
23
+ * @param options - Installation options including project directory and MCP server path
24
+ * @returns Result describing what was installed
25
+ */
26
+ install(options: InstallOptions): Promise<InstallResult>;
27
+ /**
28
+ * Uninstall CLEO from the current OpenCode project.
29
+ *
30
+ * Removes the MCP server registration from .opencode/config.json.
31
+ * Does not remove AGENTS.md references (they are harmless if CLEO is not present).
32
+ */
33
+ uninstall(): Promise<void>;
34
+ /**
35
+ * Check whether CLEO is installed in the current environment.
36
+ *
37
+ * Checks for MCP server registered in .opencode/config.json.
38
+ * Returns true if the CLEO MCP server entry is found.
39
+ */
40
+ isInstalled(): Promise<boolean>;
41
+ /**
42
+ * Ensure AGENTS.md contains @-references to CLEO instruction files.
43
+ *
44
+ * Creates AGENTS.md if it does not exist. Appends any missing references.
45
+ *
46
+ * @param projectDir - Project root directory
47
+ */
48
+ ensureInstructionReferences(projectDir: string): Promise<void>;
49
+ /**
50
+ * Register the CLEO MCP server in .opencode/config.json.
51
+ *
52
+ * OpenCode stores its MCP server configuration in .opencode/config.json
53
+ * under the mcpServers key.
54
+ *
55
+ * @returns true if registration was performed or updated
56
+ */
57
+ private registerMcpServer;
58
+ /**
59
+ * Update AGENTS.md with CLEO @-references.
60
+ *
61
+ * @returns true if the file was created or modified
62
+ */
63
+ private updateInstructionFile;
64
+ }
65
+ //# sourceMappingURL=install.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"install.d.ts","sourceRoot":"","sources":["install.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAIH,OAAO,KAAK,EAAE,sBAAsB,EAAE,cAAc,EAAE,aAAa,EAAE,MAAM,qBAAqB,CAAC;AAWjG;;;;;;GAMG;AACH,qBAAa,uBAAwB,YAAW,sBAAsB;IACpE,OAAO,CAAC,mBAAmB,CAAuB;IAElD;;;;;OAKG;IACG,OAAO,CAAC,OAAO,EAAE,cAAc,GAAG,OAAO,CAAC,aAAa,CAAC;IAgC9D;;;;;OAKG;IACG,SAAS,IAAI,OAAO,CAAC,IAAI,CAAC;IAqBhC;;;;;OAKG;IACG,WAAW,IAAI,OAAO,CAAC,OAAO,CAAC;IAkBrC;;;;;;OAMG;IACG,2BAA2B,CAAC,UAAU,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAIpE;;;;;;;OAOG;IACH,OAAO,CAAC,iBAAiB;IA6BzB;;;;OAIG;IACH,OAAO,CAAC,qBAAqB;CA8B9B"}
@@ -0,0 +1,183 @@
1
+ /**
2
+ * OpenCode Install Provider
3
+ *
4
+ * Handles CLEO installation into OpenCode environments:
5
+ * - Registers CLEO MCP server in .opencode/config.json
6
+ * - Ensures AGENTS.md has CLEO @-references
7
+ *
8
+ * @task T5240
9
+ */
10
+ import { existsSync, mkdirSync, readFileSync, writeFileSync } from 'node:fs';
11
+ import { join } from 'node:path';
12
+ /** Lines that should appear in AGENTS.md to reference CLEO. */
13
+ const INSTRUCTION_REFERENCES = [
14
+ '@~/.cleo/templates/CLEO-INJECTION.md',
15
+ '@.cleo/memory-bridge.md',
16
+ ];
17
+ /** MCP server registration key used in OpenCode config. */
18
+ const MCP_SERVER_KEY = 'cleo';
19
+ /**
20
+ * Install provider for OpenCode.
21
+ *
22
+ * Manages CLEO's integration with OpenCode by:
23
+ * 1. Registering the CLEO MCP server in .opencode/config.json
24
+ * 2. Ensuring AGENTS.md contains @-references to CLEO instruction files
25
+ */
26
+ export class OpenCodeInstallProvider {
27
+ installedProjectDir = null;
28
+ /**
29
+ * Install CLEO into an OpenCode project.
30
+ *
31
+ * @param options - Installation options including project directory and MCP server path
32
+ * @returns Result describing what was installed
33
+ */
34
+ async install(options) {
35
+ const { projectDir, mcpServerPath } = options;
36
+ const installedAt = new Date().toISOString();
37
+ let instructionFileUpdated = false;
38
+ let mcpRegistered = false;
39
+ const details = {};
40
+ // Step 1: Register MCP server in .opencode/config.json
41
+ if (mcpServerPath) {
42
+ mcpRegistered = this.registerMcpServer(projectDir, mcpServerPath);
43
+ if (mcpRegistered) {
44
+ details.mcpConfigPath = join(projectDir, '.opencode', 'config.json');
45
+ }
46
+ }
47
+ // Step 2: Ensure AGENTS.md has @-references
48
+ instructionFileUpdated = this.updateInstructionFile(projectDir);
49
+ if (instructionFileUpdated) {
50
+ details.instructionFile = join(projectDir, 'AGENTS.md');
51
+ }
52
+ this.installedProjectDir = projectDir;
53
+ return {
54
+ success: true,
55
+ installedAt,
56
+ instructionFileUpdated,
57
+ mcpRegistered,
58
+ details,
59
+ };
60
+ }
61
+ /**
62
+ * Uninstall CLEO from the current OpenCode project.
63
+ *
64
+ * Removes the MCP server registration from .opencode/config.json.
65
+ * Does not remove AGENTS.md references (they are harmless if CLEO is not present).
66
+ */
67
+ async uninstall() {
68
+ if (!this.installedProjectDir)
69
+ return;
70
+ const configPath = join(this.installedProjectDir, '.opencode', 'config.json');
71
+ if (existsSync(configPath)) {
72
+ try {
73
+ const raw = readFileSync(configPath, 'utf-8');
74
+ const config = JSON.parse(raw);
75
+ const mcpServers = config.mcpServers;
76
+ if (mcpServers && MCP_SERVER_KEY in mcpServers) {
77
+ delete mcpServers[MCP_SERVER_KEY];
78
+ writeFileSync(configPath, JSON.stringify(config, null, 2) + '\n', 'utf-8');
79
+ }
80
+ }
81
+ catch {
82
+ // Ignore errors during uninstall
83
+ }
84
+ }
85
+ this.installedProjectDir = null;
86
+ }
87
+ /**
88
+ * Check whether CLEO is installed in the current environment.
89
+ *
90
+ * Checks for MCP server registered in .opencode/config.json.
91
+ * Returns true if the CLEO MCP server entry is found.
92
+ */
93
+ async isInstalled() {
94
+ // Check current directory for .opencode/config.json with cleo server
95
+ const configPath = join(process.cwd(), '.opencode', 'config.json');
96
+ if (existsSync(configPath)) {
97
+ try {
98
+ const config = JSON.parse(readFileSync(configPath, 'utf-8'));
99
+ const mcpServers = config.mcpServers;
100
+ if (mcpServers && MCP_SERVER_KEY in mcpServers) {
101
+ return true;
102
+ }
103
+ }
104
+ catch {
105
+ // Fall through
106
+ }
107
+ }
108
+ return false;
109
+ }
110
+ /**
111
+ * Ensure AGENTS.md contains @-references to CLEO instruction files.
112
+ *
113
+ * Creates AGENTS.md if it does not exist. Appends any missing references.
114
+ *
115
+ * @param projectDir - Project root directory
116
+ */
117
+ async ensureInstructionReferences(projectDir) {
118
+ this.updateInstructionFile(projectDir);
119
+ }
120
+ /**
121
+ * Register the CLEO MCP server in .opencode/config.json.
122
+ *
123
+ * OpenCode stores its MCP server configuration in .opencode/config.json
124
+ * under the mcpServers key.
125
+ *
126
+ * @returns true if registration was performed or updated
127
+ */
128
+ registerMcpServer(projectDir, mcpServerPath) {
129
+ const openCodeDir = join(projectDir, '.opencode');
130
+ const configPath = join(openCodeDir, 'config.json');
131
+ let config = {};
132
+ mkdirSync(openCodeDir, { recursive: true });
133
+ if (existsSync(configPath)) {
134
+ try {
135
+ config = JSON.parse(readFileSync(configPath, 'utf-8'));
136
+ }
137
+ catch {
138
+ // Start fresh on parse error
139
+ }
140
+ }
141
+ if (!config.mcpServers || typeof config.mcpServers !== 'object') {
142
+ config.mcpServers = {};
143
+ }
144
+ const mcpServers = config.mcpServers;
145
+ mcpServers[MCP_SERVER_KEY] = {
146
+ command: 'node',
147
+ args: [mcpServerPath],
148
+ };
149
+ writeFileSync(configPath, JSON.stringify(config, null, 2) + '\n', 'utf-8');
150
+ return true;
151
+ }
152
+ /**
153
+ * Update AGENTS.md with CLEO @-references.
154
+ *
155
+ * @returns true if the file was created or modified
156
+ */
157
+ updateInstructionFile(projectDir) {
158
+ const agentsMdPath = join(projectDir, 'AGENTS.md');
159
+ let content = '';
160
+ let existed = false;
161
+ if (existsSync(agentsMdPath)) {
162
+ content = readFileSync(agentsMdPath, 'utf-8');
163
+ existed = true;
164
+ }
165
+ const missingRefs = INSTRUCTION_REFERENCES.filter((ref) => !content.includes(ref));
166
+ if (missingRefs.length === 0) {
167
+ return false;
168
+ }
169
+ const refsBlock = missingRefs.join('\n');
170
+ if (existed) {
171
+ // Append missing references
172
+ const separator = content.endsWith('\n') ? '' : '\n';
173
+ content = content + separator + refsBlock + '\n';
174
+ }
175
+ else {
176
+ // Create new AGENTS.md with references
177
+ content = refsBlock + '\n';
178
+ }
179
+ writeFileSync(agentsMdPath, content, 'utf-8');
180
+ return true;
181
+ }
182
+ }
183
+ //# sourceMappingURL=install.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"install.js","sourceRoot":"","sources":["install.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,206 @@
1
+ /**
2
+ * OpenCode Install Provider
3
+ *
4
+ * Handles CLEO installation into OpenCode environments:
5
+ * - Registers CLEO MCP server in .opencode/config.json
6
+ * - Ensures AGENTS.md has CLEO @-references
7
+ *
8
+ * @task T5240
9
+ */
10
+
11
+ import { existsSync, mkdirSync, readFileSync, writeFileSync } from 'node:fs';
12
+ import { join } from 'node:path';
13
+ import type { AdapterInstallProvider, InstallOptions, InstallResult } from '@cleocode/contracts';
14
+
15
+ /** Lines that should appear in AGENTS.md to reference CLEO. */
16
+ const INSTRUCTION_REFERENCES = [
17
+ '@~/.cleo/templates/CLEO-INJECTION.md',
18
+ '@.cleo/memory-bridge.md',
19
+ ];
20
+
21
+ /** MCP server registration key used in OpenCode config. */
22
+ const MCP_SERVER_KEY = 'cleo';
23
+
24
+ /**
25
+ * Install provider for OpenCode.
26
+ *
27
+ * Manages CLEO's integration with OpenCode by:
28
+ * 1. Registering the CLEO MCP server in .opencode/config.json
29
+ * 2. Ensuring AGENTS.md contains @-references to CLEO instruction files
30
+ */
31
+ export class OpenCodeInstallProvider implements AdapterInstallProvider {
32
+ private installedProjectDir: string | null = null;
33
+
34
+ /**
35
+ * Install CLEO into an OpenCode project.
36
+ *
37
+ * @param options - Installation options including project directory and MCP server path
38
+ * @returns Result describing what was installed
39
+ */
40
+ async install(options: InstallOptions): Promise<InstallResult> {
41
+ const { projectDir, mcpServerPath } = options;
42
+ const installedAt = new Date().toISOString();
43
+ let instructionFileUpdated = false;
44
+ let mcpRegistered = false;
45
+ const details: Record<string, unknown> = {};
46
+
47
+ // Step 1: Register MCP server in .opencode/config.json
48
+ if (mcpServerPath) {
49
+ mcpRegistered = this.registerMcpServer(projectDir, mcpServerPath);
50
+ if (mcpRegistered) {
51
+ details.mcpConfigPath = join(projectDir, '.opencode', 'config.json');
52
+ }
53
+ }
54
+
55
+ // Step 2: Ensure AGENTS.md has @-references
56
+ instructionFileUpdated = this.updateInstructionFile(projectDir);
57
+ if (instructionFileUpdated) {
58
+ details.instructionFile = join(projectDir, 'AGENTS.md');
59
+ }
60
+
61
+ this.installedProjectDir = projectDir;
62
+
63
+ return {
64
+ success: true,
65
+ installedAt,
66
+ instructionFileUpdated,
67
+ mcpRegistered,
68
+ details,
69
+ };
70
+ }
71
+
72
+ /**
73
+ * Uninstall CLEO from the current OpenCode project.
74
+ *
75
+ * Removes the MCP server registration from .opencode/config.json.
76
+ * Does not remove AGENTS.md references (they are harmless if CLEO is not present).
77
+ */
78
+ async uninstall(): Promise<void> {
79
+ if (!this.installedProjectDir) return;
80
+
81
+ const configPath = join(this.installedProjectDir, '.opencode', 'config.json');
82
+ if (existsSync(configPath)) {
83
+ try {
84
+ const raw = readFileSync(configPath, 'utf-8');
85
+ const config = JSON.parse(raw) as Record<string, unknown>;
86
+ const mcpServers = config.mcpServers as Record<string, unknown> | undefined;
87
+ if (mcpServers && MCP_SERVER_KEY in mcpServers) {
88
+ delete mcpServers[MCP_SERVER_KEY];
89
+ writeFileSync(configPath, JSON.stringify(config, null, 2) + '\n', 'utf-8');
90
+ }
91
+ } catch {
92
+ // Ignore errors during uninstall
93
+ }
94
+ }
95
+
96
+ this.installedProjectDir = null;
97
+ }
98
+
99
+ /**
100
+ * Check whether CLEO is installed in the current environment.
101
+ *
102
+ * Checks for MCP server registered in .opencode/config.json.
103
+ * Returns true if the CLEO MCP server entry is found.
104
+ */
105
+ async isInstalled(): Promise<boolean> {
106
+ // Check current directory for .opencode/config.json with cleo server
107
+ const configPath = join(process.cwd(), '.opencode', 'config.json');
108
+ if (existsSync(configPath)) {
109
+ try {
110
+ const config = JSON.parse(readFileSync(configPath, 'utf-8'));
111
+ const mcpServers = config.mcpServers as Record<string, unknown> | undefined;
112
+ if (mcpServers && MCP_SERVER_KEY in mcpServers) {
113
+ return true;
114
+ }
115
+ } catch {
116
+ // Fall through
117
+ }
118
+ }
119
+
120
+ return false;
121
+ }
122
+
123
+ /**
124
+ * Ensure AGENTS.md contains @-references to CLEO instruction files.
125
+ *
126
+ * Creates AGENTS.md if it does not exist. Appends any missing references.
127
+ *
128
+ * @param projectDir - Project root directory
129
+ */
130
+ async ensureInstructionReferences(projectDir: string): Promise<void> {
131
+ this.updateInstructionFile(projectDir);
132
+ }
133
+
134
+ /**
135
+ * Register the CLEO MCP server in .opencode/config.json.
136
+ *
137
+ * OpenCode stores its MCP server configuration in .opencode/config.json
138
+ * under the mcpServers key.
139
+ *
140
+ * @returns true if registration was performed or updated
141
+ */
142
+ private registerMcpServer(projectDir: string, mcpServerPath: string): boolean {
143
+ const openCodeDir = join(projectDir, '.opencode');
144
+ const configPath = join(openCodeDir, 'config.json');
145
+ let config: Record<string, unknown> = {};
146
+
147
+ mkdirSync(openCodeDir, { recursive: true });
148
+
149
+ if (existsSync(configPath)) {
150
+ try {
151
+ config = JSON.parse(readFileSync(configPath, 'utf-8'));
152
+ } catch {
153
+ // Start fresh on parse error
154
+ }
155
+ }
156
+
157
+ if (!config.mcpServers || typeof config.mcpServers !== 'object') {
158
+ config.mcpServers = {};
159
+ }
160
+
161
+ const mcpServers = config.mcpServers as Record<string, unknown>;
162
+ mcpServers[MCP_SERVER_KEY] = {
163
+ command: 'node',
164
+ args: [mcpServerPath],
165
+ };
166
+
167
+ writeFileSync(configPath, JSON.stringify(config, null, 2) + '\n', 'utf-8');
168
+ return true;
169
+ }
170
+
171
+ /**
172
+ * Update AGENTS.md with CLEO @-references.
173
+ *
174
+ * @returns true if the file was created or modified
175
+ */
176
+ private updateInstructionFile(projectDir: string): boolean {
177
+ const agentsMdPath = join(projectDir, 'AGENTS.md');
178
+ let content = '';
179
+ let existed = false;
180
+
181
+ if (existsSync(agentsMdPath)) {
182
+ content = readFileSync(agentsMdPath, 'utf-8');
183
+ existed = true;
184
+ }
185
+
186
+ const missingRefs = INSTRUCTION_REFERENCES.filter((ref) => !content.includes(ref));
187
+
188
+ if (missingRefs.length === 0) {
189
+ return false;
190
+ }
191
+
192
+ const refsBlock = missingRefs.join('\n');
193
+
194
+ if (existed) {
195
+ // Append missing references
196
+ const separator = content.endsWith('\n') ? '' : '\n';
197
+ content = content + separator + refsBlock + '\n';
198
+ } else {
199
+ // Create new AGENTS.md with references
200
+ content = refsBlock + '\n';
201
+ }
202
+
203
+ writeFileSync(agentsMdPath, content, 'utf-8');
204
+ return true;
205
+ }
206
+ }
@@ -0,0 +1,26 @@
1
+ {
2
+ "id": "opencode",
3
+ "name": "OpenCode Adapter",
4
+ "version": "1.0.0",
5
+ "description": "CLEO adapter for OpenCode AI coding assistant",
6
+ "provider": "opencode",
7
+ "entryPoint": "src/index.ts",
8
+ "capabilities": {
9
+ "supportsHooks": true,
10
+ "supportedHookEvents": ["onSessionStart", "onSessionEnd", "onToolStart", "onToolComplete", "onError", "onPromptSubmit"],
11
+ "supportsSpawn": true,
12
+ "supportsInstall": true,
13
+ "supportsMcp": true,
14
+ "supportsInstructionFiles": true,
15
+ "instructionFilePattern": "AGENTS.md",
16
+ "supportsContextMonitor": false,
17
+ "supportsStatusline": false,
18
+ "supportsProviderPaths": true,
19
+ "supportsTransport": false
20
+ },
21
+ "detectionPatterns": [
22
+ { "type": "env", "pattern": "OPENCODE_VERSION", "description": "OpenCode sets this env var" },
23
+ { "type": "file", "pattern": ".opencode/config.json", "description": "OpenCode config directory" },
24
+ { "type": "cli", "pattern": "opencode", "description": "OpenCode CLI available in PATH" }
25
+ ]
26
+ }
@@ -0,0 +1,72 @@
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 type { AdapterSpawnProvider, SpawnContext, SpawnResult } from '@cleocode/contracts';
14
+ /**
15
+ * Build the markdown content for an OpenCode agent definition file.
16
+ *
17
+ * OpenCode agents are defined as markdown files with YAML frontmatter
18
+ * in the .opencode/agent/ directory.
19
+ *
20
+ * @param description - Agent description for frontmatter
21
+ * @param instructions - Markdown instructions body
22
+ * @returns Complete agent definition markdown
23
+ */
24
+ export declare function buildOpenCodeAgentMarkdown(description: string, instructions: string): string;
25
+ /**
26
+ * Spawn provider for OpenCode.
27
+ *
28
+ * Spawns detached OpenCode CLI processes for subagent execution.
29
+ * Each spawn ensures a CLEO subagent definition exists, then runs
30
+ * `opencode run --format json --agent <name> --title <title> <prompt>`
31
+ * as a detached, unref'd child process.
32
+ */
33
+ export declare class OpenCodeSpawnProvider implements AdapterSpawnProvider {
34
+ /** Map of instance IDs to tracked process info. */
35
+ private processMap;
36
+ /**
37
+ * Check if the OpenCode CLI is available in PATH.
38
+ *
39
+ * @returns true if `opencode` is found via `which`
40
+ */
41
+ canSpawn(): Promise<boolean>;
42
+ /**
43
+ * Spawn a subagent via OpenCode CLI.
44
+ *
45
+ * Ensures the CLEO subagent definition exists in the project's
46
+ * .opencode/agent/ directory, then spawns a detached OpenCode
47
+ * process. The process runs independently of the parent.
48
+ *
49
+ * @param context - Spawn context with taskId, prompt, and options
50
+ * @returns Spawn result with instance ID and status
51
+ */
52
+ spawn(context: SpawnContext): Promise<SpawnResult>;
53
+ /**
54
+ * List currently running OpenCode subagent processes.
55
+ *
56
+ * Checks each tracked process via kill(pid, 0) to verify it is still alive.
57
+ * Dead processes are automatically cleaned from the tracking map.
58
+ *
59
+ * @returns Array of spawn results for running processes
60
+ */
61
+ listRunning(): Promise<SpawnResult[]>;
62
+ /**
63
+ * Terminate a running spawn by instance ID.
64
+ *
65
+ * Sends SIGTERM to the tracked process. If the process is not found
66
+ * or has already exited, this is a no-op.
67
+ *
68
+ * @param instanceId - ID of the spawn instance to terminate
69
+ */
70
+ terminate(instanceId: string): Promise<void>;
71
+ }
72
+ //# sourceMappingURL=spawn.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"spawn.d.ts","sourceRoot":"","sources":["spawn.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;GAWG;AAMH,OAAO,KAAK,EAAE,oBAAoB,EAAE,YAAY,EAAE,WAAW,EAAE,MAAM,qBAAqB,CAAC;AAiB3F;;;;;;;;;GASG;AACH,wBAAgB,0BAA0B,CACxC,WAAW,EAAE,MAAM,EACnB,YAAY,EAAE,MAAM,GACnB,MAAM,CAYR;AA4CD;;;;;;;GAOG;AACH,qBAAa,qBAAsB,YAAW,oBAAoB;IAChE,mDAAmD;IACnD,OAAO,CAAC,UAAU,CAAqC;IAEvD;;;;OAIG;IACG,QAAQ,IAAI,OAAO,CAAC,OAAO,CAAC;IASlC;;;;;;;;;OASG;IACG,KAAK,CAAC,OAAO,EAAE,YAAY,GAAG,OAAO,CAAC,WAAW,CAAC;IAiExD;;;;;;;OAOG;IACG,WAAW,IAAI,OAAO,CAAC,WAAW,EAAE,CAAC;IAqB3C;;;;;;;OAOG;IACG,SAAS,CAAC,UAAU,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;CAWnD"}