@cleocode/adapters 2026.4.31 → 2026.4.35

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 (99) hide show
  1. package/dist/index.js +641 -5
  2. package/dist/index.js.map +4 -4
  3. package/dist/providers/claude-code/adapter.js +184 -0
  4. package/dist/providers/claude-code/adapter.js.map +1 -0
  5. package/dist/providers/claude-code/context-monitor.js +159 -0
  6. package/dist/providers/claude-code/context-monitor.js.map +1 -0
  7. package/dist/providers/claude-code/hooks.js +286 -0
  8. package/dist/providers/claude-code/hooks.js.map +1 -0
  9. package/dist/providers/claude-code/index.js +41 -0
  10. package/dist/providers/claude-code/index.js.map +1 -0
  11. package/dist/providers/claude-code/install.js +199 -0
  12. package/dist/providers/claude-code/install.js.map +1 -0
  13. package/dist/providers/claude-code/paths.js +41 -0
  14. package/dist/providers/claude-code/paths.js.map +1 -0
  15. package/dist/providers/claude-code/spawn.js +171 -0
  16. package/dist/providers/claude-code/spawn.js.map +1 -0
  17. package/dist/providers/claude-code/statusline.js +130 -0
  18. package/dist/providers/claude-code/statusline.js.map +1 -0
  19. package/dist/providers/claude-code/task-sync.js +119 -0
  20. package/dist/providers/claude-code/task-sync.js.map +1 -0
  21. package/dist/providers/claude-code/transport.js +29 -0
  22. package/dist/providers/claude-code/transport.js.map +1 -0
  23. package/dist/providers/codex/adapter.js +146 -0
  24. package/dist/providers/codex/adapter.js.map +1 -0
  25. package/dist/providers/codex/hooks.js +113 -0
  26. package/dist/providers/codex/hooks.js.map +1 -0
  27. package/dist/providers/codex/index.js +39 -0
  28. package/dist/providers/codex/index.js.map +1 -0
  29. package/dist/providers/codex/install.js +124 -0
  30. package/dist/providers/codex/install.js.map +1 -0
  31. package/dist/providers/cursor/adapter.js +151 -0
  32. package/dist/providers/cursor/adapter.js.map +1 -0
  33. package/dist/providers/cursor/hooks.js +208 -0
  34. package/dist/providers/cursor/hooks.js.map +1 -0
  35. package/dist/providers/cursor/index.js +36 -0
  36. package/dist/providers/cursor/index.js.map +1 -0
  37. package/dist/providers/cursor/install.js +180 -0
  38. package/dist/providers/cursor/install.js.map +1 -0
  39. package/dist/providers/cursor/spawn.js +59 -0
  40. package/dist/providers/cursor/spawn.js.map +1 -0
  41. package/dist/providers/gemini-cli/adapter.js +158 -0
  42. package/dist/providers/gemini-cli/adapter.js.map +1 -0
  43. package/dist/providers/gemini-cli/hooks.js +128 -0
  44. package/dist/providers/gemini-cli/hooks.js.map +1 -0
  45. package/dist/providers/gemini-cli/index.js +39 -0
  46. package/dist/providers/gemini-cli/index.js.map +1 -0
  47. package/dist/providers/gemini-cli/install.js +124 -0
  48. package/dist/providers/gemini-cli/install.js.map +1 -0
  49. package/dist/providers/kimi/adapter.js +145 -0
  50. package/dist/providers/kimi/adapter.js.map +1 -0
  51. package/dist/providers/kimi/hooks.js +79 -0
  52. package/dist/providers/kimi/hooks.js.map +1 -0
  53. package/dist/providers/kimi/index.js +39 -0
  54. package/dist/providers/kimi/index.js.map +1 -0
  55. package/dist/providers/kimi/install.js +124 -0
  56. package/dist/providers/kimi/install.js.map +1 -0
  57. package/dist/providers/opencode/adapter.js +166 -0
  58. package/dist/providers/opencode/adapter.js.map +1 -0
  59. package/dist/providers/opencode/hooks.js +206 -0
  60. package/dist/providers/opencode/hooks.js.map +1 -0
  61. package/dist/providers/opencode/index.js +37 -0
  62. package/dist/providers/opencode/index.js.map +1 -0
  63. package/dist/providers/opencode/install.js +115 -0
  64. package/dist/providers/opencode/install.js.map +1 -0
  65. package/dist/providers/opencode/spawn.js +241 -0
  66. package/dist/providers/opencode/spawn.js.map +1 -0
  67. package/dist/providers/pi/adapter.d.ts +102 -0
  68. package/dist/providers/pi/adapter.d.ts.map +1 -0
  69. package/dist/providers/pi/adapter.js +220 -0
  70. package/dist/providers/pi/adapter.js.map +1 -0
  71. package/dist/providers/pi/hooks.d.ts +149 -0
  72. package/dist/providers/pi/hooks.d.ts.map +1 -0
  73. package/dist/providers/pi/hooks.js +223 -0
  74. package/dist/providers/pi/hooks.js.map +1 -0
  75. package/dist/providers/pi/index.d.ts +36 -0
  76. package/dist/providers/pi/index.d.ts.map +1 -0
  77. package/dist/providers/pi/index.js +38 -0
  78. package/dist/providers/pi/index.js.map +1 -0
  79. package/dist/providers/pi/install.d.ts +68 -0
  80. package/dist/providers/pi/install.d.ts.map +1 -0
  81. package/dist/providers/pi/install.js +175 -0
  82. package/dist/providers/pi/install.js.map +1 -0
  83. package/dist/providers/pi/spawn.d.ts +68 -0
  84. package/dist/providers/pi/spawn.d.ts.map +1 -0
  85. package/dist/providers/pi/spawn.js +187 -0
  86. package/dist/providers/pi/spawn.js.map +1 -0
  87. package/dist/providers/shared/transcript-reader.js +124 -0
  88. package/dist/providers/shared/transcript-reader.js.map +1 -0
  89. package/dist/registry.d.ts.map +1 -1
  90. package/dist/registry.js +92 -0
  91. package/dist/registry.js.map +1 -0
  92. package/package.json +3 -3
  93. package/src/providers/pi/adapter.ts +240 -0
  94. package/src/providers/pi/hooks.ts +232 -0
  95. package/src/providers/pi/index.ts +41 -0
  96. package/src/providers/pi/install.ts +189 -0
  97. package/src/providers/pi/manifest.json +40 -0
  98. package/src/providers/pi/spawn.ts +207 -0
  99. package/src/registry.ts +6 -1
