@dollhousemcp/mcp-server 2.0.16 → 2.0.17

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 (40) hide show
  1. package/CHANGELOG.md +6 -0
  2. package/README.md.backup +18 -0
  3. package/dist/elements/base/BaseElementManager.d.ts.map +1 -1
  4. package/dist/elements/base/BaseElementManager.js +17 -1
  5. package/dist/elements/memories/MemoryManager.d.ts.map +1 -1
  6. package/dist/elements/memories/MemoryManager.js +13 -2
  7. package/dist/generated/version.d.ts +2 -2
  8. package/dist/generated/version.js +3 -3
  9. package/dist/handlers/element-crud/createElement.d.ts.map +1 -1
  10. package/dist/handlers/element-crud/createElement.js +6 -2
  11. package/dist/handlers/element-crud/editElement.d.ts.map +1 -1
  12. package/dist/handlers/element-crud/editElement.js +6 -2
  13. package/dist/handlers/element-crud/helpers.d.ts +2 -0
  14. package/dist/handlers/element-crud/helpers.d.ts.map +1 -1
  15. package/dist/handlers/element-crud/helpers.js +21 -2
  16. package/dist/handlers/mcp-aql/IntrospectionResolver.d.ts.map +1 -1
  17. package/dist/handlers/mcp-aql/IntrospectionResolver.js +34 -7
  18. package/dist/handlers/mcp-aql/MCPAQLHandler.d.ts.map +1 -1
  19. package/dist/handlers/mcp-aql/MCPAQLHandler.js +19 -4
  20. package/dist/handlers/mcp-aql/OperationSchema.d.ts.map +1 -1
  21. package/dist/handlers/mcp-aql/OperationSchema.js +3 -2
  22. package/dist/handlers/mcp-aql/evaluatePermission.d.ts.map +1 -1
  23. package/dist/handlers/mcp-aql/evaluatePermission.js +2 -1
  24. package/dist/handlers/mcp-aql/policies/ElementPolicies.d.ts +8 -0
  25. package/dist/handlers/mcp-aql/policies/ElementPolicies.d.ts.map +1 -1
  26. package/dist/handlers/mcp-aql/policies/ElementPolicies.js +26 -1
  27. package/dist/utils/permissionHooks.d.ts +39 -3
  28. package/dist/utils/permissionHooks.d.ts.map +1 -1
  29. package/dist/utils/permissionHooks.js +651 -74
  30. package/dist/web/public/permissions.js +38 -26
  31. package/dist/web/public/setup.js +367 -94
  32. package/dist/web/routes/permissionRoutes.d.ts.map +1 -1
  33. package/dist/web/routes/permissionRoutes.js +32 -13
  34. package/dist/web/routes/setupRoutes.d.ts +1 -0
  35. package/dist/web/routes/setupRoutes.d.ts.map +1 -1
  36. package/dist/web/routes/setupRoutes.js +113 -41
  37. package/package.json +3 -1
  38. package/scripts/pretooluse-vscode.sh +163 -0
  39. package/scripts/pretooluse-windsurf.sh +166 -4
  40. package/server.json +2 -2
@@ -1,8 +1,10 @@
1
- import { accessSync, constants as fsConstants, copyFileSync, existsSync, readFileSync, statSync, writeFileSync } from 'node:fs';
2
- import { chmod, mkdir, readFile, writeFile } from 'node:fs/promises';
1
+ import { existsSync, readFileSync, readdirSync, statSync } from 'node:fs';
2
+ import { access, chmod, copyFile, mkdir, readFile, readdir, writeFile } from 'node:fs/promises';
3
3
  import { dirname, join } from 'node:path';
4
4
  import { fileURLToPath } from 'node:url';
