@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.
- package/dist/index.d.ts +27 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +1927 -0
- package/dist/index.js.map +7 -0
- package/dist/providers/claude-code/adapter.d.ts +75 -0
- package/dist/providers/claude-code/adapter.d.ts.map +1 -0
- package/dist/providers/claude-code/adapter.js +154 -0
- package/dist/providers/claude-code/adapter.js.map +1 -0
- package/dist/providers/claude-code/context-monitor.d.ts +24 -0
- package/dist/providers/claude-code/context-monitor.d.ts.map +1 -0
- package/dist/providers/claude-code/context-monitor.js +148 -0
- package/dist/providers/claude-code/context-monitor.js.map +1 -0
- package/dist/providers/claude-code/hooks.d.ts +59 -0
- package/dist/providers/claude-code/hooks.d.ts.map +1 -0
- package/dist/providers/claude-code/hooks.js +77 -0
- package/dist/providers/claude-code/hooks.js.map +1 -0
- package/dist/providers/claude-code/index.d.ts +24 -0
- package/dist/providers/claude-code/index.d.ts.map +1 -0
- package/dist/providers/claude-code/index.js +26 -0
- package/dist/providers/claude-code/index.js.map +1 -0
- package/dist/providers/claude-code/install.d.ts +75 -0
- package/dist/providers/claude-code/install.d.ts.map +1 -0
- package/dist/providers/claude-code/install.js +237 -0
- package/dist/providers/claude-code/install.js.map +1 -0
- package/dist/providers/claude-code/paths.d.ts +24 -0
- package/dist/providers/claude-code/paths.d.ts.map +1 -0
- package/dist/providers/claude-code/paths.js +33 -0
- package/dist/providers/claude-code/paths.js.map +1 -0
- package/dist/providers/claude-code/spawn.d.ts +60 -0
- package/dist/providers/claude-code/spawn.d.ts.map +1 -0
- package/dist/providers/claude-code/spawn.js +160 -0
- package/dist/providers/claude-code/spawn.js.map +1 -0
- package/dist/providers/claude-code/statusline.d.ts +24 -0
- package/dist/providers/claude-code/statusline.d.ts.map +1 -0
- package/dist/providers/claude-code/statusline.js +85 -0
- package/dist/providers/claude-code/statusline.js.map +1 -0
- package/dist/providers/claude-code/task-sync.d.ts +27 -0
- package/dist/providers/claude-code/task-sync.d.ts.map +1 -0
- package/dist/providers/claude-code/task-sync.js +124 -0
- package/dist/providers/claude-code/task-sync.js.map +1 -0
- package/dist/providers/claude-code/transport.d.ts +14 -0
- package/dist/providers/claude-code/transport.d.ts.map +1 -0
- package/dist/providers/claude-code/transport.js +18 -0
- package/dist/providers/claude-code/transport.js.map +1 -0
- package/dist/providers/cursor/adapter.d.ts +62 -0
- package/dist/providers/cursor/adapter.d.ts.map +1 -0
- package/dist/providers/cursor/adapter.js +124 -0
- package/dist/providers/cursor/adapter.js.map +1 -0
- package/dist/providers/cursor/hooks.d.ts +48 -0
- package/dist/providers/cursor/hooks.d.ts.map +1 -0
- package/dist/providers/cursor/hooks.js +55 -0
- package/dist/providers/cursor/hooks.js.map +1 -0
- package/dist/providers/cursor/index.d.ts +19 -0
- package/dist/providers/cursor/index.d.ts.map +1 -0
- package/dist/providers/cursor/index.js +21 -0
- package/dist/providers/cursor/index.js.map +1 -0
- package/dist/providers/cursor/install.d.ts +94 -0
- package/dist/providers/cursor/install.d.ts.map +1 -0
- package/dist/providers/cursor/install.js +241 -0
- package/dist/providers/cursor/install.js.map +1 -0
- package/dist/providers/cursor/spawn.d.ts +50 -0
- package/dist/providers/cursor/spawn.d.ts.map +1 -0
- package/dist/providers/cursor/spawn.js +59 -0
- package/dist/providers/cursor/spawn.js.map +1 -0
- package/dist/providers/opencode/adapter.d.ts +67 -0
- package/dist/providers/opencode/adapter.d.ts.map +1 -0
- package/dist/providers/opencode/adapter.js +144 -0
- package/dist/providers/opencode/adapter.js.map +1 -0
- package/dist/providers/opencode/hooks.d.ts +66 -0
- package/dist/providers/opencode/hooks.d.ts.map +1 -0
- package/dist/providers/opencode/hooks.js +89 -0
- package/dist/providers/opencode/hooks.js.map +1 -0
- package/dist/providers/opencode/index.d.ts +20 -0
- package/dist/providers/opencode/index.d.ts.map +1 -0
- package/dist/providers/opencode/index.js +22 -0
- package/dist/providers/opencode/index.js.map +1 -0
- package/dist/providers/opencode/install.d.ts +65 -0
- package/dist/providers/opencode/install.d.ts.map +1 -0
- package/dist/providers/opencode/install.js +183 -0
- package/dist/providers/opencode/install.js.map +1 -0
- package/dist/providers/opencode/spawn.d.ts +72 -0
- package/dist/providers/opencode/spawn.d.ts.map +1 -0
- package/dist/providers/opencode/spawn.js +219 -0
- package/dist/providers/opencode/spawn.js.map +1 -0
- package/dist/registry.d.ts +36 -0
- package/dist/registry.d.ts.map +1 -0
- package/dist/registry.js +55 -0
- package/dist/registry.js.map +1 -0
- package/package.json +32 -0
- package/src/index.d.ts +27 -0
- package/src/index.d.ts.map +1 -0
- package/src/index.js +28 -0
- package/src/index.js.map +1 -0
- package/src/index.ts +37 -0
- package/src/providers/claude-code/__tests__/adapter.test.d.ts +7 -0
- package/src/providers/claude-code/__tests__/adapter.test.d.ts.map +1 -0
- package/src/providers/claude-code/__tests__/adapter.test.js +249 -0
- package/src/providers/claude-code/__tests__/adapter.test.js.map +1 -0
- package/src/providers/claude-code/__tests__/adapter.test.ts +291 -0
- package/src/providers/claude-code/adapter.d.ts +75 -0
- package/src/providers/claude-code/adapter.d.ts.map +1 -0
- package/src/providers/claude-code/adapter.js +154 -0
- package/src/providers/claude-code/adapter.js.map +1 -0
- package/src/providers/claude-code/adapter.ts +175 -0
- package/src/providers/claude-code/context-monitor.d.ts +24 -0
- package/src/providers/claude-code/context-monitor.d.ts.map +1 -0
- package/src/providers/claude-code/context-monitor.js +148 -0
- package/src/providers/claude-code/context-monitor.js.map +1 -0
- package/src/providers/claude-code/context-monitor.ts +175 -0
- package/src/providers/claude-code/hooks.d.ts +59 -0
- package/src/providers/claude-code/hooks.d.ts.map +1 -0
- package/src/providers/claude-code/hooks.js +77 -0
- package/src/providers/claude-code/hooks.js.map +1 -0
- package/src/providers/claude-code/hooks.ts +85 -0
- package/src/providers/claude-code/index.d.ts +24 -0
- package/src/providers/claude-code/index.d.ts.map +1 -0
- package/src/providers/claude-code/index.js +26 -0
- package/src/providers/claude-code/index.js.map +1 -0
- package/src/providers/claude-code/index.ts +33 -0
- package/src/providers/claude-code/install.d.ts +75 -0
- package/src/providers/claude-code/install.d.ts.map +1 -0
- package/src/providers/claude-code/install.js +237 -0
- package/src/providers/claude-code/install.js.map +1 -0
- package/src/providers/claude-code/install.ts +267 -0
- package/src/providers/claude-code/manifest.json +26 -0
- package/src/providers/claude-code/paths.d.ts +24 -0
- package/src/providers/claude-code/paths.d.ts.map +1 -0
- package/src/providers/claude-code/paths.js +33 -0
- package/src/providers/claude-code/paths.js.map +1 -0
- package/src/providers/claude-code/paths.ts +38 -0
- package/src/providers/claude-code/spawn.d.ts +60 -0
- package/src/providers/claude-code/spawn.d.ts.map +1 -0
- package/src/providers/claude-code/spawn.js +160 -0
- package/src/providers/claude-code/spawn.js.map +1 -0
- package/src/providers/claude-code/spawn.ts +178 -0
- package/src/providers/claude-code/statusline.d.ts +24 -0
- package/src/providers/claude-code/statusline.d.ts.map +1 -0
- package/src/providers/claude-code/statusline.js +85 -0
- package/src/providers/claude-code/statusline.js.map +1 -0
- package/src/providers/claude-code/statusline.ts +99 -0
- package/src/providers/claude-code/task-sync.d.ts +27 -0
- package/src/providers/claude-code/task-sync.d.ts.map +1 -0
- package/src/providers/claude-code/task-sync.js +124 -0
- package/src/providers/claude-code/task-sync.js.map +1 -0
- package/src/providers/claude-code/task-sync.ts +158 -0
- package/src/providers/claude-code/transport.d.ts +14 -0
- package/src/providers/claude-code/transport.d.ts.map +1 -0
- package/src/providers/claude-code/transport.js +18 -0
- package/src/providers/claude-code/transport.js.map +1 -0
- package/src/providers/claude-code/transport.ts +21 -0
- package/src/providers/cursor/__tests__/adapter.test.d.ts +7 -0
- package/src/providers/cursor/__tests__/adapter.test.d.ts.map +1 -0
- package/src/providers/cursor/__tests__/adapter.test.js +246 -0
- package/src/providers/cursor/__tests__/adapter.test.js.map +1 -0
- package/src/providers/cursor/__tests__/adapter.test.ts +291 -0
- package/src/providers/cursor/adapter.d.ts +62 -0
- package/src/providers/cursor/adapter.d.ts.map +1 -0
- package/src/providers/cursor/adapter.js +124 -0
- package/src/providers/cursor/adapter.js.map +1 -0
- package/src/providers/cursor/adapter.ts +145 -0
- package/src/providers/cursor/hooks.d.ts +48 -0
- package/src/providers/cursor/hooks.d.ts.map +1 -0
- package/src/providers/cursor/hooks.js +55 -0
- package/src/providers/cursor/hooks.js.map +1 -0
- package/src/providers/cursor/hooks.ts +61 -0
- package/src/providers/cursor/index.d.ts +19 -0
- package/src/providers/cursor/index.d.ts.map +1 -0
- package/src/providers/cursor/index.js +21 -0
- package/src/providers/cursor/index.js.map +1 -0
- package/src/providers/cursor/index.ts +24 -0
- package/src/providers/cursor/install.d.ts +94 -0
- package/src/providers/cursor/install.d.ts.map +1 -0
- package/src/providers/cursor/install.js +241 -0
- package/src/providers/cursor/install.js.map +1 -0
- package/src/providers/cursor/install.ts +271 -0
- package/src/providers/cursor/manifest.json +26 -0
- package/src/providers/cursor/spawn.d.ts +50 -0
- package/src/providers/cursor/spawn.d.ts.map +1 -0
- package/src/providers/cursor/spawn.js +59 -0
- package/src/providers/cursor/spawn.js.map +1 -0
- package/src/providers/cursor/spawn.ts +66 -0
- package/src/providers/opencode/__tests__/adapter.test.d.ts +7 -0
- package/src/providers/opencode/__tests__/adapter.test.d.ts.map +1 -0
- package/src/providers/opencode/__tests__/adapter.test.js +263 -0
- package/src/providers/opencode/__tests__/adapter.test.js.map +1 -0
- package/src/providers/opencode/__tests__/adapter.test.ts +309 -0
- package/src/providers/opencode/adapter.d.ts +67 -0
- package/src/providers/opencode/adapter.d.ts.map +1 -0
- package/src/providers/opencode/adapter.js +144 -0
- package/src/providers/opencode/adapter.js.map +1 -0
- package/src/providers/opencode/adapter.ts +165 -0
- package/src/providers/opencode/hooks.d.ts +66 -0
- package/src/providers/opencode/hooks.d.ts.map +1 -0
- package/src/providers/opencode/hooks.js +89 -0
- package/src/providers/opencode/hooks.js.map +1 -0
- package/src/providers/opencode/hooks.ts +97 -0
- package/src/providers/opencode/index.d.ts +20 -0
- package/src/providers/opencode/index.d.ts.map +1 -0
- package/src/providers/opencode/index.js +22 -0
- package/src/providers/opencode/index.js.map +1 -0
- package/src/providers/opencode/index.ts +25 -0
- package/src/providers/opencode/install.d.ts +65 -0
- package/src/providers/opencode/install.d.ts.map +1 -0
- package/src/providers/opencode/install.js +183 -0
- package/src/providers/opencode/install.js.map +1 -0
- package/src/providers/opencode/install.ts +206 -0
- package/src/providers/opencode/manifest.json +26 -0
- package/src/providers/opencode/spawn.d.ts +72 -0
- package/src/providers/opencode/spawn.d.ts.map +1 -0
- package/src/providers/opencode/spawn.js +219 -0
- package/src/providers/opencode/spawn.js.map +1 -0
- package/src/providers/opencode/spawn.ts +253 -0
- package/src/registry.d.ts +36 -0
- package/src/registry.d.ts.map +1 -0
- package/src/registry.js +55 -0
- package/src/registry.js.map +1 -0
- package/src/registry.ts +81 -0
|
@@ -0,0 +1,237 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Claude Code Install Provider
|
|
3
|
+
*
|
|
4
|
+
* Handles CLEO installation into Claude Code environments:
|
|
5
|
+
* - Registers CLEO MCP server in .mcp.json
|
|
6
|
+
* - Ensures CLAUDE.md has CLEO @-references
|
|
7
|
+
* - Manages plugin registration in ~/.claude/settings.json
|
|
8
|
+
*
|
|
9
|
+
* Migrated from src/core/install/claude-plugin.ts
|
|
10
|
+
*
|
|
11
|
+
* @task T5240
|
|
12
|
+
*/
|
|
13
|
+
import { existsSync, mkdirSync, readFileSync, writeFileSync } from 'node:fs';
|
|
14
|
+
import { homedir } from 'node:os';
|
|
15
|
+
import { join } from 'node:path';
|
|
16
|
+
/** Lines that should appear in CLAUDE.md to reference CLEO. */
|
|
17
|
+
const INSTRUCTION_REFERENCES = [
|
|
18
|
+
'@~/.cleo/templates/CLEO-INJECTION.md',
|
|
19
|
+
'@.cleo/memory-bridge.md',
|
|
20
|
+
];
|
|
21
|
+
/** MCP server registration key used in .mcp.json. */
|
|
22
|
+
const MCP_SERVER_KEY = 'cleo';
|
|
23
|
+
/**
|
|
24
|
+
* Install provider for Claude Code.
|
|
25
|
+
*
|
|
26
|
+
* Manages CLEO's integration with Claude Code by:
|
|
27
|
+
* 1. Registering the CLEO MCP server in the project's .mcp.json
|
|
28
|
+
* 2. Ensuring CLAUDE.md contains @-references to CLEO instruction files
|
|
29
|
+
* 3. Registering the brain observation plugin in ~/.claude/settings.json
|
|
30
|
+
*/
|
|
31
|
+
export class ClaudeCodeInstallProvider {
|
|
32
|
+
installedProjectDir = null;
|
|
33
|
+
/**
|
|
34
|
+
* Install CLEO into a Claude Code project.
|
|
35
|
+
*
|
|
36
|
+
* @param options - Installation options including project directory and MCP server path
|
|
37
|
+
* @returns Result describing what was installed
|
|
38
|
+
*/
|
|
39
|
+
async install(options) {
|
|
40
|
+
const { projectDir, mcpServerPath } = options;
|
|
41
|
+
const installedAt = new Date().toISOString();
|
|
42
|
+
let instructionFileUpdated = false;
|
|
43
|
+
let mcpRegistered = false;
|
|
44
|
+
const details = {};
|
|
45
|
+
// Step 1: Register MCP server in .mcp.json
|
|
46
|
+
if (mcpServerPath) {
|
|
47
|
+
mcpRegistered = this.registerMcpServer(projectDir, mcpServerPath);
|
|
48
|
+
if (mcpRegistered) {
|
|
49
|
+
details.mcpConfigPath = join(projectDir, '.mcp.json');
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
// Step 2: Ensure CLAUDE.md has @-references
|
|
53
|
+
instructionFileUpdated = this.updateInstructionFile(projectDir);
|
|
54
|
+
if (instructionFileUpdated) {
|
|
55
|
+
details.instructionFile = join(projectDir, 'CLAUDE.md');
|
|
56
|
+
}
|
|
57
|
+
// Step 3: Register plugin in ~/.claude/settings.json
|
|
58
|
+
const pluginResult = this.registerPlugin();
|
|
59
|
+
if (pluginResult) {
|
|
60
|
+
details.plugin = pluginResult;
|
|
61
|
+
}
|
|
62
|
+
this.installedProjectDir = projectDir;
|
|
63
|
+
return {
|
|
64
|
+
success: true,
|
|
65
|
+
installedAt,
|
|
66
|
+
instructionFileUpdated,
|
|
67
|
+
mcpRegistered,
|
|
68
|
+
details,
|
|
69
|
+
};
|
|
70
|
+
}
|
|
71
|
+
/**
|
|
72
|
+
* Uninstall CLEO from the current Claude Code project.
|
|
73
|
+
*
|
|
74
|
+
* Removes the MCP server registration from .mcp.json.
|
|
75
|
+
* Does not remove CLAUDE.md references (they are harmless if CLEO is not present).
|
|
76
|
+
*/
|
|
77
|
+
async uninstall() {
|
|
78
|
+
if (!this.installedProjectDir)
|
|
79
|
+
return;
|
|
80
|
+
const mcpPath = join(this.installedProjectDir, '.mcp.json');
|
|
81
|
+
if (existsSync(mcpPath)) {
|
|
82
|
+
try {
|
|
83
|
+
const raw = readFileSync(mcpPath, 'utf-8');
|
|
84
|
+
const config = JSON.parse(raw);
|
|
85
|
+
const mcpServers = config.mcpServers;
|
|
86
|
+
if (mcpServers && MCP_SERVER_KEY in mcpServers) {
|
|
87
|
+
delete mcpServers[MCP_SERVER_KEY];
|
|
88
|
+
writeFileSync(mcpPath, JSON.stringify(config, null, 2) + '\n', 'utf-8');
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
catch {
|
|
92
|
+
// Ignore errors during uninstall
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
this.installedProjectDir = null;
|
|
96
|
+
}
|
|
97
|
+
/**
|
|
98
|
+
* Check whether CLEO is installed in the current environment.
|
|
99
|
+
*
|
|
100
|
+
* Checks for:
|
|
101
|
+
* 1. MCP server registered in .mcp.json
|
|
102
|
+
* 2. Plugin enabled in ~/.claude/settings.json
|
|
103
|
+
*
|
|
104
|
+
* Returns true if either condition is met (partial install counts).
|
|
105
|
+
*/
|
|
106
|
+
async isInstalled() {
|
|
107
|
+
// Check ~/.claude/settings.json for plugin registration
|
|
108
|
+
const settingsPath = join(homedir(), '.claude', 'settings.json');
|
|
109
|
+
if (existsSync(settingsPath)) {
|
|
110
|
+
try {
|
|
111
|
+
const settings = JSON.parse(readFileSync(settingsPath, 'utf-8'));
|
|
112
|
+
const plugins = settings.enabledPlugins;
|
|
113
|
+
if (plugins && plugins['cleo@cleocode'] === true) {
|
|
114
|
+
return true;
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
catch {
|
|
118
|
+
// Fall through
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
// Check current directory for .mcp.json with cleo server
|
|
122
|
+
const mcpPath = join(process.cwd(), '.mcp.json');
|
|
123
|
+
if (existsSync(mcpPath)) {
|
|
124
|
+
try {
|
|
125
|
+
const config = JSON.parse(readFileSync(mcpPath, 'utf-8'));
|
|
126
|
+
const mcpServers = config.mcpServers;
|
|
127
|
+
if (mcpServers && MCP_SERVER_KEY in mcpServers) {
|
|
128
|
+
return true;
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
catch {
|
|
132
|
+
// Fall through
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
return false;
|
|
136
|
+
}
|
|
137
|
+
/**
|
|
138
|
+
* Ensure CLAUDE.md contains @-references to CLEO instruction files.
|
|
139
|
+
*
|
|
140
|
+
* Creates CLAUDE.md if it does not exist. Appends any missing references.
|
|
141
|
+
*
|
|
142
|
+
* @param projectDir - Project root directory
|
|
143
|
+
*/
|
|
144
|
+
async ensureInstructionReferences(projectDir) {
|
|
145
|
+
this.updateInstructionFile(projectDir);
|
|
146
|
+
}
|
|
147
|
+
/**
|
|
148
|
+
* Register the CLEO MCP server in .mcp.json.
|
|
149
|
+
*
|
|
150
|
+
* @returns true if registration was performed or updated
|
|
151
|
+
*/
|
|
152
|
+
registerMcpServer(projectDir, mcpServerPath) {
|
|
153
|
+
const mcpPath = join(projectDir, '.mcp.json');
|
|
154
|
+
let config = {};
|
|
155
|
+
if (existsSync(mcpPath)) {
|
|
156
|
+
try {
|
|
157
|
+
config = JSON.parse(readFileSync(mcpPath, 'utf-8'));
|
|
158
|
+
}
|
|
159
|
+
catch {
|
|
160
|
+
// Start fresh on parse error
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
if (!config.mcpServers || typeof config.mcpServers !== 'object') {
|
|
164
|
+
config.mcpServers = {};
|
|
165
|
+
}
|
|
166
|
+
const mcpServers = config.mcpServers;
|
|
167
|
+
mcpServers[MCP_SERVER_KEY] = {
|
|
168
|
+
command: 'node',
|
|
169
|
+
args: [mcpServerPath],
|
|
170
|
+
};
|
|
171
|
+
writeFileSync(mcpPath, JSON.stringify(config, null, 2) + '\n', 'utf-8');
|
|
172
|
+
return true;
|
|
173
|
+
}
|
|
174
|
+
/**
|
|
175
|
+
* Update CLAUDE.md with CLEO @-references.
|
|
176
|
+
*
|
|
177
|
+
* @returns true if the file was created or modified
|
|
178
|
+
*/
|
|
179
|
+
updateInstructionFile(projectDir) {
|
|
180
|
+
const claudeMdPath = join(projectDir, 'CLAUDE.md');
|
|
181
|
+
let content = '';
|
|
182
|
+
let existed = false;
|
|
183
|
+
if (existsSync(claudeMdPath)) {
|
|
184
|
+
content = readFileSync(claudeMdPath, 'utf-8');
|
|
185
|
+
existed = true;
|
|
186
|
+
}
|
|
187
|
+
const missingRefs = INSTRUCTION_REFERENCES.filter((ref) => !content.includes(ref));
|
|
188
|
+
if (missingRefs.length === 0) {
|
|
189
|
+
return false;
|
|
190
|
+
}
|
|
191
|
+
const refsBlock = missingRefs.join('\n');
|
|
192
|
+
if (existed) {
|
|
193
|
+
// Append missing references
|
|
194
|
+
const separator = content.endsWith('\n') ? '' : '\n';
|
|
195
|
+
content = content + separator + refsBlock + '\n';
|
|
196
|
+
}
|
|
197
|
+
else {
|
|
198
|
+
// Create new CLAUDE.md with references
|
|
199
|
+
content = refsBlock + '\n';
|
|
200
|
+
}
|
|
201
|
+
writeFileSync(claudeMdPath, content, 'utf-8');
|
|
202
|
+
return true;
|
|
203
|
+
}
|
|
204
|
+
/**
|
|
205
|
+
* Register the CLEO brain plugin in ~/.claude/settings.json.
|
|
206
|
+
*
|
|
207
|
+
* @returns Description of what was registered, or null if no change needed
|
|
208
|
+
*/
|
|
209
|
+
registerPlugin() {
|
|
210
|
+
const home = homedir();
|
|
211
|
+
const settingsPath = join(home, '.claude', 'settings.json');
|
|
212
|
+
let settings = {};
|
|
213
|
+
if (existsSync(settingsPath)) {
|
|
214
|
+
try {
|
|
215
|
+
settings = JSON.parse(readFileSync(settingsPath, 'utf-8'));
|
|
216
|
+
}
|
|
217
|
+
catch {
|
|
218
|
+
// Start fresh
|
|
219
|
+
}
|
|
220
|
+
}
|
|
221
|
+
const enabledPlugins = settings.enabledPlugins ?? {};
|
|
222
|
+
const pluginKey = 'cleo@cleocode';
|
|
223
|
+
if (enabledPlugins[pluginKey] === true) {
|
|
224
|
+
return null;
|
|
225
|
+
}
|
|
226
|
+
// Disable old claude-mem if present
|
|
227
|
+
if (enabledPlugins['claude-mem@thedotmack'] === true) {
|
|
228
|
+
enabledPlugins['claude-mem@thedotmack'] = false;
|
|
229
|
+
}
|
|
230
|
+
enabledPlugins[pluginKey] = true;
|
|
231
|
+
settings.enabledPlugins = enabledPlugins;
|
|
232
|
+
mkdirSync(join(home, '.claude'), { recursive: true });
|
|
233
|
+
writeFileSync(settingsPath, JSON.stringify(settings, null, 2) + '\n', 'utf-8');
|
|
234
|
+
return `Enabled ${pluginKey} in ~/.claude/settings.json`;
|
|
235
|
+
}
|
|
236
|
+
}
|
|
237
|
+
//# 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,267 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Claude Code Install Provider
|
|
3
|
+
*
|
|
4
|
+
* Handles CLEO installation into Claude Code environments:
|
|
5
|
+
* - Registers CLEO MCP server in .mcp.json
|
|
6
|
+
* - Ensures CLAUDE.md has CLEO @-references
|
|
7
|
+
* - Manages plugin registration in ~/.claude/settings.json
|
|
8
|
+
*
|
|
9
|
+
* Migrated from src/core/install/claude-plugin.ts
|
|
10
|
+
*
|
|
11
|
+
* @task T5240
|
|
12
|
+
*/
|
|
13
|
+
|
|
14
|
+
import { existsSync, mkdirSync, readFileSync, writeFileSync } from 'node:fs';
|
|
15
|
+
import { homedir } from 'node:os';
|
|
16
|
+
import { join } from 'node:path';
|
|
17
|
+
import type { AdapterInstallProvider, InstallOptions, InstallResult } from '@cleocode/contracts';
|
|
18
|
+
|
|
19
|
+
/** Lines that should appear in CLAUDE.md to reference CLEO. */
|
|
20
|
+
const INSTRUCTION_REFERENCES = [
|
|
21
|
+
'@~/.cleo/templates/CLEO-INJECTION.md',
|
|
22
|
+
'@.cleo/memory-bridge.md',
|
|
23
|
+
];
|
|
24
|
+
|
|
25
|
+
/** MCP server registration key used in .mcp.json. */
|
|
26
|
+
const MCP_SERVER_KEY = 'cleo';
|
|
27
|
+
|
|
28
|
+
/**
|
|
29
|
+
* Install provider for Claude Code.
|
|
30
|
+
*
|
|
31
|
+
* Manages CLEO's integration with Claude Code by:
|
|
32
|
+
* 1. Registering the CLEO MCP server in the project's .mcp.json
|
|
33
|
+
* 2. Ensuring CLAUDE.md contains @-references to CLEO instruction files
|
|
34
|
+
* 3. Registering the brain observation plugin in ~/.claude/settings.json
|
|
35
|
+
*/
|
|
36
|
+
export class ClaudeCodeInstallProvider implements AdapterInstallProvider {
|
|
37
|
+
private installedProjectDir: string | null = null;
|
|
38
|
+
|
|
39
|
+
/**
|
|
40
|
+
* Install CLEO into a Claude Code project.
|
|
41
|
+
*
|
|
42
|
+
* @param options - Installation options including project directory and MCP server path
|
|
43
|
+
* @returns Result describing what was installed
|
|
44
|
+
*/
|
|
45
|
+
async install(options: InstallOptions): Promise<InstallResult> {
|
|
46
|
+
const { projectDir, mcpServerPath } = options;
|
|
47
|
+
const installedAt = new Date().toISOString();
|
|
48
|
+
let instructionFileUpdated = false;
|
|
49
|
+
let mcpRegistered = false;
|
|
50
|
+
const details: Record<string, unknown> = {};
|
|
51
|
+
|
|
52
|
+
// Step 1: Register MCP server in .mcp.json
|
|
53
|
+
if (mcpServerPath) {
|
|
54
|
+
mcpRegistered = this.registerMcpServer(projectDir, mcpServerPath);
|
|
55
|
+
if (mcpRegistered) {
|
|
56
|
+
details.mcpConfigPath = join(projectDir, '.mcp.json');
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
// Step 2: Ensure CLAUDE.md has @-references
|
|
61
|
+
instructionFileUpdated = this.updateInstructionFile(projectDir);
|
|
62
|
+
if (instructionFileUpdated) {
|
|
63
|
+
details.instructionFile = join(projectDir, 'CLAUDE.md');
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
// Step 3: Register plugin in ~/.claude/settings.json
|
|
67
|
+
const pluginResult = this.registerPlugin();
|
|
68
|
+
if (pluginResult) {
|
|
69
|
+
details.plugin = pluginResult;
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
this.installedProjectDir = projectDir;
|
|
73
|
+
|
|
74
|
+
return {
|
|
75
|
+
success: true,
|
|
76
|
+
installedAt,
|
|
77
|
+
instructionFileUpdated,
|
|
78
|
+
mcpRegistered,
|
|
79
|
+
details,
|
|
80
|
+
};
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
/**
|
|
84
|
+
* Uninstall CLEO from the current Claude Code project.
|
|
85
|
+
*
|
|
86
|
+
* Removes the MCP server registration from .mcp.json.
|
|
87
|
+
* Does not remove CLAUDE.md references (they are harmless if CLEO is not present).
|
|
88
|
+
*/
|
|
89
|
+
async uninstall(): Promise<void> {
|
|
90
|
+
if (!this.installedProjectDir) return;
|
|
91
|
+
|
|
92
|
+
const mcpPath = join(this.installedProjectDir, '.mcp.json');
|
|
93
|
+
if (existsSync(mcpPath)) {
|
|
94
|
+
try {
|
|
95
|
+
const raw = readFileSync(mcpPath, 'utf-8');
|
|
96
|
+
const config = JSON.parse(raw) as Record<string, unknown>;
|
|
97
|
+
const mcpServers = config.mcpServers as Record<string, unknown> | undefined;
|
|
98
|
+
if (mcpServers && MCP_SERVER_KEY in mcpServers) {
|
|
99
|
+
delete mcpServers[MCP_SERVER_KEY];
|
|
100
|
+
writeFileSync(mcpPath, JSON.stringify(config, null, 2) + '\n', 'utf-8');
|
|
101
|
+
}
|
|
102
|
+
} catch {
|
|
103
|
+
// Ignore errors during uninstall
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
this.installedProjectDir = null;
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
/**
|
|
111
|
+
* Check whether CLEO is installed in the current environment.
|
|
112
|
+
*
|
|
113
|
+
* Checks for:
|
|
114
|
+
* 1. MCP server registered in .mcp.json
|
|
115
|
+
* 2. Plugin enabled in ~/.claude/settings.json
|
|
116
|
+
*
|
|
117
|
+
* Returns true if either condition is met (partial install counts).
|
|
118
|
+
*/
|
|
119
|
+
async isInstalled(): Promise<boolean> {
|
|
120
|
+
// Check ~/.claude/settings.json for plugin registration
|
|
121
|
+
const settingsPath = join(homedir(), '.claude', 'settings.json');
|
|
122
|
+
if (existsSync(settingsPath)) {
|
|
123
|
+
try {
|
|
124
|
+
const settings = JSON.parse(readFileSync(settingsPath, 'utf-8'));
|
|
125
|
+
const plugins = settings.enabledPlugins as Record<string, boolean> | undefined;
|
|
126
|
+
if (plugins && plugins['cleo@cleocode'] === true) {
|
|
127
|
+
return true;
|
|
128
|
+
}
|
|
129
|
+
} catch {
|
|
130
|
+
// Fall through
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
// Check current directory for .mcp.json with cleo server
|
|
135
|
+
const mcpPath = join(process.cwd(), '.mcp.json');
|
|
136
|
+
if (existsSync(mcpPath)) {
|
|
137
|
+
try {
|
|
138
|
+
const config = JSON.parse(readFileSync(mcpPath, 'utf-8'));
|
|
139
|
+
const mcpServers = config.mcpServers as Record<string, unknown> | undefined;
|
|
140
|
+
if (mcpServers && MCP_SERVER_KEY in mcpServers) {
|
|
141
|
+
return true;
|
|
142
|
+
}
|
|
143
|
+
} catch {
|
|
144
|
+
// Fall through
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
return false;
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
/**
|
|
152
|
+
* Ensure CLAUDE.md contains @-references to CLEO instruction files.
|
|
153
|
+
*
|
|
154
|
+
* Creates CLAUDE.md if it does not exist. Appends any missing references.
|
|
155
|
+
*
|
|
156
|
+
* @param projectDir - Project root directory
|
|
157
|
+
*/
|
|
158
|
+
async ensureInstructionReferences(projectDir: string): Promise<void> {
|
|
159
|
+
this.updateInstructionFile(projectDir);
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
/**
|
|
163
|
+
* Register the CLEO MCP server in .mcp.json.
|
|
164
|
+
*
|
|
165
|
+
* @returns true if registration was performed or updated
|
|
166
|
+
*/
|
|
167
|
+
private registerMcpServer(projectDir: string, mcpServerPath: string): boolean {
|
|
168
|
+
const mcpPath = join(projectDir, '.mcp.json');
|
|
169
|
+
let config: Record<string, unknown> = {};
|
|
170
|
+
|
|
171
|
+
if (existsSync(mcpPath)) {
|
|
172
|
+
try {
|
|
173
|
+
config = JSON.parse(readFileSync(mcpPath, 'utf-8'));
|
|
174
|
+
} catch {
|
|
175
|
+
// Start fresh on parse error
|
|
176
|
+
}
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
if (!config.mcpServers || typeof config.mcpServers !== 'object') {
|
|
180
|
+
config.mcpServers = {};
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
const mcpServers = config.mcpServers as Record<string, unknown>;
|
|
184
|
+
mcpServers[MCP_SERVER_KEY] = {
|
|
185
|
+
command: 'node',
|
|
186
|
+
args: [mcpServerPath],
|
|
187
|
+
};
|
|
188
|
+
|
|
189
|
+
writeFileSync(mcpPath, JSON.stringify(config, null, 2) + '\n', 'utf-8');
|
|
190
|
+
return true;
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
/**
|
|
194
|
+
* Update CLAUDE.md with CLEO @-references.
|
|
195
|
+
*
|
|
196
|
+
* @returns true if the file was created or modified
|
|
197
|
+
*/
|
|
198
|
+
private updateInstructionFile(projectDir: string): boolean {
|
|
199
|
+
const claudeMdPath = join(projectDir, 'CLAUDE.md');
|
|
200
|
+
let content = '';
|
|
201
|
+
let existed = false;
|
|
202
|
+
|
|
203
|
+
if (existsSync(claudeMdPath)) {
|
|
204
|
+
content = readFileSync(claudeMdPath, 'utf-8');
|
|
205
|
+
existed = true;
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
const missingRefs = INSTRUCTION_REFERENCES.filter((ref) => !content.includes(ref));
|
|
209
|
+
|
|
210
|
+
if (missingRefs.length === 0) {
|
|
211
|
+
return false;
|
|
212
|
+
}
|
|
213
|
+
|
|
214
|
+
const refsBlock = missingRefs.join('\n');
|
|
215
|
+
|
|
216
|
+
if (existed) {
|
|
217
|
+
// Append missing references
|
|
218
|
+
const separator = content.endsWith('\n') ? '' : '\n';
|
|
219
|
+
content = content + separator + refsBlock + '\n';
|
|
220
|
+
} else {
|
|
221
|
+
// Create new CLAUDE.md with references
|
|
222
|
+
content = refsBlock + '\n';
|
|
223
|
+
}
|
|
224
|
+
|
|
225
|
+
writeFileSync(claudeMdPath, content, 'utf-8');
|
|
226
|
+
return true;
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
/**
|
|
230
|
+
* Register the CLEO brain plugin in ~/.claude/settings.json.
|
|
231
|
+
*
|
|
232
|
+
* @returns Description of what was registered, or null if no change needed
|
|
233
|
+
*/
|
|
234
|
+
private registerPlugin(): string | null {
|
|
235
|
+
const home = homedir();
|
|
236
|
+
const settingsPath = join(home, '.claude', 'settings.json');
|
|
237
|
+
|
|
238
|
+
let settings: Record<string, unknown> = {};
|
|
239
|
+
if (existsSync(settingsPath)) {
|
|
240
|
+
try {
|
|
241
|
+
settings = JSON.parse(readFileSync(settingsPath, 'utf-8'));
|
|
242
|
+
} catch {
|
|
243
|
+
// Start fresh
|
|
244
|
+
}
|
|
245
|
+
}
|
|
246
|
+
|
|
247
|
+
const enabledPlugins = (settings.enabledPlugins as Record<string, boolean>) ?? {};
|
|
248
|
+
const pluginKey = 'cleo@cleocode';
|
|
249
|
+
|
|
250
|
+
if (enabledPlugins[pluginKey] === true) {
|
|
251
|
+
return null;
|
|
252
|
+
}
|
|
253
|
+
|
|
254
|
+
// Disable old claude-mem if present
|
|
255
|
+
if (enabledPlugins['claude-mem@thedotmack'] === true) {
|
|
256
|
+
enabledPlugins['claude-mem@thedotmack'] = false;
|
|
257
|
+
}
|
|
258
|
+
|
|
259
|
+
enabledPlugins[pluginKey] = true;
|
|
260
|
+
settings.enabledPlugins = enabledPlugins;
|
|
261
|
+
|
|
262
|
+
mkdirSync(join(home, '.claude'), { recursive: true });
|
|
263
|
+
writeFileSync(settingsPath, JSON.stringify(settings, null, 2) + '\n', 'utf-8');
|
|
264
|
+
|
|
265
|
+
return `Enabled ${pluginKey} in ~/.claude/settings.json`;
|
|
266
|
+
}
|
|
267
|
+
}
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
{
|
|
2
|
+
"id": "claude-code",
|
|
3
|
+
"name": "Claude Code Adapter",
|
|
4
|
+
"version": "1.0.0",
|
|
5
|
+
"description": "CLEO adapter for Anthropic Claude Code CLI",
|
|
6
|
+
"provider": "claude-code",
|
|
7
|
+
"entryPoint": "src/index.ts",
|
|
8
|
+
"capabilities": {
|
|
9
|
+
"supportsHooks": true,
|
|
10
|
+
"supportedHookEvents": ["onSessionStart", "onSessionEnd", "onToolStart", "onToolComplete", "onError"],
|
|
11
|
+
"supportsSpawn": true,
|
|
12
|
+
"supportsInstall": true,
|
|
13
|
+
"supportsMcp": true,
|
|
14
|
+
"supportsInstructionFiles": true,
|
|
15
|
+
"instructionFilePattern": "CLAUDE.md",
|
|
16
|
+
"supportsContextMonitor": true,
|
|
17
|
+
"supportsStatusline": true,
|
|
18
|
+
"supportsProviderPaths": true,
|
|
19
|
+
"supportsTransport": true
|
|
20
|
+
},
|
|
21
|
+
"detectionPatterns": [
|
|
22
|
+
{ "type": "env", "pattern": "CLAUDE_CODE_ENTRYPOINT", "description": "Claude Code sets this env var" },
|
|
23
|
+
{ "type": "file", "pattern": ".claude/settings.json", "description": "Claude Code config directory" },
|
|
24
|
+
{ "type": "cli", "pattern": "claude", "description": "Claude CLI available in PATH" }
|
|
25
|
+
]
|
|
26
|
+
}
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Claude Code path provider.
|
|
3
|
+
*
|
|
4
|
+
* Implements AdapterPathProvider with Claude Code-specific directory locations.
|
|
5
|
+
*
|
|
6
|
+
* @task T5240
|
|
7
|
+
*/
|
|
8
|
+
import type { AdapterPathProvider } from '@cleocode/contracts';
|
|
9
|
+
/**
|
|
10
|
+
* Path provider for Anthropic Claude Code CLI.
|
|
11
|
+
*
|
|
12
|
+
* Resolves Claude Code's standard directory layout:
|
|
13
|
+
* - Config dir: ~/.claude (or CLAUDE_HOME)
|
|
14
|
+
* - Settings: ~/.claude/settings.json (or CLAUDE_SETTINGS)
|
|
15
|
+
* - Agents: ~/.claude/agents
|
|
16
|
+
* - Memory DB: ~/.claude-mem/claude-mem.db (or CLAUDE_MEM_DB)
|
|
17
|
+
*/
|
|
18
|
+
export declare class ClaudeCodePathProvider implements AdapterPathProvider {
|
|
19
|
+
getProviderDir(): string;
|
|
20
|
+
getSettingsPath(): string | null;
|
|
21
|
+
getAgentInstallDir(): string | null;
|
|
22
|
+
getMemoryDbPath(): string | null;
|
|
23
|
+
}
|
|
24
|
+
//# sourceMappingURL=paths.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"paths.d.ts","sourceRoot":"","sources":["paths.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAIH,OAAO,KAAK,EAAE,mBAAmB,EAAE,MAAM,qBAAqB,CAAC;AAE/D;;;;;;;;GAQG;AACH,qBAAa,sBAAuB,YAAW,mBAAmB;IAChE,cAAc,IAAI,MAAM;IAIxB,eAAe,IAAI,MAAM,GAAG,IAAI;IAIhC,kBAAkB,IAAI,MAAM,GAAG,IAAI;IAInC,eAAe,IAAI,MAAM,GAAG,IAAI;CAGjC"}
|
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Claude Code path provider.
|
|
3
|
+
*
|
|
4
|
+
* Implements AdapterPathProvider with Claude Code-specific directory locations.
|
|
5
|
+
*
|
|
6
|
+
* @task T5240
|
|
7
|
+
*/
|
|
8
|
+
import { homedir } from 'node:os';
|
|
9
|
+
import { join } from 'node:path';
|
|
10
|
+
/**
|
|
11
|
+
* Path provider for Anthropic Claude Code CLI.
|
|
12
|
+
*
|
|
13
|
+
* Resolves Claude Code's standard directory layout:
|
|
14
|
+
* - Config dir: ~/.claude (or CLAUDE_HOME)
|
|
15
|
+
* - Settings: ~/.claude/settings.json (or CLAUDE_SETTINGS)
|
|
16
|
+
* - Agents: ~/.claude/agents
|
|
17
|
+
* - Memory DB: ~/.claude-mem/claude-mem.db (or CLAUDE_MEM_DB)
|
|
18
|
+
*/
|
|
19
|
+
export class ClaudeCodePathProvider {
|
|
20
|
+
getProviderDir() {
|
|
21
|
+
return process.env['CLAUDE_HOME'] ?? join(homedir(), '.claude');
|
|
22
|
+
}
|
|
23
|
+
getSettingsPath() {
|
|
24
|
+
return process.env['CLAUDE_SETTINGS'] ?? join(this.getProviderDir(), 'settings.json');
|
|
25
|
+
}
|
|
26
|
+
getAgentInstallDir() {
|
|
27
|
+
return join(this.getProviderDir(), 'agents');
|
|
28
|
+
}
|
|
29
|
+
getMemoryDbPath() {
|
|
30
|
+
return process.env['CLAUDE_MEM_DB'] ?? join(homedir(), '.claude-mem', 'claude-mem.db');
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
//# sourceMappingURL=paths.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"paths.js","sourceRoot":"","sources":["paths.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAEH,OAAO,EAAE,OAAO,EAAE,MAAM,SAAS,CAAC;AAClC,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AAGjC;;;;;;;;GAQG;AACH,MAAM,OAAO,sBAAsB;IACjC,cAAc;QACZ,OAAO,OAAO,CAAC,GAAG,CAAC,aAAa,CAAC,IAAI,IAAI,CAAC,OAAO,EAAE,EAAE,SAAS,CAAC,CAAC;IAClE,CAAC;IAED,eAAe;QACb,OAAO,OAAO,CAAC,GAAG,CAAC,iBAAiB,CAAC,IAAI,IAAI,CAAC,IAAI,CAAC,cAAc,EAAE,EAAE,eAAe,CAAC,CAAC;IACxF,CAAC;IAED,kBAAkB;QAChB,OAAO,IAAI,CAAC,IAAI,CAAC,cAAc,EAAE,EAAE,QAAQ,CAAC,CAAC;IAC/C,CAAC;IAED,eAAe;QACb,OAAO,OAAO,CAAC,GAAG,CAAC,eAAe,CAAC,IAAI,IAAI,CAAC,OAAO,EAAE,EAAE,aAAa,EAAE,eAAe,CAAC,CAAC;IACzF,CAAC;CACF"}
|
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Claude Code path provider.
|
|
3
|
+
*
|
|
4
|
+
* Implements AdapterPathProvider with Claude Code-specific directory locations.
|
|
5
|
+
*
|
|
6
|
+
* @task T5240
|
|
7
|
+
*/
|
|
8
|
+
|
|
9
|
+
import { homedir } from 'node:os';
|
|
10
|
+
import { join } from 'node:path';
|
|
11
|
+
import type { AdapterPathProvider } from '@cleocode/contracts';
|
|
12
|
+
|
|
13
|
+
/**
|
|
14
|
+
* Path provider for Anthropic Claude Code CLI.
|
|
15
|
+
*
|
|
16
|
+
* Resolves Claude Code's standard directory layout:
|
|
17
|
+
* - Config dir: ~/.claude (or CLAUDE_HOME)
|
|
18
|
+
* - Settings: ~/.claude/settings.json (or CLAUDE_SETTINGS)
|
|
19
|
+
* - Agents: ~/.claude/agents
|
|
20
|
+
* - Memory DB: ~/.claude-mem/claude-mem.db (or CLAUDE_MEM_DB)
|
|
21
|
+
*/
|
|
22
|
+
export class ClaudeCodePathProvider implements AdapterPathProvider {
|
|
23
|
+
getProviderDir(): string {
|
|
24
|
+
return process.env['CLAUDE_HOME'] ?? join(homedir(), '.claude');
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
getSettingsPath(): string | null {
|
|
28
|
+
return process.env['CLAUDE_SETTINGS'] ?? join(this.getProviderDir(), 'settings.json');
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
getAgentInstallDir(): string | null {
|
|
32
|
+
return join(this.getProviderDir(), 'agents');
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
getMemoryDbPath(): string | null {
|
|
36
|
+
return process.env['CLAUDE_MEM_DB'] ?? join(homedir(), '.claude-mem', 'claude-mem.db');
|
|
37
|
+
}
|
|
38
|
+
}
|