@@ -0,0 +1,184 @@
1
+ /**
2
+ * Claude Code Adapter
3
+ *
4
+ * Main CLEOProviderAdapter implementation for Anthropic's Claude Code CLI.
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 { homedir } from 'node:os';
12
+ import { join } from 'node:path';
13
+ import { promisify } from 'node:util';
14
+ import { ClaudeCodeContextMonitorProvider } from './context-monitor.js';
15
+ import { ClaudeCodeHookProvider } from './hooks.js';
16
+ import { ClaudeCodeInstallProvider } from './install.js';
17
+ import { ClaudeCodePathProvider } from './paths.js';
18
+ import { ClaudeCodeSpawnProvider } from './spawn.js';
19
+ import { ClaudeCodeTaskSyncProvider } from './task-sync.js';
20
+ import { ClaudeCodeTransportProvider } from './transport.js';
21
+ const execAsync = promisify(exec);
22
+ /**
23
+ * CLEO provider adapter for Anthropic Claude Code CLI.
24
+ *
25
+ * Bridges CLEO's adapter system with Claude Code's native capabilities:
26
+ * - Hooks: Maps Claude Code events (SessionStart, PostToolUse, etc.) to CAAMP events
27
+ * - Spawn: Launches subagent processes via the `claude` CLI
28
+ * - Install: Manages instruction files and brain observation plugin registration
29
+ *
30
+ * @remarks
31
+ * This is the most feature-complete adapter in the CLEO system, supporting
32
+ * 14 of 16 CAAMP canonical events, subagent spawning via the `claude` CLI,
33
+ * context monitoring with statusline integration, task sync via TodoWrite,
34
+ * and inter-agent transport. It serves as the reference implementation for
35
+ * other provider adapters.
36
+ */
37
+ export class ClaudeCodeAdapter {
38
+ /** Unique provider identifier. */
39
+ id = 'claude-code';
40
+ /** Human-readable provider name. */
41
+ name = 'Claude Code';
42
+ /** Adapter version string. */
43
+ version = '1.0.0';
44
+ /** Declared capabilities for this provider. */
45
+ capabilities = {
46
+ supportsHooks: true,
47
+ // 14/16 canonical events — derived from getProviderHookProfile('claude-code') in CAAMP 1.9.1.
48
+ // PreModel and PostModel are not supported by Claude Code.
49
+ supportedHookEvents: [
50
+ 'SessionStart',
51
+ 'SessionEnd',
52
+ 'PromptSubmit',
53
+ 'ResponseComplete',
54
+ 'PreToolUse',
55
+ 'PostToolUse',
56
+ 'PostToolUseFailure',
57
+ 'PermissionRequest',
58
+ 'SubagentStart',
59
+ 'SubagentStop',
60
+ 'PreCompact',
61
+ 'PostCompact',
62
+ 'Notification',
63
+ 'ConfigChange',
64
+ ],
65
+ supportsSpawn: true,
66
+ supportsInstall: true,
67
+ supportsInstructionFiles: true,
68
+ instructionFilePattern: 'CLAUDE.md',
69
+ supportsContextMonitor: true,
70
+ supportsStatusline: true,
71
+ supportsProviderPaths: true,
72
+ supportsTransport: true,
73
+ supportsTaskSync: true,
74
+ };
75
+ /** Hook provider for CAAMP event mapping and registration. */
76
+ hooks;
77
+ /** Spawn provider for launching subagent processes via `claude` CLI. */
78
+ spawn;
79
+ /** Install provider for managing instruction files and plugin registration. */
80
+ install;
81
+ /** Path provider for resolving Claude Code directory locations. */
82
+ paths;
83
+ /** Context monitor for tracking context window usage and statusline output. */
84
+ contextMonitor;
85
+ /** Transport provider for inter-agent communication. */
86
+ transport;
87
+ /** Task sync provider bridging Claude's TodoWrite format to CLEO tasks. */
88
+ taskSync;
89
+ /** Project directory this adapter was initialized with, or null. */
90
+ projectDir = null;
91
+ /** Whether {@link initialize} has been called. */
92
+ initialized = false;
93
+ constructor() {
94
+ this.hooks = new ClaudeCodeHookProvider();
95
+ this.spawn = new ClaudeCodeSpawnProvider();
96
+ this.install = new ClaudeCodeInstallProvider();
97
+ this.paths = new ClaudeCodePathProvider();
98
+ this.contextMonitor = new ClaudeCodeContextMonitorProvider();
99
+ this.transport = new ClaudeCodeTransportProvider();
100
+ this.taskSync = new ClaudeCodeTaskSyncProvider();
101
+ }
102
+ /**
103
+ * Initialize the adapter for a given project directory.
104
+ *
105
+ * Validates the environment by checking for the Claude CLI
106
+ * and Claude Code configuration directory.
107
+ *
108
+ * @param projectDir - Root directory of the project
109
+ */
110
+ async initialize(projectDir) {
111
+ this.projectDir = projectDir;
112
+ this.initialized = true;
113
+ }
114
+ /**
115
+ * Dispose the adapter and clean up resources.
116
+ *
117
+ * Unregisters hooks and releases any tracked state.
118
+ */
119
+ async dispose() {
120
+ if (this.hooks.isRegistered()) {
121
+ await this.hooks.unregisterNativeHooks();
122
+ }
123
+ this.initialized = false;
124
+ this.projectDir = null;
125
+ }
126
+ /**
127
+ * Run a health check to verify Claude Code is accessible.
128
+ *
129
+ * Checks:
130
+ * 1. Adapter has been initialized
131
+ * 2. Claude CLI is available in PATH
132
+ * 3. ~/.claude/ configuration directory exists
133
+ *
134
+ * @returns Health status with details about each check
135
+ */
136
+ async healthCheck() {
137
+ const details = {};
138
+ if (!this.initialized) {
139
+ return {
140
+ healthy: false,
141
+ provider: this.id,
142
+ details: { error: 'Adapter not initialized' },
143
+ };
144
+ }
145
+ // Check Claude CLI availability
146
+ let cliAvailable = false;
147
+ try {
148
+ const { stdout } = await execAsync('which claude');
149
+ cliAvailable = stdout.trim().length > 0;
150
+ details.cliPath = stdout.trim();
151
+ }
152
+ catch {
153
+ details.cliAvailable = false;
154
+ }
155
+ // Check for Claude Code config directory
156
+ const claudeConfigDir = join(homedir(), '.claude');
157
+ const configExists = existsSync(claudeConfigDir);
158
+ details.configDirExists = configExists;
159
+ // Check for CLAUDE_CODE_ENTRYPOINT env var
160
+ const entrypointSet = process.env.CLAUDE_CODE_ENTRYPOINT !== undefined;
161
+ details.entrypointEnvSet = entrypointSet;
162
+ // Healthy if CLI is available (primary requirement)
163
+ const healthy = cliAvailable;
164
+ details.cliAvailable = cliAvailable;
165
+ return {
166
+ healthy,
167
+ provider: this.id,
168
+ details,
169
+ };
170
+ }
171
+ /**
172
+ * Check whether the adapter has been initialized.
173
+ */
174
+ isInitialized() {
175
+ return this.initialized;
176
+ }
177
+ /**
178
+ * Get the project directory this adapter was initialized with.
179
+ */
180
+ getProjectDir() {
181
+ return this.projectDir;
182
+ }
183
+ }
184
+ //# sourceMappingURL=adapter.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"adapter.js","sourceRoot":"","sources":["../../../src/providers/claude-code/adapter.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,159 @@
1
+ /**
2
+ * Claude Code context monitor provider.
3
+ *
4
+ * Implements AdapterContextMonitorProvider for Claude Code's context window
5
+ * tracking and statusline integration.
6
+ *
7
+ * @task T5240
8
+ */
9
+ import { existsSync, readFileSync, writeFileSync } from 'node:fs';
10
+ import { mkdir } from 'node:fs/promises';
11
+ import { homedir } from 'node:os';
12
+ import { dirname, join } from 'node:path';
13
+ import { ClaudeCodePathProvider } from './paths.js';
14
+ /** Thresholds for context window status levels. */
15
+ const THRESHOLDS = {
16
+ WARNING: 50,
17
+ CAUTION: 70,
18
+ CRITICAL: 85,
19
+ EMERGENCY: 95,
20
+ };
21
+ function getContextStatusFromPercentage(percentage) {
22
+ if (percentage >= THRESHOLDS.EMERGENCY)
23
+ return 'emergency';
24
+ if (percentage >= THRESHOLDS.CRITICAL)
25
+ return 'critical';
26
+ if (percentage >= THRESHOLDS.CAUTION)
27
+ return 'caution';
28
+ if (percentage >= THRESHOLDS.WARNING)
29
+ return 'warning';
30
+ return 'ok';
31
+ }
32
+ /**
33
+ * Context monitor provider for Claude Code.
34
+ *
35
+ * Processes context window JSON from Claude Code and writes state files
36
+ * for statusline display. Also provides statusline configuration
37
+ * and setup instructions specific to Claude Code's settings.json.
38
+ *
39
+ * @remarks
40
+ * The provider writes a JSON state file to `.cleo/context-states/.context-state.json`
41
+ * which can be read by external statusline scripts (e.g. tmux, starship).
42
+ * Context thresholds at 50%, 70%, 85%, and 95% map to warning, caution,
43
+ * critical, and emergency status levels respectively.
44
+ */
45
+ export class ClaudeCodeContextMonitorProvider {
46
+ /** Path provider for resolving Claude Code directory locations. */
47
+ pathProvider = new ClaudeCodePathProvider();
48
+ /** Process raw context window JSON and return a formatted summary string. */
49
+ async processContextInput(input, cwd) {
50
+ const typed = input;
51
+ const contextSize = typed.context_window?.context_window_size ?? 200000;
52
+ const usage = typed.context_window?.current_usage;
53
+ if (!usage)
54
+ return '-- no data';
55
+ const inputTokens = usage.input_tokens ?? 0;
56
+ const outputTokens = usage.output_tokens ?? 0;
57
+ const cacheCreate = usage.cache_creation_input_tokens ?? 0;
58
+ const totalTokens = inputTokens + outputTokens + cacheCreate;
59
+ const percentage = Math.floor((totalTokens * 100) / contextSize);
60
+ const status = getContextStatusFromPercentage(percentage);
61
+ // Write state file if CLEO dir exists
62
+ const cleoDir = cwd ? join(cwd, '.cleo') : '.cleo';
63
+ if (existsSync(cleoDir)) {
64
+ const stateDir = join(cleoDir, 'context-states');
65
+ const statePath = join(stateDir, '.context-state.json');
66
+ const state = {
67
+ $schema: 'https://cleo-dev.com/schemas/v1/context-state.schema.json',
68
+ version: '1.0.0',
69
+ timestamp: new Date().toISOString().replace(/\.\d{3}Z$/, 'Z'),
70
+ staleAfterMs: 5000,
71
+ contextWindow: {
72
+ maxTokens: contextSize,
73
+ currentTokens: totalTokens,
74
+ percentage,
75
+ breakdown: {
76
+ inputTokens,
77
+ outputTokens,
78
+ cacheCreationTokens: cacheCreate,
79
+ cacheReadTokens: usage.cache_read_input_tokens ?? 0,
80
+ },
81
+ },
82
+ thresholds: {
83
+ warning: THRESHOLDS.WARNING,
84
+ caution: THRESHOLDS.CAUTION,
85
+ critical: THRESHOLDS.CRITICAL,
86
+ emergency: THRESHOLDS.EMERGENCY,
87
+ },
88
+ status,
89
+ cleoSessionId: '',
90
+ };
91
+ try {
92
+ await mkdir(dirname(statePath), { recursive: true });
93
+ writeFileSync(statePath, JSON.stringify(state, null, 2));
94
+ }
95
+ catch {
96
+ // Non-fatal
97
+ }
98
+ }
99
+ return `${percentage}% | ${totalTokens}/${contextSize}`;
100
+ }
101
+ /** Check the current statusline integration status in Claude Code settings. */
102
+ checkStatuslineIntegration() {
103
+ const settingsPath = this.pathProvider.getSettingsPath();
104
+ if (!settingsPath || !existsSync(settingsPath))
105
+ return 'no_settings';
106
+ try {
107
+ const settings = JSON.parse(readFileSync(settingsPath, 'utf-8'));
108
+ const statusLine = settings.statusLine;
109
+ if (!statusLine?.type)
110
+ return 'not_configured';
111
+ if (statusLine.type !== 'command')
112
+ return 'custom_no_cleo';
113
+ const cmd = statusLine.command ?? '';
114
+ if (cmd.includes('context-monitor.sh') ||
115
+ cmd.includes('cleo-statusline') ||
116
+ cmd.includes('.context-state.json') ||
117
+ cmd.includes('context-states')) {
118
+ return 'configured';
119
+ }
120
+ const scriptPath = cmd.startsWith('~') ? cmd.replace('~', homedir()) : cmd;
121
+ if (existsSync(scriptPath)) {
122
+ try {
123
+ const content = readFileSync(scriptPath, 'utf-8');
124
+ if (content.includes('context-state.json'))
125
+ return 'configured';
126
+ }
127
+ catch {
128
+ /* unreadable */
129
+ }
130
+ }
131
+ return 'custom_no_cleo';
132
+ }
133
+ catch {
134
+ return 'no_settings';
135
+ }
136
+ }
137
+ /** Get the recommended statusline configuration object for Claude Code settings. */
138
+ getStatuslineConfig() {
139
+ return {
140
+ statusLine: {
141
+ type: 'command',
142
+ command: join(homedir(), '.cleo', 'lib', 'session', 'context-monitor.sh'),
143
+ },
144
+ };
145
+ }
146
+ /** Get human-readable setup instructions for enabling context monitoring. */
147
+ getSetupInstructions() {
148
+ const settingsPath = this.pathProvider.getSettingsPath() ?? '~/.claude/settings.json';
149
+ return [
150
+ 'To enable context monitoring, add to your Claude Code settings:',
151
+ `File: ${settingsPath}`,
152
+ '',
153
+ JSON.stringify(this.getStatuslineConfig(), null, 2),
154
+ '',
155
+ 'This enables real-time context window tracking in the CLI.',
156
+ ].join('\n');
157
+ }
158
+ }
159
+ //# sourceMappingURL=context-monitor.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"context-monitor.js","sourceRoot":"","sources":["../../../src/providers/claude-code/context-monitor.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,286 @@
1
+ /**
2
+ * Claude Code Hook Provider
3
+ *
4
+ * Maps Claude Code's native hook events to CAAMP canonical hook events.
5
+ * Claude Code supports 14 of 16 canonical events (all except PreModel, PostModel).
6
+ *
7
+ * Event translation uses CAAMP normalizer APIs:
8
+ * - `toCanonical(nativeName, 'claude-code')` for runtime event name resolution
9
+ * - `getSupportedEvents('claude-code')` to enumerate supported canonical events
10
+ * - `getProviderHookProfile('claude-code')` for the full provider profile
11
+ *
12
+ * A static map derived from CAAMP 1.9.1 hook-mappings.json is maintained as
13
+ * a fallback for environments where CAAMP's runtime resolution is unavailable.
14
+ *
15
+ * @task T164
16
+ * @epic T134
17
+ */
18
+ import { readdir, readFile } from 'node:fs/promises';
19
+ import { join } from 'node:path';
20
+ /** CAAMP provider identifier for Claude Code. */
21
+ const PROVIDER_ID = 'claude-code';
22
+ /**
23
+ * Fallback map from Claude Code native event names to CAAMP canonical names.
24
+ *
25
+ * Derived from `getProviderHookProfile('claude-code').mappings` in CAAMP 1.9.1.
26
+ * Covers all 14 supported events. PreModel and PostModel are not supported
27
+ * by Claude Code and are absent from this map.
28
+ *
29
+ * Used as fallback when CAAMP runtime is unavailable, and as the synchronous
30
+ * implementation of `mapProviderEvent()`.
31
+ */
32
+ const CLAUDE_CODE_EVENT_MAP = {
33
+ // CAAMP: toNative('SessionStart', 'claude-code') = 'SessionStart'
34
+ SessionStart: 'SessionStart',
35
+ // CAAMP: toNative('SessionEnd', 'claude-code') = 'SessionEnd'
36
+ SessionEnd: 'SessionEnd',
37
+ // CAAMP: toNative('PromptSubmit', 'claude-code') = 'UserPromptSubmit'
38
+ UserPromptSubmit: 'PromptSubmit',
39
+ // CAAMP: toNative('ResponseComplete', 'claude-code') = 'Stop'
40
+ Stop: 'ResponseComplete',
41
+ // CAAMP: toNative('PreToolUse', 'claude-code') = 'PreToolUse'
42
+ PreToolUse: 'PreToolUse',
43
+ // CAAMP: toNative('PostToolUse', 'claude-code') = 'PostToolUse'
44
+ PostToolUse: 'PostToolUse',
45
+ // CAAMP: toNative('PostToolUseFailure','claude-code') = 'PostToolUseFailure'
46
+ PostToolUseFailure: 'PostToolUseFailure',
47
+ // CAAMP: toNative('PermissionRequest', 'claude-code') = 'PermissionRequest'
48
+ PermissionRequest: 'PermissionRequest',
49
+ // CAAMP: toNative('SubagentStart', 'claude-code') = 'SubagentStart'
50
+ SubagentStart: 'SubagentStart',
51
+ // CAAMP: toNative('SubagentStop', 'claude-code') = 'SubagentStop'
52
+ SubagentStop: 'SubagentStop',
53
+ // CAAMP: toNative('PreCompact', 'claude-code') = 'PreCompact'
54
+ PreCompact: 'PreCompact',
55
+ // CAAMP: toNative('PostCompact', 'claude-code') = 'PostCompact'
56
+ PostCompact: 'PostCompact',
57
+ // CAAMP: toNative('Notification', 'claude-code') = 'Notification'
58
+ Notification: 'Notification',
59
+ // CAAMP: toNative('ConfigChange', 'claude-code') = 'ConfigChange'
60
+ ConfigChange: 'ConfigChange',
61
+ };
62
+ /**
63
+ * Hook provider for Claude Code.
64
+ *
65
+ * Claude Code registers hooks via its global config at `~/.claude/settings.json`.
66
+ * Supported handler types: command, http, prompt, agent.
67
+ *
68
+ * Event mapping is based on `getProviderHookProfile('claude-code')` from
69
+ * CAAMP 1.9.1. Async accessors (`getSupportedCanonicalEvents`,
70
+ * `getProviderProfile`) call CAAMP directly when available.
71
+ *
72
+ * Since hooks are registered through the config system (managed by the install
73
+ * provider), `registerNativeHooks` and `unregisterNativeHooks` track registration
74
+ * state without performing filesystem operations.
75
+ *
76
+ * @remarks
77
+ * Claude Code is the only provider that supports all 14 of its declared
78
+ * canonical events at runtime. The static event map is maintained as a
79
+ * synchronous fallback; async methods like {@link getSupportedCanonicalEvents}
80
+ * and {@link getProviderProfile} call CAAMP directly when available.
81
+ *
82
+ * @task T164
83
+ * @epic T134
84
+ */
85
+ export class ClaudeCodeHookProvider {
86
+ /** Whether hooks have been registered for the current session. */
87
+ registered = false;
88
+ /**
89
+ * Map a Claude Code native event name to a CAAMP canonical hook event name.
90
+ *
91
+ * Looks up the native event name in the map derived from
92
+ * `getProviderHookProfile('claude-code').mappings` (CAAMP 1.9.1).
93
+ * Returns null for unrecognised events (e.g. PreModel, PostModel which
94
+ * Claude Code does not support).
95
+ *
96
+ * @param providerEvent - Claude Code native event (e.g. "UserPromptSubmit", "Stop")
97
+ * @returns CAAMP canonical event name, or null if unmapped
98
+ * @task T164
99
+ */
100
+ mapProviderEvent(providerEvent) {
101
+ return CLAUDE_CODE_EVENT_MAP[providerEvent] ?? null;
102
+ }
103
+ /**
104
+ * Register native hooks for a project.
105
+ *
106
+ * For Claude Code, hooks are registered via the config system
107
+ * (`~/.claude/settings.json`), managed by the install provider.
108
+ * This method marks hooks as registered without performing filesystem operations.
109
+ *
110
+ * Iterating supported events is handled at install time using
111
+ * `getSupportedCanonicalEvents()` to enumerate all 14 supported hooks.
112
+ *
113
+ * @param _projectDir - Project directory (unused; Claude Code uses global config)
114
+ * @task T164
115
+ */
116
+ async registerNativeHooks(_projectDir) {
117
+ this.registered = true;
118
+ }
119
+ /**
120
+ * Unregister native hooks.
121
+ *
122
+ * For Claude Code, this is a no-op since hooks are managed through the config
123
+ * system. Unregistration happens via the install provider's uninstall method.
124
+ *
125
+ * @task T164
126
+ */
127
+ async unregisterNativeHooks() {
128
+ this.registered = false;
129
+ }
130
+ /**
131
+ * Check whether hooks have been registered via `registerNativeHooks`.
132
+ */
133
+ isRegistered() {
134
+ return this.registered;
135
+ }
136
+ /**
137
+ * Get the native→canonical event mapping for introspection and debugging.
138
+ *
139
+ * Returns the map derived from `getProviderHookProfile('claude-code').mappings`
140
+ * (CAAMP 1.9.1). Use `getSupportedCanonicalEvents()` to enumerate canonical
141
+ * names via live CAAMP APIs.
142
+ *
143
+ * @returns Immutable record of native event name → canonical event name
144
+ */
145
+ getEventMap() {
146
+ return { ...CLAUDE_CODE_EVENT_MAP };
147
+ }
148
+ /**
149
+ * Enumerate supported canonical events via CAAMP's `getSupportedEvents()`.
150
+ *
151
+ * Calls `getSupportedEvents('claude-code')` from the CAAMP normalizer to
152
+ * get the authoritative list. Claude Code supports 14 of 16 canonical events
153
+ * (PreModel and PostModel are not supported). Falls back to the values of
154
+ * the static event map when CAAMP is unavailable at runtime.
155
+ *
156
+ * @returns Array of CAAMP canonical event names supported by Claude Code
157
+ * @task T164
158
+ */
159
+ async getSupportedCanonicalEvents() {
160
+ try {
161
+ const { getSupportedEvents } = await import('@cleocode/caamp');
162
+ return getSupportedEvents(PROVIDER_ID);
163
+ }
164
+ catch {
165
+ return [...new Set(Object.values(CLAUDE_CODE_EVENT_MAP))];
166
+ }
167
+ }
168
+ /**
169
+ * Retrieve the full provider hook profile from CAAMP.
170
+ *
171
+ * Calls `getProviderHookProfile('claude-code')` from the CAAMP normalizer to
172
+ * get the complete profile: hook system type (`config`), config path
173
+ * (`~/.claude/settings.json`), handler types, and all event mappings.
174
+ * Returns null when CAAMP is unavailable at runtime.
175
+ *
176
+ * @returns Provider hook profile or null if CAAMP is unavailable
177
+ * @task T164
178
+ */
179
+ async getProviderProfile() {
180
+ try {
181
+ const { getProviderHookProfile } = await import('@cleocode/caamp');
182
+ return getProviderHookProfile(PROVIDER_ID) ?? null;
183
+ }
184
+ catch {
185
+ return null;
186
+ }
187
+ }
188
+ /**
189
+ * Translate a CAAMP canonical event to its Claude Code native name via CAAMP.
190
+ *
191
+ * Calls `toNative(canonical, 'claude-code')` from the CAAMP normalizer.
192
+ * Returns null for unsupported events (PreModel, PostModel) or when
193
+ * CAAMP is unavailable.
194
+ *
195
+ * @param canonical - CAAMP canonical event name (e.g. "PromptSubmit")
196
+ * @returns Claude Code native event name or null
197
+ * @task T164
198
+ */
199
+ async toNativeEvent(canonical) {
200
+ try {
201
+ const { toNative } = await import('@cleocode/caamp');
202
+ return toNative(canonical, PROVIDER_ID);
203
+ }
204
+ catch {
205
+ // Invert the static map as fallback
206
+ const entry = Object.entries(CLAUDE_CODE_EVENT_MAP).find(([, v]) => v === canonical);
207
+ return entry?.[0] ?? null;
208
+ }
209
+ }
210
+ /**
211
+ * Extract a plain-text transcript from Claude Code session JSONL files.
212
+ *
213
+ * Reads the most recent .jsonl file under `~/.claude/projects/` and
214
+ * extracts user/assistant turn text into a flat string for brain
215
+ * observation extraction.
216
+ *
217
+ * Returns null when no session data is found or on any read error.
218
+ *
219
+ * @param _sessionId - CLEO session ID (unused; reads the most recent file)
220
+ * @param _projectDir - Project directory (unused; Claude Code uses global paths)
221
+ * @task T144 @epic T134
222
+ */
223
+ async getTranscript(_sessionId, _projectDir) {
224
+ try {
225
+ const homeDir = process.env.HOME ?? process.env.USERPROFILE ?? '/root';
226
+ const projectsDir = join(homeDir, '.claude', 'projects');
227
+ // Find all JSONL files across project subdirectories
228
+ let allFiles = [];
229
+ try {
230
+ const projectDirs = await readdir(projectsDir, { withFileTypes: true });
231
+ for (const entry of projectDirs) {
232
+ if (!entry.isDirectory())
233
+ continue;
234
+ const subDir = join(projectsDir, entry.name);
235
+ try {
236
+ const files = await readdir(subDir);
237
+ for (const file of files) {
238
+ if (!file.endsWith('.jsonl'))
239
+ continue;
240
+ const filePath = join(subDir, file);
241
+ // Use file path modification heuristic (filename usually includes timestamp)
242
+ allFiles.push({ path: filePath, mtime: 0 });
243
+ }
244
+ }
245
+ catch {
246
+ // Skip unreadable subdirectories
247
+ }
248
+ }
249
+ }
250
+ catch {
251
+ return null;
252
+ }
253
+ if (allFiles.length === 0)
254
+ return null;
255
+ // Sort by path descending (timestamps in filenames sort naturally)
256
+ allFiles = allFiles.sort((a, b) => b.path.localeCompare(a.path));
257
+ const mostRecent = allFiles[0];
258
+ if (!mostRecent)
259
+ return null;
260
+ const raw = await readFile(mostRecent.path, 'utf-8');
261
+ const lines = raw.split('\n').filter((l) => l.trim());
262
+ const turns = [];
263
+ for (const line of lines) {
264
+ try {
265
+ const entry = JSON.parse(line);
266
+ const role = entry.role;
267
+ const content = entry.content;
268
+ if (role === 'assistant' && typeof content === 'string') {
269
+ turns.push(`assistant: ${content}`);
270
+ }
271
+ else if (role === 'user' && typeof content === 'string') {
272
+ turns.push(`user: ${content}`);
273
+ }
274
+ }
275
+ catch {
276
+ // Skip malformed lines
277
+ }
278
+ }
279
+ return turns.length > 0 ? turns.join('\n') : null;
280
+ }
281
+ catch {
282
+ return null;
283
+ }
284
+ }
285
+ }
286
+ //# sourceMappingURL=hooks.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"hooks.js","sourceRoot":"","sources":["../../../src/providers/claude-code/hooks.ts"],"names":[],"mappings":"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"}