5
- import { homedir } from 'node:os';
5
+ import { homedir, platform } from 'node:os';
6
+ import { UnicodeValidator } from '../security/validators/unicodeValidator.js';
7
+ import { logger } from './logger.js';
6
8
  function repoRootFromModule() {
7
9
  const currentFile = fileURLToPath(import.meta.url);
8
10
  return dirname(dirname(dirname(currentFile)));
@@ -30,36 +32,174 @@ function detectIndent(raw) {
30
32
  export function getPermissionHookScriptPath(homeDir = homedir()) {
31
33
  return join(homeDir, '.dollhouse', 'hooks', 'pretooluse-dollhouse.sh');
32
34
  }
33
- export function getPermissionHookMarkerPath(homeDir = homedir()) {
34
- return join(homeDir, '.dollhouse', 'run', 'hook-installed.json');
35
+ function getPermissionHookRunDir(homeDir = homedir()) {
36
+ return join(homeDir, '.dollhouse', 'run');
35
37
  }
36
- export function getClaudeHookSettingsPath(homeDir = homedir()) {
37
- return join(homeDir, '.claude', 'settings.json');
38
+ function normalizeHookHost(host) {
39
+ return UnicodeValidator.normalize(host).normalizedContent.trim().toLowerCase();
40
+ }
41
+ function isHookMarkerFilename(entry) {
42
+ return entry.startsWith('hook-installed-') && entry.endsWith('.json');
43
+ }
44
+ function readHostSpecificHookStatus(homeDir, host) {
45
+ const normalized = normalizeHookHost(host);
46
+ const status = readMarkerStatus(getPermissionHookMarkerPath(homeDir, normalized));
47
+ if (status.installed || status.assetsPrepared) {
48
+ return status;
49
+ }
50
+ if (normalized === 'claude-code') {
51
+ return readMarkerStatus(getPermissionHookMarkerPath(homeDir));
52
+ }
53
+ return { installed: false };
38
54
  }
39
- export function getPermissionHookStatus(homeDir = homedir()) {
40
- const markerPath = getPermissionHookMarkerPath(homeDir);
55
+ function collectHookMarkerPaths(homeDir) {
56
+ const markerPaths = new Set([getPermissionHookMarkerPath(homeDir)]);
57
+ const runDir = getPermissionHookRunDir(homeDir);
41
58
  try {
42
- const raw = readFileSync(markerPath, 'utf-8');
43
- const parsed = JSON.parse(raw);
44
- if (typeof parsed.host !== 'string' ||
45
- typeof parsed.scriptPath !== 'string' ||
46
- typeof parsed.settingsPath !== 'string') {
47
- return { installed: false };
59
+ for (const entry of readdirSync(runDir)) {
60
+ if (isHookMarkerFilename(entry)) {
61
+ markerPaths.add(join(runDir, entry));
62
+ }
48
63
  }
49
- if (!existsSync(parsed.scriptPath) || !existsSync(parsed.settingsPath)) {
50
- return { installed: false };
64
+ }
65
+ catch {
66
+ // No run dir yet — fall through to default false.
67
+ }
68
+ return markerPaths;
69
+ }
70
+ async function collectHookMarkerPathsAsync(homeDir) {
71
+ const markerPaths = new Set([getPermissionHookMarkerPath(homeDir)]);
72
+ const runDir = getPermissionHookRunDir(homeDir);
73
+ try {
74
+ for (const entry of await readdir(runDir)) {
75
+ if (isHookMarkerFilename(entry)) {
76
+ markerPaths.add(join(runDir, entry));
77
+ }
51
78
  }
52
- return {
53
- installed: true,
54
- host: parsed.host,
55
- scriptPath: parsed.scriptPath,
56
- settingsPath: parsed.settingsPath,
57
- };
58
79
  }
59
80
  catch {
81
+ // No run dir yet — fall through to default false.
82
+ }
83
+ return markerPaths;
84
+ }
85
+ function summarizeMarkerStatuses(markerPaths) {
86
+ let fallback = { installed: false };
87
+ for (const markerPath of markerPaths) {
88
+ const status = readMarkerStatus(markerPath);
89
+ if (status.installed)
90
+ return status;
91
+ if (!fallback.assetsPrepared && status.assetsPrepared)
92
+ fallback = status;
93
+ }
94
+ return fallback;
95
+ }
96
+ function getHookWrapperBasename(host) {
97
+ switch (normalizeHookHost(host)) {
98
+ case 'vscode':
99
+ return 'pretooluse-vscode.sh';
100
+ case 'cursor':
101
+ return 'pretooluse-cursor.sh';
102
+ case 'windsurf':
103
+ return 'pretooluse-windsurf.sh';
104
+ case 'gemini-cli':
105
+ return 'pretooluse-gemini.sh';
106
+ case 'codex':
107
+ return 'pretooluse-codex.sh';
108
+ default:
109
+ return null;
110
+ }
111
+ }
112
+ function getHookWrapperPath(host, homeDir = homedir()) {
113
+ const basename = getHookWrapperBasename(host);
114
+ return basename ? join(homeDir, '.dollhouse', 'hooks', basename) : null;
115
+ }
116
+ function getHookSourcePath(host) {
117
+ const root = repoRootFromModule();
118
+ const basename = getHookWrapperBasename(host);
119
+ return basename ? join(root, 'scripts', basename) : join(root, 'scripts', 'pretooluse-dollhouse.sh');
120
+ }
121
+ export function getPermissionHookMarkerPath(homeDir = homedir(), host) {
122
+ if (!host) {
123
+ return join(getPermissionHookRunDir(homeDir), 'hook-installed.json');
124
+ }
125
+ return join(getPermissionHookRunDir(homeDir), `hook-installed-${normalizeHookHost(host)}.json`);
126
+ }
127
+ export function getClaudeHookSettingsPath(homeDir = homedir()) {
128
+ return join(homeDir, '.claude', 'settings.json');
129
+ }
130
+ export function getVsCodeHookSettingsPath(homeDir = homedir()) {
131
+ return join(homeDir, '.copilot', 'hooks', 'dollhouse-permissions.json');
132
+ }
133
+ export function getVsCodeUserSettingsPath(homeDir = homedir()) {
134
+ const currentPlatform = platform();
135
+ if (currentPlatform === 'darwin') {
136
+ return join(homeDir, 'Library', 'Application Support', 'Code', 'User', 'settings.json');
137
+ }
138
+ if (currentPlatform === 'win32') {
139
+ const appData = process.env.APPDATA || join(homeDir, 'AppData', 'Roaming');
140
+ return join(appData, 'Code', 'User', 'settings.json');
141
+ }
142
+ return join(homeDir, '.config', 'Code', 'User', 'settings.json');
143
+ }
144
+ export function getGeminiHookSettingsPath(homeDir = homedir()) {
145
+ return join(homeDir, '.gemini', 'settings.json');
146
+ }
147
+ export function getCursorHookSettingsPath(homeDir = homedir()) {
148
+ return join(homeDir, '.cursor', 'hooks.json');
149
+ }
150
+ export function getWindsurfHookSettingsPath(homeDir = homedir()) {
151
+ return join(homeDir, '.codeium', 'windsurf', 'hooks.json');
152
+ }
153
+ export function getCodexHookSettingsPath(homeDir = homedir()) {
154
+ return join(homeDir, '.codex', 'hooks.json');
155
+ }
156
+ export function getCodexConfigPath(homeDir = homedir()) {
157
+ return join(homeDir, '.codex', 'config.toml');
158
+ }
159
+ function toPermissionHookStatus(raw) {
160
+ if (typeof raw.host !== 'string' ||
161
+ typeof raw.scriptPath !== 'string') {
162
+ return { installed: false };
163
+ }
164
+ const scriptReady = existsSync(raw.scriptPath);
165
+ const settingsReady = !raw.settingsPath || existsSync(raw.settingsPath);
166
+ const additionalPathsReady = !raw.additionalPaths || raw.additionalPaths.every((path) => existsSync(path));
167
+ const assetsPrepared = (raw.assetsPrepared ?? raw.configured ?? true) && scriptReady;
168
+ const configured = (raw.configured ?? true) && scriptReady && settingsReady && additionalPathsReady;
169
+ return {
170
+ installed: configured,
171
+ configured,
172
+ assetsPrepared,
173
+ host: raw.host,
174
+ scriptPath: raw.scriptPath,
175
+ settingsPath: raw.settingsPath,
176
+ additionalPaths: raw.additionalPaths,
177
+ };
178
+ }
179
+ function readMarkerStatus(markerPath) {
180
+ try {
181
+ const raw = readFileSync(markerPath, 'utf-8');
182
+ return toPermissionHookStatus(JSON.parse(raw));
183
+ }
184
+ catch (error) {
185
+ if (!isMissingFileError(error)) {
186
+ logger.warn(`[Permissions] Failed to read hook marker at ${markerPath}: ${String(error)}`);
187
+ }
60
188
  return { installed: false };
61
189
  }
62
190
  }
191
+ export function getPermissionHookStatus(homeDir = homedir(), host) {
192
+ if (host) {
193
+ return readHostSpecificHookStatus(homeDir, host);
194
+ }
195
+ return summarizeMarkerStatuses(collectHookMarkerPaths(homeDir));
196
+ }
197
+ export async function getPermissionHookStatusAsync(homeDir = homedir(), host) {
198
+ if (host) {
199
+ return readHostSpecificHookStatus(homeDir, host);
200
+ }
201
+ return summarizeMarkerStatuses(await collectHookMarkerPathsAsync(homeDir));
202
+ }
63
203
  function normalizeHooksRoot(parsed) {
64
204
  const hooksValue = parsed.hooks;
65
205
  if (!hooksValue || typeof hooksValue !== 'object' || Array.isArray(hooksValue)) {
@@ -67,12 +207,12 @@ function normalizeHooksRoot(parsed) {
67
207
  }
68
208
  return parsed.hooks;
69
209
  }
70
- export function ensureClaudePreToolUseHook(parsed, command) {
210
+ function ensureCommandHook(parsed, eventName, command, matcher, extraHookFields = {}) {
71
211
  const hooksRoot = normalizeHooksRoot(parsed);
72
- const existingEntries = Array.isArray(hooksRoot.PreToolUse)
73
- ? hooksRoot.PreToolUse.filter((entry) => typeof entry === 'object' && entry !== null)
212
+ const existingEntries = Array.isArray(hooksRoot[eventName])
213
+ ? hooksRoot[eventName].filter((entry) => typeof entry === 'object' && entry !== null)
74
214
  : [];
75
- hooksRoot.PreToolUse = existingEntries;
215
+ hooksRoot[eventName] = existingEntries;
76
216
  const commandExists = existingEntries.some((entry) => {
77
217
  const hooks = Array.isArray(entry?.hooks) ? entry.hooks : [];
78
218
  return hooks.some((hook) => hook?.type === 'command' && hook?.command === command);
@@ -80,28 +220,87 @@ export function ensureClaudePreToolUseHook(parsed, command) {
80
220
  if (commandExists) {
81
221
  return { changed: false, parsed };
82
222
  }
83
- const wildcardEntry = existingEntries.find((entry) => (entry?.matcher === '*' || entry?.matcher === undefined) && Array.isArray(entry?.hooks));
223
+ const wildcardEntry = existingEntries.find((entry) => (entry?.matcher === matcher || entry?.matcher === undefined) && Array.isArray(entry?.hooks));
84
224
  if (wildcardEntry) {
85
225
  const hooks = wildcardEntry.hooks;
86
226
  hooks.push({
87
227
  type: 'command',
88
228
  command,
229
+ ...extraHookFields,
89
230
  });
90
231
  }
91
232
  else {
92
233
  existingEntries.push({
93
- matcher: '*',
234
+ matcher,
94
235
  hooks: [
95
236
  {
96
237
  type: 'command',
97
238
  command,
239
+ ...extraHookFields,
98
240
  },
99
241
  ],
100
242
  });
101
243
  }
102
244
  return { changed: true, parsed };
103
245
  }
104
- async function copyHookScript(sourcePath, targetPath) {
246
+ export function ensureClaudePreToolUseHook(parsed, command) {
247
+ return ensureCommandHook(parsed, 'PreToolUse', command, '*');
248
+ }
249
+ export function ensureVsCodePreToolUseHook(parsed, command) {
250
+ return ensureCommandHook(parsed, 'PreToolUse', command, '*');
251
+ }
252
+ export function ensureGeminiBeforeToolHook(parsed, command) {
253
+ return ensureCommandHook(parsed, 'BeforeTool', command, '.*');
254
+ }
255
+ export function ensureCodexPreToolUseHook(parsed, command) {
256
+ return ensureCommandHook(parsed, 'PreToolUse', command, 'Bash', {
257
+ statusMessage: 'Checking Bash permissions',
258
+ });
259
+ }
260
+ export function ensureCursorPreToolUseHook(parsed, command) {
261
+ if (parsed.version !== 1) {
262
+ parsed.version = 1;
263
+ }
264
+ const hooksRoot = normalizeHooksRoot(parsed);
265
+ const existingEntries = Array.isArray(hooksRoot.preToolUse)
266
+ ? hooksRoot.preToolUse.filter((entry) => typeof entry === 'object' && entry !== null)
267
+ : [];
268
+ hooksRoot.preToolUse = existingEntries;
269
+ const commandExists = existingEntries.some((entry) => entry.command === command
270
+ && (entry.type === 'command' || entry.type === undefined));
271
+ if (commandExists) {
272
+ return { changed: false, parsed };
273
+ }
274
+ existingEntries.push({
275
+ type: 'command',
276
+ command,
277
+ matcher: '.*',
278
+ });
279
+ return { changed: true, parsed };
280
+ }
281
+ export function ensureWindsurfHooks(parsed, command) {
282
+ const hooksRoot = normalizeHooksRoot(parsed);
283
+ let changed = false;
284
+ const ensureEventHook = (eventName) => {
285
+ const existingEntries = Array.isArray(hooksRoot[eventName])
286
+ ? hooksRoot[eventName].filter((entry) => typeof entry === 'object' && entry !== null)
287
+ : [];
288
+ hooksRoot[eventName] = existingEntries;
289
+ const commandExists = existingEntries.some((entry) => entry.command === command && (entry.type === 'command' || entry.type === undefined));
290
+ if (commandExists) {
291
+ return;
292
+ }
293
+ existingEntries.push({
294
+ type: 'command',
295
+ command,
296
+ });
297
+ changed = true;
298
+ };
299
+ ensureEventHook('pre_run_command');
300
+ ensureEventHook('pre_mcp_tool_use');
301
+ return { changed, parsed };
302
+ }
303
+ async function copyHookAsset(sourcePath, targetPath) {
105
304
  await mkdir(dirname(targetPath), { recursive: true });
106
305
  const sourceRaw = await readFile(sourcePath, 'utf-8');
107
306
  let targetRaw;
@@ -115,80 +314,458 @@ async function copyHookScript(sourcePath, targetPath) {
115
314
  }
116
315
  const changed = targetRaw === undefined || sourceRaw !== targetRaw;
117
316
  if (changed) {
118
- copyFileSync(sourcePath, targetPath);
317
+ await copyFile(sourcePath, targetPath);
119
318
  }
120
319
  else {
121
- accessSync(targetPath, fsConstants.F_OK);
320
+ await access(targetPath);
122
321
  }
123
322
  await chmod(targetPath, 0o755);
124
323
  return changed;
125
324
  }
126
- async function mergeClaudeSettings(settingsPath, command) {
127
- await mkdir(dirname(settingsPath), { recursive: true });
128
- let raw = '{}\n';
325
+ async function readOptionalUtf8(filePath, fallback) {
129
326
  try {
130
- raw = await readFile(settingsPath, 'utf-8');
327
+ return await readFile(filePath, 'utf-8');
131
328
  }
132
- catch {
133
- raw = '{}\n';
329
+ catch (error) {
330
+ if (isMissingFileError(error)) {
331
+ return fallback;
332
+ }
333
+ throw error;
334
+ }
335
+ }
336
+ async function writeBackupIfPresent(filePath, raw) {
337
+ if (!existsSync(filePath)) {
338
+ return undefined;
134
339
  }
340
+ const backupPath = `${filePath}.dollhouse.bak`;
341
+ await writeFile(backupPath, raw, 'utf-8');
342
+ return backupPath;
343
+ }
344
+ async function mergeClaudeSettings(settingsPath, command) {
345
+ await mkdir(dirname(settingsPath), { recursive: true });
346
+ const raw = await readOptionalUtf8(settingsPath, '{}\n');
135
347
  const indent = detectIndent(raw);
136
348
  const parsed = raw.trim().length === 0 ? {} : JSON.parse(raw);
137
349
  const { changed, parsed: updated } = ensureClaudePreToolUseHook(parsed, command);
138
350
  if (!changed) {
139
351
  return { changed: false };
140
352
  }
141
- let backupPath;
142
- if (existsSync(settingsPath)) {
143
- backupPath = `${settingsPath}.dollhouse.bak`;
144
- writeFileSync(backupPath, raw, 'utf-8');
353
+ const backupPath = await writeBackupIfPresent(settingsPath, raw);
354
+ await writeFile(settingsPath, JSON.stringify(updated, null, indent) + '\n', 'utf-8');
355
+ return { changed: true, backupPath };
356
+ }
357
+ async function mergeVsCodeHookSettings(settingsPath, command) {
358
+ await mkdir(dirname(settingsPath), { recursive: true });
359
+ const raw = await readOptionalUtf8(settingsPath, '{}\n');
360
+ const indent = detectIndent(raw);
361
+ const parsed = raw.trim().length === 0 ? {} : JSON.parse(raw);
362
+ const { changed, parsed: updated } = ensureVsCodePreToolUseHook(parsed, command);
363
+ if (!changed) {
364
+ return { changed: false };
145
365
  }
366
+ const backupPath = await writeBackupIfPresent(settingsPath, raw);
146
367
  await writeFile(settingsPath, JSON.stringify(updated, null, indent) + '\n', 'utf-8');
147
368
  return { changed: true, backupPath };
148
369
  }
149
- export async function installPermissionHook(client, options = {}) {
150
- const normalizedClient = client.normalize('NFC').trim().toLowerCase();
151
- if (normalizedClient !== 'claude-code') {
152
- return {
153
- supported: false,
154
- installed: false,
155
- configured: false,
156
- host: normalizedClient,
157
- message: `Automatic permission hook wiring is not yet supported for ${normalizedClient}.`,
158
- };
370
+ async function mergeVsCodeUserSettings(settingsPath) {
371
+ await mkdir(dirname(settingsPath), { recursive: true });
372
+ const raw = await readOptionalUtf8(settingsPath, '{}\n');
373
+ const indent = detectIndent(raw);
374
+ const parsed = raw.trim().length === 0 ? {} : JSON.parse(raw);
375
+ const current = parsed['chat.hookFilesLocations'];
376
+ const locations = (current && typeof current === 'object' && !Array.isArray(current))
377
+ ? { ...current }
378
+ : {};
379
+ if (locations['~/.copilot/hooks'] === true) {
380
+ return { changed: false };
159
381
  }
160
- const homeDir = options.homeDir ?? homedir();
161
- const sourceScriptPath = options.sourceScriptPath
162
- ?? join(repoRootFromModule(), 'scripts', 'pretooluse-dollhouse.sh');
163
- const targetScriptPath = getPermissionHookScriptPath(homeDir);
164
- const settingsPath = getClaudeHookSettingsPath(homeDir);
165
- const markerPath = getPermissionHookMarkerPath(homeDir);
166
- const command = `bash ${targetScriptPath}`;
167
- const sourceStat = statSync(sourceScriptPath);
168
- if (!sourceStat.isFile()) {
169
- throw new Error(`Permission hook source script not found: ${sourceScriptPath}`);
170
- }
171
- await copyHookScript(sourceScriptPath, targetScriptPath);
172
- const settingsResult = await mergeClaudeSettings(settingsPath, command);
382
+ locations['~/.copilot/hooks'] = true;
383
+ parsed['chat.hookFilesLocations'] = locations;
384
+ const backupPath = await writeBackupIfPresent(settingsPath, raw);
385
+ await writeFile(settingsPath, JSON.stringify(parsed, null, indent) + '\n', 'utf-8');
386
+ return { changed: true, backupPath };
387
+ }
388
+ async function mergeGeminiSettings(settingsPath, command) {
389
+ await mkdir(dirname(settingsPath), { recursive: true });
390
+ const raw = await readOptionalUtf8(settingsPath, '{}\n');
391
+ const indent = detectIndent(raw);
392
+ const parsed = raw.trim().length === 0 ? {} : JSON.parse(raw);
393
+ const { changed, parsed: updated } = ensureGeminiBeforeToolHook(parsed, command);
394
+ if (!changed) {
395
+ return { changed: false };
396
+ }
397
+ const backupPath = await writeBackupIfPresent(settingsPath, raw);
398
+ await writeFile(settingsPath, JSON.stringify(updated, null, indent) + '\n', 'utf-8');
399
+ return { changed: true, backupPath };
400
+ }
401
+ async function mergeCursorHooks(settingsPath, command) {
402
+ await mkdir(dirname(settingsPath), { recursive: true });
403
+ const raw = await readOptionalUtf8(settingsPath, '{\n "version": 1,\n "hooks": {}\n}\n');
404
+ const indent = detectIndent(raw);
405
+ const parsed = raw.trim().length === 0 ? {} : JSON.parse(raw);
406
+ const { changed, parsed: updated } = ensureCursorPreToolUseHook(parsed, command);
407
+ if (!changed) {
408
+ return { changed: false };
409
+ }
410
+ const backupPath = await writeBackupIfPresent(settingsPath, raw);
411
+ await writeFile(settingsPath, JSON.stringify(updated, null, indent) + '\n', 'utf-8');
412
+ return { changed: true, backupPath };
413
+ }
414
+ async function mergeWindsurfHooks(settingsPath, command) {
415
+ await mkdir(dirname(settingsPath), { recursive: true });
416
+ const raw = await readOptionalUtf8(settingsPath, '{\n "hooks": {}\n}\n');
417
+ const indent = detectIndent(raw);
418
+ const parsed = raw.trim().length === 0 ? {} : JSON.parse(raw);
419
+ const { changed, parsed: updated } = ensureWindsurfHooks(parsed, command);
420
+ if (!changed) {
421
+ return { changed: false };
422
+ }
423
+ const backupPath = await writeBackupIfPresent(settingsPath, raw);
424
+ await writeFile(settingsPath, JSON.stringify(updated, null, indent) + '\n', 'utf-8');
425
+ return { changed: true, backupPath };
426
+ }
427
+ async function mergeCodexHooks(hooksPath, command) {
428
+ await mkdir(dirname(hooksPath), { recursive: true });
429
+ const raw = await readOptionalUtf8(hooksPath, '{}\n');
430
+ const indent = detectIndent(raw);
431
+ const parsed = raw.trim().length === 0 ? {} : JSON.parse(raw);
432
+ const { changed, parsed: updated } = ensureCodexPreToolUseHook(parsed, command);
433
+ if (!changed) {
434
+ return { changed: false };
435
+ }
436
+ const backupPath = await writeBackupIfPresent(hooksPath, raw);
437
+ await writeFile(hooksPath, JSON.stringify(updated, null, indent) + '\n', 'utf-8');
438
+ return { changed: true, backupPath };
439
+ }
440
+ function getTomlLineContent(line) {
441
+ const commentIndex = line.indexOf('#');
442
+ return (commentIndex >= 0 ? line.slice(0, commentIndex) : line).trim();
443
+ }
444
+ function isTomlSectionLine(line, section) {
445
+ return getTomlLineContent(line) === `[${section}]`;
446
+ }
447
+ function parseTomlBooleanAssignment(line, key) {
448
+ const content = getTomlLineContent(line);
449
+ if (!content.startsWith(`${key} = `)) {
450
+ return null;
451
+ }
452
+ const value = content.slice(`${key} = `.length).trim();
453
+ if (value === 'true')
454
+ return true;
455
+ if (value === 'false')
456
+ return false;
457
+ return null;
458
+ }
459
+ function updateTomlBooleanAssignment(line, key, nextValue) {
460
+ const commentIndex = line.indexOf('#');
461
+ const commentSuffix = commentIndex >= 0 ? line.slice(commentIndex) : '';
462
+ let prefixLength = 0;
463
+ while (prefixLength < line.length && /\s/.test(line.charAt(prefixLength))) {
464
+ prefixLength += 1;
465
+ }
466
+ const prefix = line.slice(0, prefixLength);
467
+ const assignment = `${prefix}${key} = ${nextValue ? 'true' : 'false'}`;
468
+ const trimmedCommentSuffix = commentSuffix.trimStart();
469
+ const suffix = trimmedCommentSuffix.length > 0 ? ` ${trimmedCommentSuffix}` : '';
470
+ return `${assignment}${suffix}`.trimEnd();
471
+ }
472
+ function stripTrailingNewlines(value) {
473
+ let end = value.length;
474
+ while (end > 0 && value.charAt(end - 1) === '\n') {
475
+ end -= 1;
476
+ }
477
+ return value.slice(0, end);
478
+ }
479
+ function ensureCodexHooksEnabled(raw) {
480
+ const lines = raw.length > 0 ? raw.split('\n') : [];
481
+ const dottedIndex = lines.findIndex((line) => parseTomlBooleanAssignment(line, 'features.codex_hooks') !== null);
482
+ if (dottedIndex >= 0) {
483
+ if (parseTomlBooleanAssignment(lines[dottedIndex], 'features.codex_hooks') === true) {
484
+ return { changed: false, content: raw };
485
+ }
486
+ const updatedLines = [...lines];
487
+ updatedLines[dottedIndex] = updateTomlBooleanAssignment(updatedLines[dottedIndex], 'features.codex_hooks', true);
488
+ return { changed: true, content: `${stripTrailingNewlines(updatedLines.join('\n'))}\n` };
489
+ }
490
+ const sectionIndex = lines.findIndex((line) => isTomlSectionLine(line, 'features'));
491
+ if (sectionIndex >= 0) {
492
+ const nextSectionIndex = lines.findIndex((line, index) => index > sectionIndex && getTomlLineContent(line).startsWith('[') && getTomlLineContent(line).endsWith(']'));
493
+ const sectionEnd = nextSectionIndex >= 0 ? nextSectionIndex : lines.length;
494
+ const keyIndex = lines.findIndex((line, index) => index > sectionIndex && index < sectionEnd && parseTomlBooleanAssignment(line, 'codex_hooks') !== null);
495
+ if (keyIndex >= 0) {
496
+ if (parseTomlBooleanAssignment(lines[keyIndex], 'codex_hooks') === true) {
497
+ return { changed: false, content: raw };
498
+ }
499
+ const updatedLines = [...lines];
500
+ updatedLines[keyIndex] = updateTomlBooleanAssignment(updatedLines[keyIndex], 'codex_hooks', true);
501
+ return { changed: true, content: `${stripTrailingNewlines(updatedLines.join('\n'))}\n` };
502
+ }
503
+ const updatedLines = [...lines];
504
+ updatedLines.splice(sectionIndex + 1, 0, 'codex_hooks = true');
505
+ return { changed: true, content: `${stripTrailingNewlines(updatedLines.join('\n'))}\n` };
506
+ }
507
+ const prefix = raw.trim().length > 0 ? `${stripTrailingNewlines(raw)}\n\n` : '';
508
+ return {
509
+ changed: true,
510
+ content: `${prefix}[features]\ncodex_hooks = true\n`,
511
+ };
512
+ }
513
+ async function mergeCodexConfig(configPath) {
514
+ await mkdir(dirname(configPath), { recursive: true });
515
+ const raw = await readOptionalUtf8(configPath, '');
516
+ const { changed, content } = ensureCodexHooksEnabled(raw);
517
+ if (!changed) {
518
+ return { changed: false };
519
+ }
520
+ const backupPath = await writeBackupIfPresent(configPath, raw);
521
+ await writeFile(configPath, content, 'utf-8');
522
+ return { changed: true, backupPath };
523
+ }
524
+ async function writeHookMarker(homeDir, marker) {
525
+ const markerPath = getPermissionHookMarkerPath(homeDir, marker.host);
173
526
  await mkdir(dirname(markerPath), { recursive: true });
174
- const installedAt = (options.now ?? new Date()).toISOString();
175
- const marker = {
176
- host: normalizedClient,
177
- scriptPath: targetScriptPath,
527
+ await writeFile(markerPath, JSON.stringify(marker, null, 2) + '\n', 'utf-8');
528
+ return markerPath;
529
+ }
530
+ async function installHookAssetsForHost(client, homeDir, sourceScriptPath) {
531
+ const normalizedClient = normalizeHookHost(client);
532
+ const sharedTargetPath = getPermissionHookScriptPath(homeDir);
533
+ const sharedSourcePath = sourceScriptPath ?? join(repoRootFromModule(), 'scripts', 'pretooluse-dollhouse.sh');
534
+ const sharedStat = statSync(sharedSourcePath);
535
+ if (!sharedStat.isFile()) {
536
+ logger.warn(`[PermissionHooks] Shared hook bridge missing for ${normalizedClient}: ${sharedSourcePath}`);
537
+ throw new Error(`Permission hook source script not found: ${sharedSourcePath}`);
538
+ }
539
+ await copyHookAsset(sharedSourcePath, sharedTargetPath);
540
+ const wrapperTargetPath = getHookWrapperPath(normalizedClient, homeDir);
541
+ if (!wrapperTargetPath) {
542
+ return { scriptPath: sharedTargetPath };
543
+ }
544
+ const wrapperSourcePath = getHookSourcePath(normalizedClient);
545
+ const wrapperStat = statSync(wrapperSourcePath);
546
+ if (!wrapperStat.isFile()) {
547
+ logger.warn(`[PermissionHooks] Wrapper hook script missing for ${normalizedClient}: ${wrapperSourcePath}`);
548
+ throw new Error(`Permission hook wrapper script not found: ${wrapperSourcePath}`);
549
+ }
550
+ await copyHookAsset(wrapperSourcePath, wrapperTargetPath);
551
+ return { scriptPath: wrapperTargetPath };
552
+ }
553
+ async function installClaudeCodePermissionHook(homeDir, installedAt, sourceScriptPath) {
554
+ const host = 'claude-code';
555
+ const { scriptPath } = await installHookAssetsForHost(host, homeDir, sourceScriptPath);
556
+ const settingsPath = getClaudeHookSettingsPath(homeDir);
557
+ const settingsResult = await mergeClaudeSettings(settingsPath, `bash ${scriptPath}`);
558
+ const markerPath = await writeHookMarker(homeDir, {
559
+ host,
560
+ scriptPath,
178
561
  settingsPath,
562
+ configured: true,
563
+ assetsPrepared: true,
179
564
  installedAt,
180
- };
181
- await writeFile(markerPath, JSON.stringify(marker, null, 2) + '\n', 'utf-8');
565
+ });
182
566
  return {
183
567
  supported: true,
184
568
  installed: true,
185
569
  configured: true,
186
- host: normalizedClient,
187
- scriptPath: targetScriptPath,
570
+ assetsPrepared: true,
571
+ host,
572
+ scriptPath,
188
573
  settingsPath,
189
574
  markerPath,
190
575
  backupPath: settingsResult.backupPath,
191
576
  message: 'Installed Claude Code permission hook and updated settings.json.',
192
577
  };
193
578
  }
194
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"permissionHooks.js","sourceRoot":"","sources":["../../src/utils/permissionHooks.ts"],"names":[],"mappings":"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","sourcesContent":["import { accessSync, constants as fsConstants, copyFileSync, existsSync, readFileSync, statSync, writeFileSync } from 'node:fs';\nimport { chmod, mkdir, readFile, writeFile } from 'node:fs/promises';\nimport { dirname, join } from 'node:path';\nimport { fileURLToPath } from 'node:url';\nimport { homedir } from 'node:os';\n\nexport interface PermissionHookMarker {\n  host: string;\n  scriptPath: string;\n  settingsPath: string;\n  installedAt: string;\n}\n\nexport interface PermissionHookStatus {\n  installed: boolean;\n  host?: string;\n  scriptPath?: string;\n  settingsPath?: string;\n}\n\nexport interface InstallPermissionHookResult {\n  supported: boolean;\n  installed: boolean;\n  configured: boolean;\n  host: string;\n  scriptPath?: string;\n  settingsPath?: string;\n  markerPath?: string;\n  backupPath?: string;\n  message: string;\n}\n\nexport interface InstallPermissionHookOptions {\n  homeDir?: string;\n  sourceScriptPath?: string;\n  now?: Date;\n}\n\nfunction repoRootFromModule(): string {\n  const currentFile = fileURLToPath(import.meta.url);\n  return dirname(dirname(dirname(currentFile)));\n}\n\nfunction isMissingFileError(error: unknown): boolean {\n  return Boolean(\n    error\n    && typeof error === 'object'\n    && 'code' in error\n    && (error as { code?: string }).code === 'ENOENT',\n  );\n}\n\nfunction detectIndent(raw: string): number | string {\n  for (const line of raw.split('\\n')) {\n    if (line.length === 0 || line.startsWith('{') || line.startsWith('}')) continue;\n    if (line.startsWith('\\t')) return '\\t';\n    if (line.startsWith(' ')) {\n      const spaces = line.length - line.trimStart().length;\n      if (spaces >= 2) return spaces;\n    }\n  }\n  return 2;\n}\n\nexport function getPermissionHookScriptPath(homeDir = homedir()): string {\n  return join(homeDir, '.dollhouse', 'hooks', 'pretooluse-dollhouse.sh');\n}\n\nexport function getPermissionHookMarkerPath(homeDir = homedir()): string {\n  return join(homeDir, '.dollhouse', 'run', 'hook-installed.json');\n}\n\nexport function getClaudeHookSettingsPath(homeDir = homedir()): string {\n  return join(homeDir, '.claude', 'settings.json');\n}\n\nexport function getPermissionHookStatus(homeDir = homedir()): PermissionHookStatus {\n  const markerPath = getPermissionHookMarkerPath(homeDir);\n  try {\n    const raw = readFileSync(markerPath, 'utf-8');\n    const parsed = JSON.parse(raw) as PermissionHookMarker;\n    if (\n      typeof parsed.host !== 'string' ||\n      typeof parsed.scriptPath !== 'string' ||\n      typeof parsed.settingsPath !== 'string'\n    ) {\n      return { installed: false };\n    }\n\n    if (!existsSync(parsed.scriptPath) || !existsSync(parsed.settingsPath)) {\n      return { installed: false };\n    }\n\n    return {\n      installed: true,\n      host: parsed.host,\n      scriptPath: parsed.scriptPath,\n      settingsPath: parsed.settingsPath,\n    };\n  } catch {\n    return { installed: false };\n  }\n}\n\nfunction normalizeHooksRoot(parsed: Record<string, unknown>): Record<string, unknown[]> {\n  const hooksValue = parsed.hooks;\n  if (!hooksValue || typeof hooksValue !== 'object' || Array.isArray(hooksValue)) {\n    parsed.hooks = {};\n  }\n  return parsed.hooks as Record<string, unknown[]>;\n}\n\nexport function ensureClaudePreToolUseHook(\n  parsed: Record<string, unknown>,\n  command: string,\n): { changed: boolean; parsed: Record<string, unknown> } {\n  const hooksRoot = normalizeHooksRoot(parsed);\n  const existingEntries: Array<Record<string, unknown>> = Array.isArray(hooksRoot.PreToolUse)\n    ? hooksRoot.PreToolUse.filter((entry): entry is Record<string, unknown> => typeof entry === 'object' && entry !== null)\n    : [];\n  hooksRoot.PreToolUse = existingEntries;\n\n  const commandExists = existingEntries.some((entry) => {\n    const hooks = Array.isArray(entry?.hooks) ? entry.hooks as Array<Record<string, unknown>> : [];\n    return hooks.some((hook) => hook?.type === 'command' && hook?.command === command);\n  });\n  if (commandExists) {\n    return { changed: false, parsed };\n  }\n\n  const wildcardEntry = existingEntries.find((entry): entry is Record<string, unknown> =>\n    (entry?.matcher === '*' || entry?.matcher === undefined) && Array.isArray(entry?.hooks),\n  );\n\n  if (wildcardEntry) {\n    const hooks = wildcardEntry.hooks as Array<Record<string, unknown>>;\n    hooks.push({\n      type: 'command',\n      command,\n    });\n  } else {\n    existingEntries.push({\n      matcher: '*',\n      hooks: [\n        {\n          type: 'command',\n          command,\n        },\n      ],\n    });\n  }\n\n  return { changed: true, parsed };\n}\n\nasync function copyHookScript(sourcePath: string, targetPath: string): Promise<boolean> {\n  await mkdir(dirname(targetPath), { recursive: true });\n\n  const sourceRaw = await readFile(sourcePath, 'utf-8');\n\n  let targetRaw: string | undefined;\n  try {\n    targetRaw = await readFile(targetPath, 'utf-8');\n  } catch (error) {\n    if (!isMissingFileError(error)) {\n      throw error;\n    }\n  }\n  const changed = targetRaw === undefined || sourceRaw !== targetRaw;\n\n  if (changed) {\n    copyFileSync(sourcePath, targetPath);\n  } else {\n    accessSync(targetPath, fsConstants.F_OK);\n  }\n\n  await chmod(targetPath, 0o755);\n  return changed;\n}\n\nasync function mergeClaudeSettings(settingsPath: string, command: string): Promise<{ changed: boolean; backupPath?: string }> {\n  await mkdir(dirname(settingsPath), { recursive: true });\n\n  let raw = '{}\\n';\n  try {\n    raw = await readFile(settingsPath, 'utf-8');\n  } catch {\n    raw = '{}\\n';\n  }\n\n  const indent = detectIndent(raw);\n  const parsed = raw.trim().length === 0 ? {} : JSON.parse(raw) as Record<string, unknown>;\n  const { changed, parsed: updated } = ensureClaudePreToolUseHook(parsed, command);\n  if (!changed) {\n    return { changed: false };\n  }\n\n  let backupPath: string | undefined;\n  if (existsSync(settingsPath)) {\n    backupPath = `${settingsPath}.dollhouse.bak`;\n    writeFileSync(backupPath, raw, 'utf-8');\n  }\n\n  await writeFile(settingsPath, JSON.stringify(updated, null, indent) + '\\n', 'utf-8');\n  return { changed: true, backupPath };\n}\n\nexport async function installPermissionHook(\n  client: string,\n  options: InstallPermissionHookOptions = {},\n): Promise<InstallPermissionHookResult> {\n  const normalizedClient = client.normalize('NFC').trim().toLowerCase();\n  if (normalizedClient !== 'claude-code') {\n    return {\n      supported: false,\n      installed: false,\n      configured: false,\n      host: normalizedClient,\n      message: `Automatic permission hook wiring is not yet supported for ${normalizedClient}.`,\n    };\n  }\n\n  const homeDir = options.homeDir ?? homedir();\n  const sourceScriptPath = options.sourceScriptPath\n    ?? join(repoRootFromModule(), 'scripts', 'pretooluse-dollhouse.sh');\n  const targetScriptPath = getPermissionHookScriptPath(homeDir);\n  const settingsPath = getClaudeHookSettingsPath(homeDir);\n  const markerPath = getPermissionHookMarkerPath(homeDir);\n  const command = `bash ${targetScriptPath}`;\n\n  const sourceStat = statSync(sourceScriptPath);\n  if (!sourceStat.isFile()) {\n    throw new Error(`Permission hook source script not found: ${sourceScriptPath}`);\n  }\n\n  await copyHookScript(sourceScriptPath, targetScriptPath);\n  const settingsResult = await mergeClaudeSettings(settingsPath, command);\n\n  await mkdir(dirname(markerPath), { recursive: true });\n  const installedAt = (options.now ?? new Date()).toISOString();\n  const marker: PermissionHookMarker = {\n    host: normalizedClient,\n    scriptPath: targetScriptPath,\n    settingsPath,\n    installedAt,\n  };\n  await writeFile(markerPath, JSON.stringify(marker, null, 2) + '\\n', 'utf-8');\n\n  return {\n    supported: true,\n    installed: true,\n    configured: true,\n    host: normalizedClient,\n    scriptPath: targetScriptPath,\n    settingsPath,\n    markerPath,\n    backupPath: settingsResult.backupPath,\n    message: 'Installed Claude Code permission hook and updated settings.json.',\n  };\n}\n"]}
579
+ async function installVsCodePermissionHook(homeDir, installedAt, sourceScriptPath) {
580
+ const host = 'vscode';
581
+ const { scriptPath } = await installHookAssetsForHost(host, homeDir, sourceScriptPath);
582
+ const settingsPath = getVsCodeHookSettingsPath(homeDir);
583
+ const userSettingsPath = getVsCodeUserSettingsPath(homeDir);
584
+ const hookResult = await mergeVsCodeHookSettings(settingsPath, `bash ${scriptPath}`);
585
+ const userSettingsResult = await mergeVsCodeUserSettings(userSettingsPath);
586
+ const markerPath = await writeHookMarker(homeDir, {
587
+ host,
588
+ scriptPath,
589
+ settingsPath,
590
+ additionalPaths: [userSettingsPath],
591
+ configured: true,
592
+ assetsPrepared: true,
593
+ installedAt,
594
+ });
595
+ return {
596
+ supported: true,
597
+ installed: true,
598
+ configured: true,
599
+ assetsPrepared: true,
600
+ host,
601
+ scriptPath,
602
+ settingsPath,
603
+ additionalPaths: [userSettingsPath],
604
+ markerPath,
605
+ backupPath: hookResult.backupPath ?? userSettingsResult.backupPath,
606
+ message: 'Installed VS Code permission hook and enabled chat.hookFilesLocations for ~/.copilot/hooks.',
607
+ };
608
+ }
609
+ async function installGeminiCliPermissionHook(homeDir, installedAt, sourceScriptPath) {
610
+ const host = 'gemini-cli';
611
+ const { scriptPath } = await installHookAssetsForHost(host, homeDir, sourceScriptPath);
612
+ const settingsPath = getGeminiHookSettingsPath(homeDir);
613
+ const settingsResult = await mergeGeminiSettings(settingsPath, `bash ${scriptPath}`);
614
+ const markerPath = await writeHookMarker(homeDir, {
615
+ host,
616
+ scriptPath,
617
+ settingsPath,
618
+ configured: true,
619
+ assetsPrepared: true,
620
+ installedAt,
621
+ });
622
+ return {
623
+ supported: true,
624
+ installed: true,
625
+ configured: true,
626
+ assetsPrepared: true,
627
+ host,
628
+ scriptPath,
629
+ settingsPath,
630
+ markerPath,
631
+ backupPath: settingsResult.backupPath,
632
+ message: 'Installed Gemini CLI permission hook and updated settings.json.',
633
+ };
634
+ }
635
+ async function installCursorPermissionHook(homeDir, installedAt, sourceScriptPath) {
636
+ const host = 'cursor';
637
+ const { scriptPath } = await installHookAssetsForHost(host, homeDir, sourceScriptPath);
638
+ const settingsPath = getCursorHookSettingsPath(homeDir);
639
+ const settingsResult = await mergeCursorHooks(settingsPath, `bash ${scriptPath}`);
640
+ const markerPath = await writeHookMarker(homeDir, {
641
+ host,
642
+ scriptPath,
643
+ settingsPath,
644
+ configured: true,
645
+ assetsPrepared: true,
646
+ installedAt,
647
+ });
648
+ return {
649
+ supported: true,
650
+ installed: true,
651
+ configured: true,
652
+ assetsPrepared: true,
653
+ host,
654
+ scriptPath,
655
+ settingsPath,
656
+ markerPath,
657
+ backupPath: settingsResult.backupPath,
658
+ message: 'Installed Cursor permission hook and updated hooks.json.',
659
+ };
660
+ }
661
+ async function installWindsurfPermissionHook(homeDir, installedAt, sourceScriptPath) {
662
+ const host = 'windsurf';
663
+ const { scriptPath } = await installHookAssetsForHost(host, homeDir, sourceScriptPath);
664
+ const settingsPath = getWindsurfHookSettingsPath(homeDir);
665
+ const settingsResult = await mergeWindsurfHooks(settingsPath, `bash ${scriptPath}`);
666
+ const markerPath = await writeHookMarker(homeDir, {
667
+ host,
668
+ scriptPath,
669
+ settingsPath,
670
+ configured: true,
671
+ assetsPrepared: true,
672
+ installedAt,
673
+ });
674
+ return {
675
+ supported: true,
676
+ installed: true,
677
+ configured: true,
678
+ assetsPrepared: true,
679
+ host,
680
+ scriptPath,
681
+ settingsPath,
682
+ markerPath,
683
+ backupPath: settingsResult.backupPath,
684
+ message: 'Installed Windsurf permission hooks and updated hooks.json.',
685
+ };
686
+ }
687
+ async function installCodexPermissionHook(homeDir, installedAt, sourceScriptPath) {
688
+ const host = 'codex';
689
+ const { scriptPath } = await installHookAssetsForHost(host, homeDir, sourceScriptPath);
690
+ const settingsPath = getCodexHookSettingsPath(homeDir);
691
+ const configPath = getCodexConfigPath(homeDir);
692
+ const hooksResult = await mergeCodexHooks(settingsPath, `bash ${scriptPath}`);
693
+ const configResult = await mergeCodexConfig(configPath);
694
+ const markerPath = await writeHookMarker(homeDir, {
695
+ host,
696
+ scriptPath,
697
+ settingsPath,
698
+ additionalPaths: [configPath],
699
+ configured: true,
700
+ assetsPrepared: true,
701
+ installedAt,
702
+ });
703
+ return {
704
+ supported: true,
705
+ installed: true,
706
+ configured: true,
707
+ assetsPrepared: true,
708
+ host,
709
+ scriptPath,
710
+ settingsPath,
711
+ additionalPaths: [configPath],
712
+ markerPath,
713
+ backupPath: hooksResult.backupPath ?? configResult.backupPath,
714
+ message: 'Installed Codex Bash permission hook, created hooks.json, and enabled features.codex_hooks in config.toml.',
715
+ };
716
+ }
717
+ async function installManualPermissionHookAssets(normalizedClient, homeDir, installedAt, sourceScriptPath) {
718
+ const { scriptPath } = await installHookAssetsForHost(normalizedClient, homeDir, sourceScriptPath);
719
+ const markerPath = await writeHookMarker(homeDir, {
720
+ host: normalizedClient,
721
+ scriptPath,
722
+ settingsPath: undefined,
723
+ configured: false,
724
+ assetsPrepared: true,
725
+ installedAt,
726
+ });
727
+ return {
728
+ supported: true,
729
+ installed: true,
730
+ configured: false,
731
+ assetsPrepared: true,
732
+ host: normalizedClient,
733
+ scriptPath,
734
+ markerPath,
735
+ message: `Installed Dollhouse permission hook assets for ${normalizedClient}. Finish the client-specific hook registration manually.`,
736
+ };
737
+ }
738
+ export async function installPermissionHook(client, options = {}) {
739
+ const normalizedClient = normalizeHookHost(client);
740
+ const homeDir = options.homeDir ?? homedir();
741
+ const installedAt = (options.now ?? new Date()).toISOString();
742
+ if (normalizedClient === 'claude-code') {
743
+ return installClaudeCodePermissionHook(homeDir, installedAt, options.sourceScriptPath);
744
+ }
745
+ if (normalizedClient === 'vscode') {
746
+ return installVsCodePermissionHook(homeDir, installedAt, options.sourceScriptPath);
747
+ }
748
+ if (normalizedClient === 'gemini-cli') {
749
+ return installGeminiCliPermissionHook(homeDir, installedAt, options.sourceScriptPath);
750
+ }
751
+ if (normalizedClient === 'cursor') {
752
+ return installCursorPermissionHook(homeDir, installedAt, options.sourceScriptPath);
753
+ }
754
+ if (normalizedClient === 'windsurf') {
755
+ return installWindsurfPermissionHook(homeDir, installedAt, options.sourceScriptPath);
756
+ }
757
+ if (normalizedClient === 'codex') {
758
+ return installCodexPermissionHook(homeDir, installedAt, options.sourceScriptPath);
759
+ }
760
+ if (getHookWrapperBasename(normalizedClient)) {
761
+ return installManualPermissionHookAssets(normalizedClient, homeDir, installedAt, options.sourceScriptPath);
762
+ }
763
+ return {
764
+ supported: false,
765
+ installed: false,
766
+ configured: false,
767
+ host: normalizedClient,
768
+ message: `Automatic permission hook wiring is not yet supported for ${normalizedClient}.`,
769
+ };
770
+ }
771
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"permissionHooks.js","sourceRoot":"","sources":["../../src/utils/permissionHooks.ts"],"names":[],"mappings":"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","sourcesContent":["import { existsSync, readFileSync, readdirSync, statSync } from 'node:fs';\nimport { access, chmod, copyFile, mkdir, readFile, readdir, writeFile } from 'node:fs/promises';\nimport { dirname, join } from 'node:path';\nimport { fileURLToPath } from 'node:url';\nimport { homedir, platform } from 'node:os';\nimport { UnicodeValidator } from '../security/validators/unicodeValidator.js';\nimport { logger } from './logger.js';\n\nexport interface PermissionHookMarker {\n  host: string;\n  scriptPath: string;\n  settingsPath?: string;\n  additionalPaths?: string[];\n  configured?: boolean;\n  assetsPrepared?: boolean;\n  installedAt: string;\n}\n\nexport interface PermissionHookStatus {\n  installed: boolean;\n  configured?: boolean;\n  assetsPrepared?: boolean;\n  host?: string;\n  scriptPath?: string;\n  settingsPath?: string;\n  additionalPaths?: string[];\n}\n\nexport interface InstallPermissionHookResult {\n  supported: boolean;\n  installed: boolean;\n  configured: boolean;\n  assetsPrepared?: boolean;\n  host: string;\n  scriptPath?: string;\n  settingsPath?: string;\n  additionalPaths?: string[];\n  markerPath?: string;\n  backupPath?: string;\n  message: string;\n}\n\nexport interface InstallPermissionHookOptions {\n  homeDir?: string;\n  sourceScriptPath?: string;\n  now?: Date;\n}\n\nfunction repoRootFromModule(): string {\n  const currentFile = fileURLToPath(import.meta.url);\n  return dirname(dirname(dirname(currentFile)));\n}\n\nfunction isMissingFileError(error: unknown): boolean {\n  return Boolean(\n    error\n    && typeof error === 'object'\n    && 'code' in error\n    && (error as { code?: string }).code === 'ENOENT',\n  );\n}\n\nfunction detectIndent(raw: string): number | string {\n  for (const line of raw.split('\\n')) {\n    if (line.length === 0 || line.startsWith('{') || line.startsWith('}')) continue;\n    if (line.startsWith('\\t')) return '\\t';\n    if (line.startsWith(' ')) {\n      const spaces = line.length - line.trimStart().length;\n      if (spaces >= 2) return spaces;\n    }\n  }\n  return 2;\n}\n\nexport function getPermissionHookScriptPath(homeDir = homedir()): string {\n  return join(homeDir, '.dollhouse', 'hooks', 'pretooluse-dollhouse.sh');\n}\n\nfunction getPermissionHookRunDir(homeDir = homedir()): string {\n  return join(homeDir, '.dollhouse', 'run');\n}\n\nfunction normalizeHookHost(host: string): string {\n  return UnicodeValidator.normalize(host).normalizedContent.trim().toLowerCase();\n}\n\nfunction isHookMarkerFilename(entry: string): boolean {\n  return entry.startsWith('hook-installed-') && entry.endsWith('.json');\n}\n\nfunction readHostSpecificHookStatus(homeDir: string, host: string): PermissionHookStatus {\n  const normalized = normalizeHookHost(host);\n  const status = readMarkerStatus(getPermissionHookMarkerPath(homeDir, normalized));\n  if (status.installed || status.assetsPrepared) {\n    return status;\n  }\n  if (normalized === 'claude-code') {\n    return readMarkerStatus(getPermissionHookMarkerPath(homeDir));\n  }\n  return { installed: false };\n}\n\nfunction collectHookMarkerPaths(homeDir: string): Set<string> {\n  const markerPaths = new Set<string>([getPermissionHookMarkerPath(homeDir)]);\n  const runDir = getPermissionHookRunDir(homeDir);\n  try {\n    for (const entry of readdirSync(runDir)) {\n      if (isHookMarkerFilename(entry)) {\n        markerPaths.add(join(runDir, entry));\n      }\n    }\n  } catch {\n    // No run dir yet — fall through to default false.\n  }\n  return markerPaths;\n}\n\nasync function collectHookMarkerPathsAsync(homeDir: string): Promise<Set<string>> {\n  const markerPaths = new Set<string>([getPermissionHookMarkerPath(homeDir)]);\n  const runDir = getPermissionHookRunDir(homeDir);\n  try {\n    for (const entry of await readdir(runDir)) {\n      if (isHookMarkerFilename(entry)) {\n        markerPaths.add(join(runDir, entry));\n      }\n    }\n  } catch {\n    // No run dir yet — fall through to default false.\n  }\n  return markerPaths;\n}\n\nfunction summarizeMarkerStatuses(markerPaths: Iterable<string>): PermissionHookStatus {\n  let fallback: PermissionHookStatus = { installed: false };\n  for (const markerPath of markerPaths) {\n    const status = readMarkerStatus(markerPath);\n    if (status.installed) return status;\n    if (!fallback.assetsPrepared && status.assetsPrepared) fallback = status;\n  }\n  return fallback;\n}\n\nfunction getHookWrapperBasename(host: string): string | null {\n  switch (normalizeHookHost(host)) {\n    case 'vscode':\n      return 'pretooluse-vscode.sh';\n    case 'cursor':\n      return 'pretooluse-cursor.sh';\n    case 'windsurf':\n      return 'pretooluse-windsurf.sh';\n    case 'gemini-cli':\n      return 'pretooluse-gemini.sh';\n    case 'codex':\n      return 'pretooluse-codex.sh';\n    default:\n      return null;\n  }\n}\n\nfunction getHookWrapperPath(host: string, homeDir = homedir()): string | null {\n  const basename = getHookWrapperBasename(host);\n  return basename ? join(homeDir, '.dollhouse', 'hooks', basename) : null;\n}\n\nfunction getHookSourcePath(host: string): string {\n  const root = repoRootFromModule();\n  const basename = getHookWrapperBasename(host);\n  return basename ? join(root, 'scripts', basename) : join(root, 'scripts', 'pretooluse-dollhouse.sh');\n}\n\nexport function getPermissionHookMarkerPath(homeDir = homedir(), host?: string): string {\n  if (!host) {\n    return join(getPermissionHookRunDir(homeDir), 'hook-installed.json');\n  }\n  return join(getPermissionHookRunDir(homeDir), `hook-installed-${normalizeHookHost(host)}.json`);\n}\n\nexport function getClaudeHookSettingsPath(homeDir = homedir()): string {\n  return join(homeDir, '.claude', 'settings.json');\n}\n\nexport function getVsCodeHookSettingsPath(homeDir = homedir()): string {\n  return join(homeDir, '.copilot', 'hooks', 'dollhouse-permissions.json');\n}\n\nexport function getVsCodeUserSettingsPath(homeDir = homedir()): string {\n  const currentPlatform = platform();\n  if (currentPlatform === 'darwin') {\n    return join(homeDir, 'Library', 'Application Support', 'Code', 'User', 'settings.json');\n  }\n  if (currentPlatform === 'win32') {\n    const appData = process.env.APPDATA || join(homeDir, 'AppData', 'Roaming');\n    return join(appData, 'Code', 'User', 'settings.json');\n  }\n  return join(homeDir, '.config', 'Code', 'User', 'settings.json');\n}\n\nexport function getGeminiHookSettingsPath(homeDir = homedir()): string {\n  return join(homeDir, '.gemini', 'settings.json');\n}\n\nexport function getCursorHookSettingsPath(homeDir = homedir()): string {\n  return join(homeDir, '.cursor', 'hooks.json');\n}\n\nexport function getWindsurfHookSettingsPath(homeDir = homedir()): string {\n  return join(homeDir, '.codeium', 'windsurf', 'hooks.json');\n}\n\nexport function getCodexHookSettingsPath(homeDir = homedir()): string {\n  return join(homeDir, '.codex', 'hooks.json');\n}\n\nexport function getCodexConfigPath(homeDir = homedir()): string {\n  return join(homeDir, '.codex', 'config.toml');\n}\n\nfunction toPermissionHookStatus(raw: PermissionHookMarker): PermissionHookStatus {\n  if (\n    typeof raw.host !== 'string' ||\n    typeof raw.scriptPath !== 'string'\n  ) {\n    return { installed: false };\n  }\n\n  const scriptReady = existsSync(raw.scriptPath);\n  const settingsReady = !raw.settingsPath || existsSync(raw.settingsPath);\n  const additionalPathsReady = !raw.additionalPaths || raw.additionalPaths.every((path) => existsSync(path));\n  const assetsPrepared = (raw.assetsPrepared ?? raw.configured ?? true) && scriptReady;\n  const configured = (raw.configured ?? true) && scriptReady && settingsReady && additionalPathsReady;\n\n  return {\n    installed: configured,\n    configured,\n    assetsPrepared,\n    host: raw.host,\n    scriptPath: raw.scriptPath,\n    settingsPath: raw.settingsPath,\n    additionalPaths: raw.additionalPaths,\n  };\n}\n\nfunction readMarkerStatus(markerPath: string): PermissionHookStatus {\n  try {\n    const raw = readFileSync(markerPath, 'utf-8');\n    return toPermissionHookStatus(JSON.parse(raw) as PermissionHookMarker);\n  } catch (error) {\n    if (!isMissingFileError(error)) {\n      logger.warn(`[Permissions] Failed to read hook marker at ${markerPath}: ${String(error)}`);\n    }\n    return { installed: false };\n  }\n}\n\nexport function getPermissionHookStatus(homeDir = homedir(), host?: string): PermissionHookStatus {\n  if (host) {\n    return readHostSpecificHookStatus(homeDir, host);\n  }\n\n  return summarizeMarkerStatuses(collectHookMarkerPaths(homeDir));\n}\n\nexport async function getPermissionHookStatusAsync(homeDir = homedir(), host?: string): Promise<PermissionHookStatus> {\n  if (host) {\n    return readHostSpecificHookStatus(homeDir, host);\n  }\n\n  return summarizeMarkerStatuses(await collectHookMarkerPathsAsync(homeDir));\n}\n\nfunction normalizeHooksRoot(parsed: Record<string, unknown>): Record<string, unknown[]> {\n  const hooksValue = parsed.hooks;\n  if (!hooksValue || typeof hooksValue !== 'object' || Array.isArray(hooksValue)) {\n    parsed.hooks = {};\n  }\n  return parsed.hooks as Record<string, unknown[]>;\n}\n\nfunction ensureCommandHook(\n  parsed: Record<string, unknown>,\n  eventName: string,\n  command: string,\n  matcher: string,\n  extraHookFields: Record<string, unknown> = {},\n): { changed: boolean; parsed: Record<string, unknown> } {\n  const hooksRoot = normalizeHooksRoot(parsed);\n  const existingEntries: Array<Record<string, unknown>> = Array.isArray(hooksRoot[eventName])\n    ? hooksRoot[eventName].filter((entry): entry is Record<string, unknown> => typeof entry === 'object' && entry !== null)\n    : [];\n  hooksRoot[eventName] = existingEntries;\n\n  const commandExists = existingEntries.some((entry) => {\n    const hooks = Array.isArray(entry?.hooks) ? entry.hooks as Array<Record<string, unknown>> : [];\n    return hooks.some((hook) => hook?.type === 'command' && hook?.command === command);\n  });\n  if (commandExists) {\n    return { changed: false, parsed };\n  }\n\n  const wildcardEntry = existingEntries.find((entry): entry is Record<string, unknown> =>\n    (entry?.matcher === matcher || entry?.matcher === undefined) && Array.isArray(entry?.hooks),\n  );\n\n  if (wildcardEntry) {\n    const hooks = wildcardEntry.hooks as Array<Record<string, unknown>>;\n    hooks.push({\n      type: 'command',\n      command,\n      ...extraHookFields,\n    });\n  } else {\n    existingEntries.push({\n      matcher,\n      hooks: [\n        {\n          type: 'command',\n          command,\n          ...extraHookFields,\n        },\n      ],\n    });\n  }\n\n  return { changed: true, parsed };\n}\n\nexport function ensureClaudePreToolUseHook(\n  parsed: Record<string, unknown>,\n  command: string,\n): { changed: boolean; parsed: Record<string, unknown> } {\n  return ensureCommandHook(parsed, 'PreToolUse', command, '*');\n}\n\nexport function ensureVsCodePreToolUseHook(\n  parsed: Record<string, unknown>,\n  command: string,\n): { changed: boolean; parsed: Record<string, unknown> } {\n  return ensureCommandHook(parsed, 'PreToolUse', command, '*');\n}\n\nexport function ensureGeminiBeforeToolHook(\n  parsed: Record<string, unknown>,\n  command: string,\n): { changed: boolean; parsed: Record<string, unknown> } {\n  return ensureCommandHook(parsed, 'BeforeTool', command, '.*');\n}\n\nexport function ensureCodexPreToolUseHook(\n  parsed: Record<string, unknown>,\n  command: string,\n): { changed: boolean; parsed: Record<string, unknown> } {\n  return ensureCommandHook(parsed, 'PreToolUse', command, 'Bash', {\n    statusMessage: 'Checking Bash permissions',\n  });\n}\n\nexport function ensureCursorPreToolUseHook(\n  parsed: Record<string, unknown>,\n  command: string,\n): { changed: boolean; parsed: Record<string, unknown> } {\n  if (parsed.version !== 1) {\n    parsed.version = 1;\n  }\n  const hooksRoot = normalizeHooksRoot(parsed);\n  const existingEntries: Array<Record<string, unknown>> = Array.isArray(hooksRoot.preToolUse)\n    ? hooksRoot.preToolUse.filter((entry): entry is Record<string, unknown> => typeof entry === 'object' && entry !== null)\n    : [];\n  hooksRoot.preToolUse = existingEntries;\n\n  const commandExists = existingEntries.some((entry) =>\n    entry.command === command\n    && (entry.type === 'command' || entry.type === undefined),\n  );\n  if (commandExists) {\n    return { changed: false, parsed };\n  }\n\n  existingEntries.push({\n    type: 'command',\n    command,\n    matcher: '.*',\n  });\n\n  return { changed: true, parsed };\n}\n\nexport function ensureWindsurfHooks(\n  parsed: Record<string, unknown>,\n  command: string,\n): { changed: boolean; parsed: Record<string, unknown> } {\n  const hooksRoot = normalizeHooksRoot(parsed);\n  let changed = false;\n\n  const ensureEventHook = (eventName: string) => {\n    const existingEntries: Array<Record<string, unknown>> = Array.isArray(hooksRoot[eventName])\n      ? hooksRoot[eventName].filter((entry): entry is Record<string, unknown> => typeof entry === 'object' && entry !== null)\n      : [];\n    hooksRoot[eventName] = existingEntries;\n\n    const commandExists = existingEntries.some((entry) =>\n      entry.command === command && (entry.type === 'command' || entry.type === undefined),\n    );\n    if (commandExists) {\n      return;\n    }\n\n    existingEntries.push({\n      type: 'command',\n      command,\n    });\n    changed = true;\n  };\n\n  ensureEventHook('pre_run_command');\n  ensureEventHook('pre_mcp_tool_use');\n\n  return { changed, parsed };\n}\n\nasync function copyHookAsset(sourcePath: string, targetPath: string): Promise<boolean> {\n  await mkdir(dirname(targetPath), { recursive: true });\n\n  const sourceRaw = await readFile(sourcePath, 'utf-8');\n\n  let targetRaw: string | undefined;\n  try {\n    targetRaw = await readFile(targetPath, 'utf-8');\n  } catch (error) {\n    if (!isMissingFileError(error)) {\n      throw error;\n    }\n  }\n  const changed = targetRaw === undefined || sourceRaw !== targetRaw;\n\n  if (changed) {\n    await copyFile(sourcePath, targetPath);\n  } else {\n    await access(targetPath);\n  }\n\n  await chmod(targetPath, 0o755);\n  return changed;\n}\n\nasync function readOptionalUtf8(filePath: string, fallback: string): Promise<string> {\n  try {\n    return await readFile(filePath, 'utf-8');\n  } catch (error) {\n    if (isMissingFileError(error)) {\n      return fallback;\n    }\n    throw error;\n  }\n}\n\nasync function writeBackupIfPresent(filePath: string, raw: string): Promise<string | undefined> {\n  if (!existsSync(filePath)) {\n    return undefined;\n  }\n\n  const backupPath = `${filePath}.dollhouse.bak`;\n  await writeFile(backupPath, raw, 'utf-8');\n  return backupPath;\n}\n\nasync function mergeClaudeSettings(settingsPath: string, command: string): Promise<{ changed: boolean; backupPath?: string }> {\n  await mkdir(dirname(settingsPath), { recursive: true });\n\n  const raw = await readOptionalUtf8(settingsPath, '{}\\n');\n\n  const indent = detectIndent(raw);\n  const parsed = raw.trim().length === 0 ? {} : JSON.parse(raw) as Record<string, unknown>;\n  const { changed, parsed: updated } = ensureClaudePreToolUseHook(parsed, command);\n  if (!changed) {\n    return { changed: false };\n  }\n\n  const backupPath = await writeBackupIfPresent(settingsPath, raw);\n\n  await writeFile(settingsPath, JSON.stringify(updated, null, indent) + '\\n', 'utf-8');\n  return { changed: true, backupPath };\n}\n\nasync function mergeVsCodeHookSettings(settingsPath: string, command: string): Promise<{ changed: boolean; backupPath?: string }> {\n  await mkdir(dirname(settingsPath), { recursive: true });\n\n  const raw = await readOptionalUtf8(settingsPath, '{}\\n');\n\n  const indent = detectIndent(raw);\n  const parsed = raw.trim().length === 0 ? {} : JSON.parse(raw) as Record<string, unknown>;\n  const { changed, parsed: updated } = ensureVsCodePreToolUseHook(parsed, command);\n  if (!changed) {\n    return { changed: false };\n  }\n\n  const backupPath = await writeBackupIfPresent(settingsPath, raw);\n\n  await writeFile(settingsPath, JSON.stringify(updated, null, indent) + '\\n', 'utf-8');\n  return { changed: true, backupPath };\n}\n\nasync function mergeVsCodeUserSettings(settingsPath: string): Promise<{ changed: boolean; backupPath?: string }> {\n  await mkdir(dirname(settingsPath), { recursive: true });\n\n  const raw = await readOptionalUtf8(settingsPath, '{}\\n');\n  const indent = detectIndent(raw);\n  const parsed = raw.trim().length === 0 ? {} : JSON.parse(raw) as Record<string, unknown>;\n  const current = parsed['chat.hookFilesLocations'];\n  const locations = (current && typeof current === 'object' && !Array.isArray(current))\n    ? { ...(current as Record<string, unknown>) }\n    : {};\n\n  if (locations['~/.copilot/hooks'] === true) {\n    return { changed: false };\n  }\n\n  locations['~/.copilot/hooks'] = true;\n  parsed['chat.hookFilesLocations'] = locations;\n\n  const backupPath = await writeBackupIfPresent(settingsPath, raw);\n  await writeFile(settingsPath, JSON.stringify(parsed, null, indent) + '\\n', 'utf-8');\n  return { changed: true, backupPath };\n}\n\nasync function mergeGeminiSettings(settingsPath: string, command: string): Promise<{ changed: boolean; backupPath?: string }> {\n  await mkdir(dirname(settingsPath), { recursive: true });\n\n  const raw = await readOptionalUtf8(settingsPath, '{}\\n');\n\n  const indent = detectIndent(raw);\n  const parsed = raw.trim().length === 0 ? {} : JSON.parse(raw) as Record<string, unknown>;\n  const { changed, parsed: updated } = ensureGeminiBeforeToolHook(parsed, command);\n  if (!changed) {\n    return { changed: false };\n  }\n\n  const backupPath = await writeBackupIfPresent(settingsPath, raw);\n\n  await writeFile(settingsPath, JSON.stringify(updated, null, indent) + '\\n', 'utf-8');\n  return { changed: true, backupPath };\n}\n\nasync function mergeCursorHooks(settingsPath: string, command: string): Promise<{ changed: boolean; backupPath?: string }> {\n  await mkdir(dirname(settingsPath), { recursive: true });\n\n  const raw = await readOptionalUtf8(settingsPath, '{\\n  \"version\": 1,\\n  \"hooks\": {}\\n}\\n');\n\n  const indent = detectIndent(raw);\n  const parsed = raw.trim().length === 0 ? {} : JSON.parse(raw) as Record<string, unknown>;\n  const { changed, parsed: updated } = ensureCursorPreToolUseHook(parsed, command);\n  if (!changed) {\n    return { changed: false };\n  }\n\n  const backupPath = await writeBackupIfPresent(settingsPath, raw);\n\n  await writeFile(settingsPath, JSON.stringify(updated, null, indent) + '\\n', 'utf-8');\n  return { changed: true, backupPath };\n}\n\nasync function mergeWindsurfHooks(settingsPath: string, command: string): Promise<{ changed: boolean; backupPath?: string }> {\n  await mkdir(dirname(settingsPath), { recursive: true });\n\n  const raw = await readOptionalUtf8(settingsPath, '{\\n  \"hooks\": {}\\n}\\n');\n\n  const indent = detectIndent(raw);\n  const parsed = raw.trim().length === 0 ? {} : JSON.parse(raw) as Record<string, unknown>;\n  const { changed, parsed: updated } = ensureWindsurfHooks(parsed, command);\n  if (!changed) {\n    return { changed: false };\n  }\n\n  const backupPath = await writeBackupIfPresent(settingsPath, raw);\n\n  await writeFile(settingsPath, JSON.stringify(updated, null, indent) + '\\n', 'utf-8');\n  return { changed: true, backupPath };\n}\n\nasync function mergeCodexHooks(hooksPath: string, command: string): Promise<{ changed: boolean; backupPath?: string }> {\n  await mkdir(dirname(hooksPath), { recursive: true });\n\n  const raw = await readOptionalUtf8(hooksPath, '{}\\n');\n\n  const indent = detectIndent(raw);\n  const parsed = raw.trim().length === 0 ? {} : JSON.parse(raw) as Record<string, unknown>;\n  const { changed, parsed: updated } = ensureCodexPreToolUseHook(parsed, command);\n  if (!changed) {\n    return { changed: false };\n  }\n\n  const backupPath = await writeBackupIfPresent(hooksPath, raw);\n\n  await writeFile(hooksPath, JSON.stringify(updated, null, indent) + '\\n', 'utf-8');\n  return { changed: true, backupPath };\n}\n\nfunction getTomlLineContent(line: string): string {\n  const commentIndex = line.indexOf('#');\n  return (commentIndex >= 0 ? line.slice(0, commentIndex) : line).trim();\n}\n\nfunction isTomlSectionLine(line: string, section: string): boolean {\n  return getTomlLineContent(line) === `[${section}]`;\n}\n\nfunction parseTomlBooleanAssignment(line: string, key: string): boolean | null {\n  const content = getTomlLineContent(line);\n  if (!content.startsWith(`${key} = `)) {\n    return null;\n  }\n  const value = content.slice(`${key} = `.length).trim();\n  if (value === 'true') return true;\n  if (value === 'false') return false;\n  return null;\n}\n\nfunction updateTomlBooleanAssignment(line: string, key: string, nextValue: boolean): string {\n  const commentIndex = line.indexOf('#');\n  const commentSuffix = commentIndex >= 0 ? line.slice(commentIndex) : '';\n  let prefixLength = 0;\n  while (prefixLength < line.length && /\\s/.test(line.charAt(prefixLength))) {\n    prefixLength += 1;\n  }\n  const prefix = line.slice(0, prefixLength);\n  const assignment = `${prefix}${key} = ${nextValue ? 'true' : 'false'}`;\n  const trimmedCommentSuffix = commentSuffix.trimStart();\n  const suffix = trimmedCommentSuffix.length > 0 ? ` ${trimmedCommentSuffix}` : '';\n  return `${assignment}${suffix}`.trimEnd();\n}\n\nfunction stripTrailingNewlines(value: string): string {\n  let end = value.length;\n  while (end > 0 && value.charAt(end - 1) === '\\n') {\n    end -= 1;\n  }\n  return value.slice(0, end);\n}\n\nfunction ensureCodexHooksEnabled(raw: string): { changed: boolean; content: string } {\n  const lines = raw.length > 0 ? raw.split('\\n') : [];\n  const dottedIndex = lines.findIndex((line) => parseTomlBooleanAssignment(line, 'features.codex_hooks') !== null);\n  if (dottedIndex >= 0) {\n    if (parseTomlBooleanAssignment(lines[dottedIndex], 'features.codex_hooks') === true) {\n      return { changed: false, content: raw };\n    }\n    const updatedLines = [...lines];\n    updatedLines[dottedIndex] = updateTomlBooleanAssignment(updatedLines[dottedIndex], 'features.codex_hooks', true);\n    return { changed: true, content: `${stripTrailingNewlines(updatedLines.join('\\n'))}\\n` };\n  }\n\n  const sectionIndex = lines.findIndex((line) => isTomlSectionLine(line, 'features'));\n  if (sectionIndex >= 0) {\n    const nextSectionIndex = lines.findIndex((line, index) => index > sectionIndex && getTomlLineContent(line).startsWith('[') && getTomlLineContent(line).endsWith(']'));\n    const sectionEnd = nextSectionIndex >= 0 ? nextSectionIndex : lines.length;\n    const keyIndex = lines.findIndex((line, index) => index > sectionIndex && index < sectionEnd && parseTomlBooleanAssignment(line, 'codex_hooks') !== null);\n\n    if (keyIndex >= 0) {\n      if (parseTomlBooleanAssignment(lines[keyIndex], 'codex_hooks') === true) {\n        return { changed: false, content: raw };\n      }\n      const updatedLines = [...lines];\n      updatedLines[keyIndex] = updateTomlBooleanAssignment(updatedLines[keyIndex], 'codex_hooks', true);\n      return { changed: true, content: `${stripTrailingNewlines(updatedLines.join('\\n'))}\\n` };\n    }\n\n    const updatedLines = [...lines];\n    updatedLines.splice(sectionIndex + 1, 0, 'codex_hooks = true');\n    return { changed: true, content: `${stripTrailingNewlines(updatedLines.join('\\n'))}\\n` };\n  }\n\n  const prefix = raw.trim().length > 0 ? `${stripTrailingNewlines(raw)}\\n\\n` : '';\n  return {\n    changed: true,\n    content: `${prefix}[features]\\ncodex_hooks = true\\n`,\n  };\n}\n\nasync function mergeCodexConfig(configPath: string): Promise<{ changed: boolean; backupPath?: string }> {\n  await mkdir(dirname(configPath), { recursive: true });\n\n  const raw = await readOptionalUtf8(configPath, '');\n\n  const { changed, content } = ensureCodexHooksEnabled(raw);\n  if (!changed) {\n    return { changed: false };\n  }\n\n  const backupPath = await writeBackupIfPresent(configPath, raw);\n\n  await writeFile(configPath, content, 'utf-8');\n  return { changed: true, backupPath };\n}\n\nasync function writeHookMarker(\n  homeDir: string,\n  marker: PermissionHookMarker,\n): Promise<string> {\n  const markerPath = getPermissionHookMarkerPath(homeDir, marker.host);\n  await mkdir(dirname(markerPath), { recursive: true });\n  await writeFile(markerPath, JSON.stringify(marker, null, 2) + '\\n', 'utf-8');\n  return markerPath;\n}\n\nasync function installHookAssetsForHost(\n  client: string,\n  homeDir: string,\n  sourceScriptPath?: string,\n): Promise<{ scriptPath: string }> {\n  const normalizedClient = normalizeHookHost(client);\n  const sharedTargetPath = getPermissionHookScriptPath(homeDir);\n  const sharedSourcePath = sourceScriptPath ?? join(repoRootFromModule(), 'scripts', 'pretooluse-dollhouse.sh');\n\n  const sharedStat = statSync(sharedSourcePath);\n  if (!sharedStat.isFile()) {\n    logger.warn(`[PermissionHooks] Shared hook bridge missing for ${normalizedClient}: ${sharedSourcePath}`);\n    throw new Error(`Permission hook source script not found: ${sharedSourcePath}`);\n  }\n  await copyHookAsset(sharedSourcePath, sharedTargetPath);\n\n  const wrapperTargetPath = getHookWrapperPath(normalizedClient, homeDir);\n  if (!wrapperTargetPath) {\n    return { scriptPath: sharedTargetPath };\n  }\n\n  const wrapperSourcePath = getHookSourcePath(normalizedClient);\n  const wrapperStat = statSync(wrapperSourcePath);\n  if (!wrapperStat.isFile()) {\n    logger.warn(`[PermissionHooks] Wrapper hook script missing for ${normalizedClient}: ${wrapperSourcePath}`);\n    throw new Error(`Permission hook wrapper script not found: ${wrapperSourcePath}`);\n  }\n  await copyHookAsset(wrapperSourcePath, wrapperTargetPath);\n  return { scriptPath: wrapperTargetPath };\n}\n\nasync function installClaudeCodePermissionHook(\n  homeDir: string,\n  installedAt: string,\n  sourceScriptPath?: string,\n): Promise<InstallPermissionHookResult> {\n  const host = 'claude-code';\n  const { scriptPath } = await installHookAssetsForHost(host, homeDir, sourceScriptPath);\n  const settingsPath = getClaudeHookSettingsPath(homeDir);\n  const settingsResult = await mergeClaudeSettings(settingsPath, `bash ${scriptPath}`);\n  const markerPath = await writeHookMarker(homeDir, {\n    host,\n    scriptPath,\n    settingsPath,\n    configured: true,\n    assetsPrepared: true,\n    installedAt,\n  });\n\n  return {\n    supported: true,\n    installed: true,\n    configured: true,\n    assetsPrepared: true,\n    host,\n    scriptPath,\n    settingsPath,\n    markerPath,\n    backupPath: settingsResult.backupPath,\n    message: 'Installed Claude Code permission hook and updated settings.json.',\n  };\n}\n\nasync function installVsCodePermissionHook(\n  homeDir: string,\n  installedAt: string,\n  sourceScriptPath?: string,\n): Promise<InstallPermissionHookResult> {\n  const host = 'vscode';\n  const { scriptPath } = await installHookAssetsForHost(host, homeDir, sourceScriptPath);\n  const settingsPath = getVsCodeHookSettingsPath(homeDir);\n  const userSettingsPath = getVsCodeUserSettingsPath(homeDir);\n  const hookResult = await mergeVsCodeHookSettings(settingsPath, `bash ${scriptPath}`);\n  const userSettingsResult = await mergeVsCodeUserSettings(userSettingsPath);\n  const markerPath = await writeHookMarker(homeDir, {\n    host,\n    scriptPath,\n    settingsPath,\n    additionalPaths: [userSettingsPath],\n    configured: true,\n    assetsPrepared: true,\n    installedAt,\n  });\n\n  return {\n    supported: true,\n    installed: true,\n    configured: true,\n    assetsPrepared: true,\n    host,\n    scriptPath,\n    settingsPath,\n    additionalPaths: [userSettingsPath],\n    markerPath,\n    backupPath: hookResult.backupPath ?? userSettingsResult.backupPath,\n    message: 'Installed VS Code permission hook and enabled chat.hookFilesLocations for ~/.copilot/hooks.',\n  };\n}\n\nasync function installGeminiCliPermissionHook(\n  homeDir: string,\n  installedAt: string,\n  sourceScriptPath?: string,\n): Promise<InstallPermissionHookResult> {\n  const host = 'gemini-cli';\n  const { scriptPath } = await installHookAssetsForHost(host, homeDir, sourceScriptPath);\n  const settingsPath = getGeminiHookSettingsPath(homeDir);\n  const settingsResult = await mergeGeminiSettings(settingsPath, `bash ${scriptPath}`);\n  const markerPath = await writeHookMarker(homeDir, {\n    host,\n    scriptPath,\n    settingsPath,\n    configured: true,\n    assetsPrepared: true,\n    installedAt,\n  });\n\n  return {\n    supported: true,\n    installed: true,\n    configured: true,\n    assetsPrepared: true,\n    host,\n    scriptPath,\n    settingsPath,\n    markerPath,\n    backupPath: settingsResult.backupPath,\n    message: 'Installed Gemini CLI permission hook and updated settings.json.',\n  };\n}\n\nasync function installCursorPermissionHook(\n  homeDir: string,\n  installedAt: string,\n  sourceScriptPath?: string,\n): Promise<InstallPermissionHookResult> {\n  const host = 'cursor';\n  const { scriptPath } = await installHookAssetsForHost(host, homeDir, sourceScriptPath);\n  const settingsPath = getCursorHookSettingsPath(homeDir);\n  const settingsResult = await mergeCursorHooks(settingsPath, `bash ${scriptPath}`);\n  const markerPath = await writeHookMarker(homeDir, {\n    host,\n    scriptPath,\n    settingsPath,\n    configured: true,\n    assetsPrepared: true,\n    installedAt,\n  });\n\n  return {\n    supported: true,\n    installed: true,\n    configured: true,\n    assetsPrepared: true,\n    host,\n    scriptPath,\n    settingsPath,\n    markerPath,\n    backupPath: settingsResult.backupPath,\n    message: 'Installed Cursor permission hook and updated hooks.json.',\n  };\n}\n\nasync function installWindsurfPermissionHook(\n  homeDir: string,\n  installedAt: string,\n  sourceScriptPath?: string,\n): Promise<InstallPermissionHookResult> {\n  const host = 'windsurf';\n  const { scriptPath } = await installHookAssetsForHost(host, homeDir, sourceScriptPath);\n  const settingsPath = getWindsurfHookSettingsPath(homeDir);\n  const settingsResult = await mergeWindsurfHooks(settingsPath, `bash ${scriptPath}`);\n  const markerPath = await writeHookMarker(homeDir, {\n    host,\n    scriptPath,\n    settingsPath,\n    configured: true,\n    assetsPrepared: true,\n    installedAt,\n  });\n\n  return {\n    supported: true,\n    installed: true,\n    configured: true,\n    assetsPrepared: true,\n    host,\n    scriptPath,\n    settingsPath,\n    markerPath,\n    backupPath: settingsResult.backupPath,\n    message: 'Installed Windsurf permission hooks and updated hooks.json.',\n  };\n}\n\nasync function installCodexPermissionHook(\n  homeDir: string,\n  installedAt: string,\n  sourceScriptPath?: string,\n): Promise<InstallPermissionHookResult> {\n  const host = 'codex';\n  const { scriptPath } = await installHookAssetsForHost(host, homeDir, sourceScriptPath);\n  const settingsPath = getCodexHookSettingsPath(homeDir);\n  const configPath = getCodexConfigPath(homeDir);\n  const hooksResult = await mergeCodexHooks(settingsPath, `bash ${scriptPath}`);\n  const configResult = await mergeCodexConfig(configPath);\n  const markerPath = await writeHookMarker(homeDir, {\n    host,\n    scriptPath,\n    settingsPath,\n    additionalPaths: [configPath],\n    configured: true,\n    assetsPrepared: true,\n    installedAt,\n  });\n\n  return {\n    supported: true,\n    installed: true,\n    configured: true,\n    assetsPrepared: true,\n    host,\n    scriptPath,\n    settingsPath,\n    additionalPaths: [configPath],\n    markerPath,\n    backupPath: hooksResult.backupPath ?? configResult.backupPath,\n    message: 'Installed Codex Bash permission hook, created hooks.json, and enabled features.codex_hooks in config.toml.',\n  };\n}\n\nasync function installManualPermissionHookAssets(\n  normalizedClient: string,\n  homeDir: string,\n  installedAt: string,\n  sourceScriptPath?: string,\n): Promise<InstallPermissionHookResult> {\n  const { scriptPath } = await installHookAssetsForHost(normalizedClient, homeDir, sourceScriptPath);\n  const markerPath = await writeHookMarker(homeDir, {\n    host: normalizedClient,\n    scriptPath,\n    settingsPath: undefined,\n    configured: false,\n    assetsPrepared: true,\n    installedAt,\n  });\n\n  return {\n    supported: true,\n    installed: true,\n    configured: false,\n    assetsPrepared: true,\n    host: normalizedClient,\n    scriptPath,\n    markerPath,\n    message: `Installed Dollhouse permission hook assets for ${normalizedClient}. Finish the client-specific hook registration manually.`,\n  };\n}\n\nexport async function installPermissionHook(\n  client: string,\n  options: InstallPermissionHookOptions = {},\n): Promise<InstallPermissionHookResult> {\n  const normalizedClient = normalizeHookHost(client);\n  const homeDir = options.homeDir ?? homedir();\n  const installedAt = (options.now ?? new Date()).toISOString();\n\n  if (normalizedClient === 'claude-code') {\n    return installClaudeCodePermissionHook(homeDir, installedAt, options.sourceScriptPath);\n  }\n\n  if (normalizedClient === 'vscode') {\n    return installVsCodePermissionHook(homeDir, installedAt, options.sourceScriptPath);\n  }\n\n  if (normalizedClient === 'gemini-cli') {\n    return installGeminiCliPermissionHook(homeDir, installedAt, options.sourceScriptPath);\n  }\n\n  if (normalizedClient === 'cursor') {\n    return installCursorPermissionHook(homeDir, installedAt, options.sourceScriptPath);\n  }\n\n  if (normalizedClient === 'windsurf') {\n    return installWindsurfPermissionHook(homeDir, installedAt, options.sourceScriptPath);\n  }\n\n  if (normalizedClient === 'codex') {\n    return installCodexPermissionHook(homeDir, installedAt, options.sourceScriptPath);\n  }\n\n  if (getHookWrapperBasename(normalizedClient)) {\n    return installManualPermissionHookAssets(normalizedClient, homeDir, installedAt, options.sourceScriptPath);\n  }\n\n  return {\n    supported: false,\n    installed: false,\n    configured: false,\n    host: normalizedClient,\n    message: `Automatic permission hook wiring is not yet supported for ${normalizedClient}.`,\n  };\n}\n"]}