@dollhousemcp/mcp-server 2.0.14 → 2.0.16

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 (35) hide show
  1. package/CHANGELOG.md +12 -0
  2. package/dist/generated/version.d.ts +2 -2
  3. package/dist/generated/version.js +3 -3
  4. package/dist/handlers/ElementCRUDHandler.d.ts.map +1 -1
  5. package/dist/handlers/ElementCRUDHandler.js +7 -3
  6. package/dist/handlers/mcp-aql/MCPAQLHandler.d.ts.map +1 -1
  7. package/dist/handlers/mcp-aql/MCPAQLHandler.js +35 -19
  8. package/dist/handlers/mcp-aql/OperationSchema.d.ts.map +1 -1
  9. package/dist/handlers/mcp-aql/OperationSchema.js +4 -3
  10. package/dist/handlers/mcp-aql/evaluatePermission.d.ts +2 -1
  11. package/dist/handlers/mcp-aql/evaluatePermission.d.ts.map +1 -1
  12. package/dist/handlers/mcp-aql/evaluatePermission.js +22 -11
  13. package/dist/handlers/strategies/BaseActivationStrategy.d.ts.map +1 -1
  14. package/dist/handlers/strategies/BaseActivationStrategy.js +12 -3
  15. package/dist/handlers/strategies/PersonaActivationStrategy.js +2 -2
  16. package/dist/utils/permissionHooks.d.ts +38 -0
  17. package/dist/utils/permissionHooks.d.ts.map +1 -0
  18. package/dist/utils/permissionHooks.js +194 -0
  19. package/dist/web/public/index.html +12 -6
  20. package/dist/web/public/permissions.css +11 -0
  21. package/dist/web/public/permissions.js +43 -12
  22. package/dist/web/public/setup.css +172 -1
  23. package/dist/web/public/setup.js +353 -20
  24. package/dist/web/routes/permissionRoutes.d.ts.map +1 -1
  25. package/dist/web/routes/permissionRoutes.js +64 -15
  26. package/dist/web/routes/setupRoutes.d.ts +3 -0
  27. package/dist/web/routes/setupRoutes.d.ts.map +1 -1
  28. package/dist/web/routes/setupRoutes.js +14 -3
  29. package/package.json +6 -1
  30. package/scripts/pretooluse-codex.sh +6 -0
  31. package/scripts/pretooluse-cursor.sh +6 -0
  32. package/scripts/pretooluse-dollhouse.sh +110 -0
  33. package/scripts/pretooluse-gemini.sh +6 -0
  34. package/scripts/pretooluse-windsurf.sh +6 -0
  35. package/server.json +2 -2
@@ -27,7 +27,7 @@ export interface EvaluatePermissionDeps {
27
27
  permissionPromptLimiter: RateLimiter;
28
28
  classifyTool: (toolName: string, toolInput: Record<string, unknown>) => ToolClassificationResult;
29
29
  evaluateCliToolPolicy: (toolName: string, toolInput: Record<string, unknown>, elements: ActiveElement[]) => CliToolPolicyResult;
30
- getActiveElements: () => Promise<ActiveElement[]>;
30
+ getActiveElements: (sessionId?: string) => Promise<ActiveElement[]>;
31
31
  }
32
32
  /** Known platform identifiers */
33
33
  export declare const SUPPORTED_PLATFORMS: string[];
@@ -49,5 +49,6 @@ export declare function evaluatePermission(params: {
49
49
  tool_name?: unknown;
50
50
  input?: unknown;
51
51
  platform?: unknown;
52
+ session_id?: unknown;
52
53
  }, deps: EvaluatePermissionDeps): Promise<Record<string, unknown>>;
53
54
  //# sourceMappingURL=evaluatePermission.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"evaluatePermission.d.ts","sourceRoot":"","sources":["../../../src/handlers/mcp-aql/evaluatePermission.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;GAcG;AAEH,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,4BAA4B,CAAC;AAC9D,OAAO,KAAK,EAAE,wBAAwB,EAAE,mBAAmB,EAAE,MAAM,kCAAkC,CAAC;AACtG,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,+BAA+B,CAAC;AAEnE,wEAAwE;AACxE,qBAAa,yBAA0B,SAAQ,KAAK;IAClD,SAAgB,KAAK,EAAE,YAAY,GAAG,gBAAgB,GAAG,QAAQ,GAAG,eAAe,CAAC;IACpF,SAAgB,QAAQ,EAAE,MAAM,CAAC;gBAG/B,OAAO,EAAE,MAAM,EACf,KAAK,EAAE,YAAY,GAAG,gBAAgB,GAAG,QAAQ,GAAG,eAAe,EACnE,QAAQ,EAAE,MAAM,EAChB,KAAK,CAAC,EAAE,OAAO;CAOlB;AAED,+CAA+C;AAC/C,MAAM,WAAW,sBAAsB;IACrC,uBAAuB,EAAE,WAAW,CAAC;IACrC,YAAY,EAAE,CAAC,QAAQ,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,KAAK,wBAAwB,CAAC;IACjG,qBAAqB,EAAE,CAAC,QAAQ,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAAE,QAAQ,EAAE,aAAa,EAAE,KAAK,mBAAmB,CAAC;IAChI,iBAAiB,EAAE,MAAM,OAAO,CAAC,aAAa,EAAE,CAAC,CAAC;CACnD;AA2CD,iCAAiC;AACjC,eAAO,MAAM,mBAAmB,UAAkC,CAAC;AAEnE;;;;;GAKG;AACH,wBAAgB,wBAAwB,CACtC,QAAQ,EAAE,OAAO,GAAG,MAAM,GAAG,KAAK,EAClC,QAAQ,EAAE,MAAM,EAChB,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAC/B,MAAM,CAAC,EAAE,MAAM,GACd,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAczB;AAED;;;;;;GAMG;AACH,wBAAsB,kBAAkB,CACtC,MAAM,EAAE;IAAE,SAAS,CAAC,EAAE,OAAO,CAAC;IAAC,KAAK,CAAC,EAAE,OAAO,CAAC;IAAC,QAAQ,CAAC,EAAE,OAAO,CAAA;CAAE,EACpE,IAAI,EAAE,sBAAsB,GAC3B,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC,CA8ClC"}
1
+ {"version":3,"file":"evaluatePermission.d.ts","sourceRoot":"","sources":["../../../src/handlers/mcp-aql/evaluatePermission.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;GAcG;AAEH,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,4BAA4B,CAAC;AAC9D,OAAO,KAAK,EAAE,wBAAwB,EAAE,mBAAmB,EAAE,MAAM,kCAAkC,CAAC;AACtG,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,+BAA+B,CAAC;AAEnE,wEAAwE;AACxE,qBAAa,yBAA0B,SAAQ,KAAK;IAClD,SAAgB,KAAK,EAAE,YAAY,GAAG,gBAAgB,GAAG,QAAQ,GAAG,eAAe,CAAC;IACpF,SAAgB,QAAQ,EAAE,MAAM,CAAC;gBAG/B,OAAO,EAAE,MAAM,EACf,KAAK,EAAE,YAAY,GAAG,gBAAgB,GAAG,QAAQ,GAAG,eAAe,EACnE,QAAQ,EAAE,MAAM,EAChB,KAAK,CAAC,EAAE,OAAO;CAOlB;AAED,+CAA+C;AAC/C,MAAM,WAAW,sBAAsB;IACrC,uBAAuB,EAAE,WAAW,CAAC;IACrC,YAAY,EAAE,CAAC,QAAQ,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,KAAK,wBAAwB,CAAC;IACjG,qBAAqB,EAAE,CAAC,QAAQ,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAAE,QAAQ,EAAE,aAAa,EAAE,KAAK,mBAAmB,CAAC;IAChI,iBAAiB,EAAE,CAAC,SAAS,CAAC,EAAE,MAAM,KAAK,OAAO,CAAC,aAAa,EAAE,CAAC,CAAC;CACrE;AA8CD,iCAAiC;AACjC,eAAO,MAAM,mBAAmB,UAAkC,CAAC;AAiBnE;;;;;GAKG;AACH,wBAAgB,wBAAwB,CACtC,QAAQ,EAAE,OAAO,GAAG,MAAM,GAAG,KAAK,EAClC,QAAQ,EAAE,MAAM,EAChB,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAC/B,MAAM,CAAC,EAAE,MAAM,GACd,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAYzB;AAED;;;;;;GAMG;AACH,wBAAsB,kBAAkB,CACtC,MAAM,EAAE;IAAE,SAAS,CAAC,EAAE,OAAO,CAAC;IAAC,KAAK,CAAC,EAAE,OAAO,CAAC;IAAC,QAAQ,CAAC,EAAE,OAAO,CAAC;IAAC,UAAU,CAAC,EAAE,OAAO,CAAA;CAAE,EAC1F,IAAI,EAAE,sBAAsB,GAC3B,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC,CAiDlC"}
@@ -44,13 +44,14 @@ function formatWindsurf(decision, reason) {
44
44
  function formatCodex(decision, reason) {
45
45
  return { hookSpecificOutput: withReason({ permissionDecision: decision === 'ask' ? 'deny' : decision }, reason) };
46
46
  }
47
- /** Claude Code (default): uses 'decision' with 'message' for ask, 'reason' for deny */
47
+ /** Claude Code (default): uses hookSpecificOutput.permissionDecision for PreToolUse */
48
48
  function formatClaudeCode(decision, reason) {
49
- if (decision === 'allow')
50
- return { decision: 'allow' };
51
- if (decision === 'ask')
52
- return withReason({ decision: 'ask' }, reason, 'message');
53
- return withReason({ decision: 'deny' }, reason);
49
+ return {
50
+ hookSpecificOutput: withReason({
51
+ hookEventName: 'PreToolUse',
52
+ permissionDecision: decision,
53
+ }, reason, 'permissionDecisionReason'),
54
+ };
54
55
  }
55
56
  /** Platform formatter lookup */
56
57
  const platformFormatters = {
@@ -62,6 +63,15 @@ const platformFormatters = {
62
63
  };
63
64
  /** Known platform identifiers */
64
65
  export const SUPPORTED_PLATFORMS = Object.keys(platformFormatters);
66
+ function warnOnUnknownPlatform(platform) {
67
+ void import('../../utils/logger.js')
68
+ .then(({ logger }) => {
69
+ logger.warn(`[evaluatePermission] Unknown platform "${platform}", defaulting to claude_code format. Supported: ${SUPPORTED_PLATFORMS.join(', ')}`);
70
+ })
71
+ .catch((error) => {
72
+ console.warn(`[evaluatePermission] Failed to load logger while handling unknown platform "${platform}".`, error);
73
+ });
74
+ }
65
75
  /**
66
76
  * Format permission evaluation response for platform-specific hook scripts.
67
77
  * Each platform expects a different JSON shape from its hook response.
@@ -77,9 +87,7 @@ export function formatPermissionResponse(decision, platform, _input, reason) {
77
87
  case 'claude_code': return formatClaudeCode(decision, reason);
78
88
  default:
79
89
  // Import lazily to avoid circular dependency at module load time
80
- import('../../utils/logger.js').then(({ logger }) => {
81
- logger.warn(`[evaluatePermission] Unknown platform "${platform}", defaulting to claude_code format. Supported: ${SUPPORTED_PLATFORMS.join(', ')}`);
82
- }).catch(() => { });
90
+ warnOnUnknownPlatform(platform);
83
91
  return formatClaudeCode(decision, reason);
84
92
  }
85
93
  }
@@ -97,6 +105,9 @@ export async function evaluatePermission(params, deps) {
97
105
  ? inputRaw
98
106
  : {};
99
107
  const platform = typeof params.platform === 'string' ? params.platform : 'claude_code';
108
+ const sessionId = typeof params.session_id === 'string' && params.session_id.trim() !== ''
109
+ ? params.session_id
110
+ : undefined;
100
111
  // Rate limit
101
112
  const rateStatus = deps.permissionPromptLimiter.checkLimit();
102
113
  if (!rateStatus.allowed) {
@@ -114,7 +125,7 @@ export async function evaluatePermission(params, deps) {
114
125
  // Stage 2: Element policy evaluation
115
126
  let elements;
116
127
  try {
117
- elements = await deps.getActiveElements();
128
+ elements = await deps.getActiveElements(sessionId);
118
129
  }
119
130
  catch (err) {
120
131
  throw new PermissionEvaluationError(`Failed to fetch active elements for policy evaluation: ${err instanceof Error ? err.message : String(err)}`, 'element_fetch', toolName, err);
@@ -129,4 +140,4 @@ export async function evaluatePermission(params, deps) {
129
140
  // Default: allow
130
141
  return formatPermissionResponse('allow', platform, input);
131
142
  }
132
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"evaluatePermission.js","sourceRoot":"","sources":["../../../src/handlers/mcp-aql/evaluatePermission.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Permission evaluation for PreToolUse hooks across all AI platforms.\n *\n * Provides the `evaluate_permission` MCP-AQL READ operation, enabling\n * Claude Code, Cursor, Gemini CLI, Windsurf, and Codex to use\n * DollhouseMCP as their permission evaluation backend via hooks.\n *\n * Three-stage evaluation pipeline:\n * 1. Rate limiting — prevents abuse\n * 2. Static tool classification — built-in allow/deny rules\n * 3. Element policy evaluation — active element gatekeeper policies\n *\n * Returns platform-specific response formats so each platform's hook\n * script receives the JSON shape it expects.\n */\n\nimport type { RateLimiter } from '../../utils/RateLimiter.js';\nimport type { ToolClassificationResult, CliToolPolicyResult } from './policies/ToolClassification.js';\nimport type { ActiveElement } from './policies/ElementPolicies.js';\n\n/** Error thrown when permission evaluation fails at a specific stage */\nexport class PermissionEvaluationError extends Error {\n  public readonly stage: 'rate_limit' | 'classification' | 'policy' | 'element_fetch';\n  public readonly toolName: string;\n\n  constructor(\n    message: string,\n    stage: 'rate_limit' | 'classification' | 'policy' | 'element_fetch',\n    toolName: string,\n    cause?: unknown,\n  ) {\n    super(message, cause ? { cause } : undefined);\n    this.name = 'PermissionEvaluationError';\n    this.stage = stage;\n    this.toolName = toolName;\n  }\n}\n\n/** Dependencies injected from MCPAQLHandler */\nexport interface EvaluatePermissionDeps {\n  permissionPromptLimiter: RateLimiter;\n  classifyTool: (toolName: string, toolInput: Record<string, unknown>) => ToolClassificationResult;\n  evaluateCliToolPolicy: (toolName: string, toolInput: Record<string, unknown>, elements: ActiveElement[]) => CliToolPolicyResult;\n  getActiveElements: () => Promise<ActiveElement[]>;\n}\n\n/** Optional reason field, only included when reason is provided */\nfunction withReason(obj: Record<string, unknown>, reason?: string, key = 'reason'): Record<string, unknown> {\n  return reason ? { ...obj, [key]: reason } : obj;\n}\n\n/** Gemini: maps 'ask' to 'deny' (no interactive support) */\nfunction formatGemini(decision: string, reason?: string): Record<string, unknown> {\n  return withReason({ decision: decision === 'ask' ? 'deny' : decision }, reason);\n}\n\n/** Cursor: uses 'permission' field instead of 'decision' */\nfunction formatCursor(decision: string, reason?: string): Record<string, unknown> {\n  return withReason({ permission: decision }, reason);\n}\n\n/** Windsurf: uses boolean 'allowed' field */\nfunction formatWindsurf(decision: string, reason?: string): Record<string, unknown> {\n  return withReason({ allowed: decision === 'allow' }, reason);\n}\n\n/** Codex: wraps in hookSpecificOutput, maps 'ask' to 'deny' */\nfunction formatCodex(decision: string, reason?: string): Record<string, unknown> {\n  return { hookSpecificOutput: withReason({ permissionDecision: decision === 'ask' ? 'deny' : decision }, reason) };\n}\n\n/** Claude Code (default): uses 'decision' with 'message' for ask, 'reason' for deny */\nfunction formatClaudeCode(decision: string, reason?: string): Record<string, unknown> {\n  if (decision === 'allow') return { decision: 'allow' };\n  if (decision === 'ask') return withReason({ decision: 'ask' }, reason, 'message');\n  return withReason({ decision: 'deny' }, reason);\n}\n\n/** Platform formatter lookup */\nconst platformFormatters: Record<string, (decision: string, reason?: string) => Record<string, unknown>> = {\n  gemini: formatGemini,\n  cursor: formatCursor,\n  windsurf: formatWindsurf,\n  codex: formatCodex,\n  claude_code: formatClaudeCode,\n};\n\n/** Known platform identifiers */\nexport const SUPPORTED_PLATFORMS = Object.keys(platformFormatters);\n\n/**\n * Format permission evaluation response for platform-specific hook scripts.\n * Each platform expects a different JSON shape from its hook response.\n *\n * Unknown platforms default to claude_code format with a warning log.\n */\nexport function formatPermissionResponse(\n  decision: 'allow' | 'deny' | 'ask',\n  platform: string,\n  _input: Record<string, unknown>,\n  reason?: string,\n): Record<string, unknown> {\n  switch (platform) {\n    case 'gemini': return formatGemini(decision, reason);\n    case 'cursor': return formatCursor(decision, reason);\n    case 'windsurf': return formatWindsurf(decision, reason);\n    case 'codex': return formatCodex(decision, reason);\n    case 'claude_code': return formatClaudeCode(decision, reason);\n    default:\n      // Import lazily to avoid circular dependency at module load time\n      import('../../utils/logger.js').then(({ logger }) => {\n        logger.warn(`[evaluatePermission] Unknown platform \"${platform}\", defaulting to claude_code format. Supported: ${SUPPORTED_PLATFORMS.join(', ')}`);\n      }).catch(() => { /* logger not available */ });\n      return formatClaudeCode(decision, reason);\n  }\n}\n\n/**\n * Evaluate a CLI permission request for PreToolUse hooks.\n *\n * @param params - Tool name, input, and target platform\n * @param deps - Injected dependencies from MCPAQLHandler\n * @returns Platform-formatted permission decision\n */\nexport async function evaluatePermission(\n  params: { tool_name?: unknown; input?: unknown; platform?: unknown },\n  deps: EvaluatePermissionDeps,\n): Promise<Record<string, unknown>> {\n  const toolName = typeof params.tool_name === 'string' ? params.tool_name : '';\n  const inputRaw = params.input;\n  const input = (inputRaw && typeof inputRaw === 'object')\n    ? inputRaw as Record<string, unknown>\n    : {};\n  const platform = typeof params.platform === 'string' ? params.platform : 'claude_code';\n\n  // Rate limit\n  const rateStatus = deps.permissionPromptLimiter.checkLimit();\n  if (!rateStatus.allowed) {\n    return formatPermissionResponse('deny', platform, input, 'Rate limit exceeded');\n  }\n  deps.permissionPromptLimiter.consumeToken();\n\n  // Stage 1: Static classification\n  const classification = deps.classifyTool(toolName, input);\n  if (classification.behavior === 'allow') {\n    return formatPermissionResponse('allow', platform, input);\n  }\n  if (classification.behavior === 'deny') {\n    return formatPermissionResponse('deny', platform, input, classification.reason);\n  }\n\n  // Stage 2: Element policy evaluation\n  let elements: ActiveElement[];\n  try {\n    elements = await deps.getActiveElements();\n  } catch (err) {\n    throw new PermissionEvaluationError(\n      `Failed to fetch active elements for policy evaluation: ${err instanceof Error ? err.message : String(err)}`,\n      'element_fetch', toolName, err,\n    );\n  }\n  const decision = deps.evaluateCliToolPolicy(toolName, input, elements);\n\n  if (decision.behavior === 'deny') {\n    return formatPermissionResponse('deny', platform, input, decision.message);\n  }\n  if (decision.behavior === 'confirm') {\n    return formatPermissionResponse('ask', platform, input,\n      decision.message || 'Requires confirmation per element policy');\n  }\n\n  // Default: allow\n  return formatPermissionResponse('allow', platform, input);\n}\n"]}
143
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"evaluatePermission.js","sourceRoot":"","sources":["../../../src/handlers/mcp-aql/evaluatePermission.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Permission evaluation for PreToolUse hooks across all AI platforms.\n *\n * Provides the `evaluate_permission` MCP-AQL READ operation, enabling\n * Claude Code, Cursor, Gemini CLI, Windsurf, and Codex to use\n * DollhouseMCP as their permission evaluation backend via hooks.\n *\n * Three-stage evaluation pipeline:\n * 1. Rate limiting — prevents abuse\n * 2. Static tool classification — built-in allow/deny rules\n * 3. Element policy evaluation — active element gatekeeper policies\n *\n * Returns platform-specific response formats so each platform's hook\n * script receives the JSON shape it expects.\n */\n\nimport type { RateLimiter } from '../../utils/RateLimiter.js';\nimport type { ToolClassificationResult, CliToolPolicyResult } from './policies/ToolClassification.js';\nimport type { ActiveElement } from './policies/ElementPolicies.js';\n\n/** Error thrown when permission evaluation fails at a specific stage */\nexport class PermissionEvaluationError extends Error {\n  public readonly stage: 'rate_limit' | 'classification' | 'policy' | 'element_fetch';\n  public readonly toolName: string;\n\n  constructor(\n    message: string,\n    stage: 'rate_limit' | 'classification' | 'policy' | 'element_fetch',\n    toolName: string,\n    cause?: unknown,\n  ) {\n    super(message, cause ? { cause } : undefined);\n    this.name = 'PermissionEvaluationError';\n    this.stage = stage;\n    this.toolName = toolName;\n  }\n}\n\n/** Dependencies injected from MCPAQLHandler */\nexport interface EvaluatePermissionDeps {\n  permissionPromptLimiter: RateLimiter;\n  classifyTool: (toolName: string, toolInput: Record<string, unknown>) => ToolClassificationResult;\n  evaluateCliToolPolicy: (toolName: string, toolInput: Record<string, unknown>, elements: ActiveElement[]) => CliToolPolicyResult;\n  getActiveElements: (sessionId?: string) => Promise<ActiveElement[]>;\n}\n\n/** Optional reason field, only included when reason is provided */\nfunction withReason(obj: Record<string, unknown>, reason?: string, key = 'reason'): Record<string, unknown> {\n  return reason ? { ...obj, [key]: reason } : obj;\n}\n\n/** Gemini: maps 'ask' to 'deny' (no interactive support) */\nfunction formatGemini(decision: string, reason?: string): Record<string, unknown> {\n  return withReason({ decision: decision === 'ask' ? 'deny' : decision }, reason);\n}\n\n/** Cursor: uses 'permission' field instead of 'decision' */\nfunction formatCursor(decision: string, reason?: string): Record<string, unknown> {\n  return withReason({ permission: decision }, reason);\n}\n\n/** Windsurf: uses boolean 'allowed' field */\nfunction formatWindsurf(decision: string, reason?: string): Record<string, unknown> {\n  return withReason({ allowed: decision === 'allow' }, reason);\n}\n\n/** Codex: wraps in hookSpecificOutput, maps 'ask' to 'deny' */\nfunction formatCodex(decision: string, reason?: string): Record<string, unknown> {\n  return { hookSpecificOutput: withReason({ permissionDecision: decision === 'ask' ? 'deny' : decision }, reason) };\n}\n\n/** Claude Code (default): uses hookSpecificOutput.permissionDecision for PreToolUse */\nfunction formatClaudeCode(decision: string, reason?: string): Record<string, unknown> {\n  return {\n    hookSpecificOutput: withReason({\n      hookEventName: 'PreToolUse',\n      permissionDecision: decision,\n    }, reason, 'permissionDecisionReason'),\n  };\n}\n\n/** Platform formatter lookup */\nconst platformFormatters: Record<string, (decision: string, reason?: string) => Record<string, unknown>> = {\n  gemini: formatGemini,\n  cursor: formatCursor,\n  windsurf: formatWindsurf,\n  codex: formatCodex,\n  claude_code: formatClaudeCode,\n};\n\n/** Known platform identifiers */\nexport const SUPPORTED_PLATFORMS = Object.keys(platformFormatters);\n\nfunction warnOnUnknownPlatform(platform: string): void {\n  void import('../../utils/logger.js')\n    .then(({ logger }) => {\n      logger.warn(\n        `[evaluatePermission] Unknown platform \"${platform}\", defaulting to claude_code format. Supported: ${SUPPORTED_PLATFORMS.join(', ')}`,\n      );\n    })\n    .catch((error) => {\n      console.warn(\n        `[evaluatePermission] Failed to load logger while handling unknown platform \"${platform}\".`,\n        error,\n      );\n    });\n}\n\n/**\n * Format permission evaluation response for platform-specific hook scripts.\n * Each platform expects a different JSON shape from its hook response.\n *\n * Unknown platforms default to claude_code format with a warning log.\n */\nexport function formatPermissionResponse(\n  decision: 'allow' | 'deny' | 'ask',\n  platform: string,\n  _input: Record<string, unknown>,\n  reason?: string,\n): Record<string, unknown> {\n  switch (platform) {\n    case 'gemini': return formatGemini(decision, reason);\n    case 'cursor': return formatCursor(decision, reason);\n    case 'windsurf': return formatWindsurf(decision, reason);\n    case 'codex': return formatCodex(decision, reason);\n    case 'claude_code': return formatClaudeCode(decision, reason);\n    default:\n      // Import lazily to avoid circular dependency at module load time\n      warnOnUnknownPlatform(platform);\n      return formatClaudeCode(decision, reason);\n  }\n}\n\n/**\n * Evaluate a CLI permission request for PreToolUse hooks.\n *\n * @param params - Tool name, input, and target platform\n * @param deps - Injected dependencies from MCPAQLHandler\n * @returns Platform-formatted permission decision\n */\nexport async function evaluatePermission(\n  params: { tool_name?: unknown; input?: unknown; platform?: unknown; session_id?: unknown },\n  deps: EvaluatePermissionDeps,\n): Promise<Record<string, unknown>> {\n  const toolName = typeof params.tool_name === 'string' ? params.tool_name : '';\n  const inputRaw = params.input;\n  const input = (inputRaw && typeof inputRaw === 'object')\n    ? inputRaw as Record<string, unknown>\n    : {};\n  const platform = typeof params.platform === 'string' ? params.platform : 'claude_code';\n  const sessionId = typeof params.session_id === 'string' && params.session_id.trim() !== ''\n    ? params.session_id\n    : undefined;\n\n  // Rate limit\n  const rateStatus = deps.permissionPromptLimiter.checkLimit();\n  if (!rateStatus.allowed) {\n    return formatPermissionResponse('deny', platform, input, 'Rate limit exceeded');\n  }\n  deps.permissionPromptLimiter.consumeToken();\n\n  // Stage 1: Static classification\n  const classification = deps.classifyTool(toolName, input);\n  if (classification.behavior === 'allow') {\n    return formatPermissionResponse('allow', platform, input);\n  }\n  if (classification.behavior === 'deny') {\n    return formatPermissionResponse('deny', platform, input, classification.reason);\n  }\n\n  // Stage 2: Element policy evaluation\n  let elements: ActiveElement[];\n  try {\n    elements = await deps.getActiveElements(sessionId);\n  } catch (err) {\n    throw new PermissionEvaluationError(\n      `Failed to fetch active elements for policy evaluation: ${err instanceof Error ? err.message : String(err)}`,\n      'element_fetch', toolName, err,\n    );\n  }\n  const decision = deps.evaluateCliToolPolicy(toolName, input, elements);\n\n  if (decision.behavior === 'deny') {\n    return formatPermissionResponse('deny', platform, input, decision.message);\n  }\n  if (decision.behavior === 'confirm') {\n    return formatPermissionResponse('ask', platform, input,\n      decision.message || 'Requires confirmation per element policy');\n  }\n\n  // Default: allow\n  return formatPermissionResponse('allow', platform, input);\n}\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"BaseActivationStrategy.d.ts","sourceRoot":"","sources":["../../../src/handlers/strategies/BaseActivationStrategy.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAIH,OAAO,EAAE,WAAW,EAAE,MAAM,gCAAgC,CAAC;AAE7D;;GAEG;AACH,8BAAsB,sBAAsB;IAC1C;;;;;OAKG;cACa,mBAAmB,CAAC,IAAI,EAAE,MAAM,EAAE,WAAW,EAAE,GAAG,EAAE,GAAG,OAAO,CAAC,GAAG,CAAC;IAInF;;;;OAIG;IACH,SAAS,CAAC,mBAAmB,CAAC,OAAO,EAAE,MAAM,GAAG,WAAW;IAS3D;;;;OAIG;IACH,SAAS,CAAC,qBAAqB,CAAC,OAAO,EAAE,MAAM,GAAG,WAAW;IAS7D;;;;;;;;;;OAUG;IACH,SAAS,CAAC,kBAAkB,CAAC,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,KAAK;IAI/D;;;;OAIG;IACH,SAAS,CAAC,sBAAsB,CAAC,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,WAAW;IAKzE;;;;;;;;OAQG;IACH,SAAS,CAAC,wBAAwB,CAAC,QAAQ,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,GAAG,MAAM;CA4B9E"}
1
+ {"version":3,"file":"BaseActivationStrategy.d.ts","sourceRoot":"","sources":["../../../src/handlers/strategies/BaseActivationStrategy.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAIH,OAAO,EAAE,WAAW,EAAE,MAAM,gCAAgC,CAAC;AAG7D;;GAEG;AACH,8BAAsB,sBAAsB;IAC1C;;;;;OAKG;cACa,mBAAmB,CAAC,IAAI,EAAE,MAAM,EAAE,WAAW,EAAE,GAAG,EAAE,GAAG,OAAO,CAAC,GAAG,CAAC;IAInF;;;;OAIG;IACH,SAAS,CAAC,mBAAmB,CAAC,OAAO,EAAE,MAAM,GAAG,WAAW;IAS3D;;;;OAIG;IACH,SAAS,CAAC,qBAAqB,CAAC,OAAO,EAAE,MAAM,GAAG,WAAW;IAS7D;;;;;;;;;;OAUG;IACH,SAAS,CAAC,kBAAkB,CAAC,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,KAAK;IAI/D;;;;OAIG;IACH,SAAS,CAAC,sBAAsB,CAAC,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,WAAW;IAKzE;;;;;;;;OAQG;IACH,SAAS,CAAC,wBAAwB,CAAC,QAAQ,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,GAAG,MAAM;CAsC9E"}
@@ -7,6 +7,7 @@
7
7
  */
8
8
  import { findElementFlexibly as findHelper } from '../element-crud/helpers.js';
9
9
  import { ElementNotFoundError } from '../../utils/ErrorHandler.js';
10
+ import { getPermissionHookStatus } from '../../utils/permissionHooks.js';
10
11
  /**
11
12
  * Base class with shared utilities for activation strategies
12
13
  */
@@ -83,7 +84,10 @@ export class BaseActivationStrategy {
83
84
  const restrictions = gatekeeper?.externalRestrictions;
84
85
  if (!restrictions)
85
86
  return '';
86
- const parts = ['\n---\n**CLI Restrictions Active:**'];
87
+ const hookStatus = getPermissionHookStatus();
88
+ const parts = [hookStatus.installed
89
+ ? '\n---\n**CLI Policies Loaded:**'
90
+ : '\n---\n**CLI Policies Loaded (Hook Not Detected):**'];
87
91
  if (restrictions.description) {
88
92
  parts.push(`> ${restrictions.description}`);
89
93
  }
@@ -103,8 +107,13 @@ export class BaseActivationStrategy {
103
107
  parts.push(`> Requires approval for: ${requireApproval.join(', ')} risk tools`);
104
108
  }
105
109
  parts.push('> Use `get_effective_cli_policies` to see combined policy state.');
106
- parts.push('> NOTE: These restrictions require the CLI client to be launched with `--permission-prompt-tool` to be enforced.');
110
+ if (hookStatus.installed) {
111
+ parts.push(`> Permission hook detected for ${hookStatus.host ?? 'this client'}. Enforcement depends on using that client configuration.`);
112
+ }
113
+ else {
114
+ parts.push('> No permission hook detected. These policies are not automatically enforced unless the CLI is launched with `--permission-prompt-tool`.', '> Run `open_setup` and reinstall to wire automatic enforcement.');
115
+ }
107
116
  return parts.join('\n');
108
117
  }
109
118
  }
110
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"BaseActivationStrategy.js","sourceRoot":"","sources":["../../../src/handlers/strategies/BaseActivationStrategy.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * BaseActivationStrategy - Shared functionality for all activation strategies\n *\n * Provides common helper methods used by multiple strategy implementations:\n * - Flexible element finding (by name or filename)\n * - Error response formatting\n */\n\nimport { findElementFlexibly as findHelper } from '../element-crud/helpers.js';\nimport { ElementNotFoundError } from '../../utils/ErrorHandler.js';\nimport { MCPResponse } from './ElementActivationStrategy.js';\n\n/**\n * Base class with shared utilities for activation strategies\n */\nexport abstract class BaseActivationStrategy {\n  /**\n   * Find an element by name, supporting both exact display name and filename (slug) matching\n   * @param name - The name or filename to search for\n   * @param elementList - List of elements to search through\n   * @returns The found element or undefined\n   */\n  protected async findElementFlexibly(name: string, elementList: any[]): Promise<any> {\n    return findHelper(name, elementList);\n  }\n\n  /**\n   * Create a standard error response\n   * @param message - The error message\n   * @returns MCP-formatted error response\n   */\n  protected createErrorResponse(message: string): MCPResponse {\n    return {\n      content: [{\n        type: \"text\",\n        text: message\n      }]\n    };\n  }\n\n  /**\n   * Create a standard success response\n   * @param message - The success message\n   * @returns MCP-formatted success response\n   */\n  protected createSuccessResponse(message: string): MCPResponse {\n    return {\n      content: [{\n        type: \"text\",\n        text: message\n      }]\n    };\n  }\n\n  /**\n   * Throw an ElementNotFoundError for missing elements.\n   *\n   * This replaces the previous createNotFoundResponse which returned a success\n   * response with error text. Now we throw to ensure MCP-AQL returns success=false.\n   *\n   * @param name - The name of the element that wasn't found\n   * @param type - The type of element (for error message)\n   * @throws {ElementNotFoundError} Always throws\n   * @see Issue #275 - Handlers return success=true for missing elements\n   */\n  protected throwNotFoundError(name: string, type: string): never {\n    throw new ElementNotFoundError(type, name);\n  }\n\n  /**\n   * @deprecated Use throwNotFoundError instead to ensure proper error handling.\n   * This method returns a success response which causes MCP-AQL to return success=true.\n   * @see Issue #275 - Handlers return success=true for missing elements\n   */\n  protected createNotFoundResponse(name: string, type: string): MCPResponse {\n    // Keep for backward compatibility but strategies should migrate to throwNotFoundError\n    return this.createErrorResponse(`❌ ${type} '${name}' not found`);\n  }\n\n  /**\n   * Format a fail-safe warning for elements with CLI external restrictions.\n   * Appended to activation text so users are aware of restrictions even when\n   * permission_prompt is not available (non-Claude-Code clients).\n   *\n   * @param metadata - Element metadata (may contain gatekeeper.externalRestrictions)\n   * @returns Warning text or empty string if no restrictions\n   * @see Issue #642 — Fail-safe enforcement for externalRestrictions\n   */\n  protected formatRestrictionWarning(metadata: Record<string, unknown>): string {\n    const gatekeeper = metadata?.gatekeeper as Record<string, unknown> | undefined;\n    const restrictions = gatekeeper?.externalRestrictions as Record<string, unknown> | undefined;\n    if (!restrictions) return '';\n\n    const parts: string[] = ['\\n---\\n**CLI Restrictions Active:**'];\n    if (restrictions.description) {\n      parts.push(`> ${restrictions.description}`);\n    }\n    const denyPatterns = restrictions.denyPatterns as string[] | undefined;\n    if (denyPatterns?.length) {\n      const shown = denyPatterns.slice(0, 5).join(', ');\n      parts.push(`> Denied: ${shown}${denyPatterns.length > 5 ? '...' : ''}`);\n    }\n    const allowPatterns = restrictions.allowPatterns as string[] | undefined;\n    if (allowPatterns?.length) {\n      const shown = allowPatterns.slice(0, 5).join(', ');\n      parts.push(`> Allowed only: ${shown}${allowPatterns.length > 5 ? '...' : ''}`);\n    }\n    const approvalPolicy = restrictions.approvalPolicy as Record<string, unknown> | undefined;\n    const requireApproval = approvalPolicy?.requireApproval as string[] | undefined;\n    if (requireApproval?.length) {\n      parts.push(`> Requires approval for: ${requireApproval.join(', ')} risk tools`);\n    }\n    parts.push('> Use `get_effective_cli_policies` to see combined policy state.');\n    parts.push('> NOTE: These restrictions require the CLI client to be launched with `--permission-prompt-tool` to be enforced.');\n    return parts.join('\\n');\n  }\n}\n"]}
119
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"BaseActivationStrategy.js","sourceRoot":"","sources":["../../../src/handlers/strategies/BaseActivationStrategy.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * BaseActivationStrategy - Shared functionality for all activation strategies\n *\n * Provides common helper methods used by multiple strategy implementations:\n * - Flexible element finding (by name or filename)\n * - Error response formatting\n */\n\nimport { findElementFlexibly as findHelper } from '../element-crud/helpers.js';\nimport { ElementNotFoundError } from '../../utils/ErrorHandler.js';\nimport { MCPResponse } from './ElementActivationStrategy.js';\nimport { getPermissionHookStatus } from '../../utils/permissionHooks.js';\n\n/**\n * Base class with shared utilities for activation strategies\n */\nexport abstract class BaseActivationStrategy {\n  /**\n   * Find an element by name, supporting both exact display name and filename (slug) matching\n   * @param name - The name or filename to search for\n   * @param elementList - List of elements to search through\n   * @returns The found element or undefined\n   */\n  protected async findElementFlexibly(name: string, elementList: any[]): Promise<any> {\n    return findHelper(name, elementList);\n  }\n\n  /**\n   * Create a standard error response\n   * @param message - The error message\n   * @returns MCP-formatted error response\n   */\n  protected createErrorResponse(message: string): MCPResponse {\n    return {\n      content: [{\n        type: \"text\",\n        text: message\n      }]\n    };\n  }\n\n  /**\n   * Create a standard success response\n   * @param message - The success message\n   * @returns MCP-formatted success response\n   */\n  protected createSuccessResponse(message: string): MCPResponse {\n    return {\n      content: [{\n        type: \"text\",\n        text: message\n      }]\n    };\n  }\n\n  /**\n   * Throw an ElementNotFoundError for missing elements.\n   *\n   * This replaces the previous createNotFoundResponse which returned a success\n   * response with error text. Now we throw to ensure MCP-AQL returns success=false.\n   *\n   * @param name - The name of the element that wasn't found\n   * @param type - The type of element (for error message)\n   * @throws {ElementNotFoundError} Always throws\n   * @see Issue #275 - Handlers return success=true for missing elements\n   */\n  protected throwNotFoundError(name: string, type: string): never {\n    throw new ElementNotFoundError(type, name);\n  }\n\n  /**\n   * @deprecated Use throwNotFoundError instead to ensure proper error handling.\n   * This method returns a success response which causes MCP-AQL to return success=true.\n   * @see Issue #275 - Handlers return success=true for missing elements\n   */\n  protected createNotFoundResponse(name: string, type: string): MCPResponse {\n    // Keep for backward compatibility but strategies should migrate to throwNotFoundError\n    return this.createErrorResponse(`❌ ${type} '${name}' not found`);\n  }\n\n  /**\n   * Format a fail-safe warning for elements with CLI external restrictions.\n   * Appended to activation text so users are aware of restrictions even when\n   * permission_prompt is not available (non-Claude-Code clients).\n   *\n   * @param metadata - Element metadata (may contain gatekeeper.externalRestrictions)\n   * @returns Warning text or empty string if no restrictions\n   * @see Issue #642 — Fail-safe enforcement for externalRestrictions\n   */\n  protected formatRestrictionWarning(metadata: Record<string, unknown>): string {\n    const gatekeeper = metadata?.gatekeeper as Record<string, unknown> | undefined;\n    const restrictions = gatekeeper?.externalRestrictions as Record<string, unknown> | undefined;\n    if (!restrictions) return '';\n\n    const hookStatus = getPermissionHookStatus();\n    const parts: string[] = [hookStatus.installed\n      ? '\\n---\\n**CLI Policies Loaded:**'\n      : '\\n---\\n**CLI Policies Loaded (Hook Not Detected):**'];\n    if (restrictions.description) {\n      parts.push(`> ${restrictions.description}`);\n    }\n    const denyPatterns = restrictions.denyPatterns as string[] | undefined;\n    if (denyPatterns?.length) {\n      const shown = denyPatterns.slice(0, 5).join(', ');\n      parts.push(`> Denied: ${shown}${denyPatterns.length > 5 ? '...' : ''}`);\n    }\n    const allowPatterns = restrictions.allowPatterns as string[] | undefined;\n    if (allowPatterns?.length) {\n      const shown = allowPatterns.slice(0, 5).join(', ');\n      parts.push(`> Allowed only: ${shown}${allowPatterns.length > 5 ? '...' : ''}`);\n    }\n    const approvalPolicy = restrictions.approvalPolicy as Record<string, unknown> | undefined;\n    const requireApproval = approvalPolicy?.requireApproval as string[] | undefined;\n    if (requireApproval?.length) {\n      parts.push(`> Requires approval for: ${requireApproval.join(', ')} risk tools`);\n    }\n    parts.push('> Use `get_effective_cli_policies` to see combined policy state.');\n    if (hookStatus.installed) {\n      parts.push(`> Permission hook detected for ${hookStatus.host ?? 'this client'}. Enforcement depends on using that client configuration.`);\n    } else {\n      parts.push(\n        '> No permission hook detected. These policies are not automatically enforced unless the CLI is launched with `--permission-prompt-tool`.',\n        '> Run `open_setup` and reinstall to wire automatic enforcement.',\n      );\n    }\n    return parts.join('\\n');\n  }\n}\n"]}
@@ -118,7 +118,7 @@ export class PersonaActivationStrategy extends BaseActivationStrategy {
118
118
  const r = gk.externalRestrictions;
119
119
  return ` **${p.metadata.name}**: ${r.description}`;
120
120
  }).join('\n');
121
- text += `\n\n**Active CLI Restrictions:**\n${summary}\n> Use \`get_effective_cli_policies\` for full details.`;
121
+ text += `\n\n**Loaded CLI Restrictions:**\n${summary}\n> Use \`get_effective_cli_policies\` for full details.`;
122
122
  }
123
123
  return {
124
124
  content: [{
@@ -156,4 +156,4 @@ export class PersonaActivationStrategy extends BaseActivationStrategy {
156
156
  };
157
157
  }
158
158
  }
159
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"PersonaActivationStrategy.js","sourceRoot":"","sources":["../../../src/handlers/strategies/PersonaActivationStrategy.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * PersonaActivationStrategy - Strategy for persona element activation\n *\n * Handles activation, deactivation, and status tracking for persona elements.\n * Uses the PersonaManager's unique API and PersonaIndicatorService for formatting.\n */\n\nimport { PersonaManager } from '../../persona/PersonaManager.js';\nimport { PersonaIndicatorService } from '../../services/PersonaIndicatorService.js';\nimport { ElementNotFoundError } from '../../utils/ErrorHandler.js';\nimport { BaseActivationStrategy } from './BaseActivationStrategy.js';\nimport { ElementActivationStrategy, MCPResponse } from './ElementActivationStrategy.js';\n\nexport class PersonaActivationStrategy extends BaseActivationStrategy implements ElementActivationStrategy {\n  constructor(\n    private readonly personaManager: PersonaManager,\n    private readonly indicatorService: PersonaIndicatorService\n  ) {\n    super();\n  }\n\n  /**\n   * Get the persona indicator prefix\n   */\n  private getPersonaIndicator(): string {\n    return this.indicatorService.getPersonaIndicator();\n  }\n\n  /**\n   * Activate a persona\n   * Extracted from ElementCRUDHandler.ts lines 161-182\n   */\n  async activate(name: string): Promise<MCPResponse> {\n    const result = await this.personaManager.activatePersona(name);\n\n    if (!result.success || !result.persona) {\n      return {\n        content: [{\n          type: \"text\",\n          text: `${this.getPersonaIndicator()}❌ ${result.message}`\n        }]\n      };\n    }\n\n    const persona = result.persona;\n    const instructions = persona.instructions?.trim() || persona.content?.trim() || 'No instructions provided.';\n    const referenceContent = persona.instructions?.trim() ? persona.content?.trim() : '';\n\n    let text = `${this.getPersonaIndicator()}Persona Activated: **${persona.metadata.name}**\\n\\n${persona.metadata.description}\\n\\n**Instructions:**\\n${instructions}`;\n    if (referenceContent) {\n      text += `\\n\\n**Reference:**\\n${referenceContent}`;\n    }\n\n    // Issue #642: Fail-safe warning for CLI restrictions\n    const restrictionWarning = this.formatRestrictionWarning(persona.metadata as unknown as Record<string, unknown>);\n    if (restrictionWarning) {\n      text += restrictionWarning;\n    }\n\n    return {\n      content: [{\n        type: \"text\",\n        text\n      }]\n    };\n  }\n\n  /**\n   * Deactivate a specific persona\n   * Issue #281: Updated to support multiple active personas\n   *\n   * @throws {ElementNotFoundError} When named persona does not exist\n   * @throws {Error} When name parameter is missing\n   * @see Issue #275 - Handlers return success=true for missing elements\n   */\n  async deactivate(name: string): Promise<MCPResponse> {\n    // Issue #275: Require name parameter for consistent error handling\n    if (!name || name === '') {\n      throw new Error('Name parameter is required for deactivate operation');\n    }\n\n    // Issue #275: Verify the named persona exists before deactivating\n    const persona = this.personaManager.findPersona(name);\n    if (!persona) {\n      throw new ElementNotFoundError('Persona', name);\n    }\n\n    // Issue #281: Pass the name to deactivate the specific persona\n    const result = this.personaManager.deactivatePersona(name);\n    const indicator = this.getPersonaIndicator();\n\n    if (!result.success) {\n      return {\n        content: [{\n          type: \"text\",\n          text: `${indicator}❌ ${result.message}`\n        }]\n      };\n    }\n\n    return {\n      content: [{\n        type: \"text\",\n        text: `${indicator}✅ ${result.message}`\n      }]\n    };\n  }\n\n  /**\n   * Get all active personas\n   * Issue #281: Updated to show all active personas (supports multiple)\n   */\n  async getActiveElements(): Promise<MCPResponse> {\n    const activePersonas = this.personaManager.getActivePersonas();\n    const indicator = this.getPersonaIndicator();\n\n    if (activePersonas.length === 0) {\n      return {\n        content: [{\n          type: \"text\",\n          text: `${indicator}No personas are currently active.`\n        }]\n      };\n    }\n\n    const personaList = activePersonas.map(p =>\n      `🔹 **${p.metadata.name}** (${p.unique_id})\\n   ${p.metadata.description}\\n   📁 ${p.metadata.category || 'general'} | 🎭 ${p.metadata.author || 'Unknown'}`\n    ).join('\\n\\n');\n\n    const header = activePersonas.length === 1\n      ? `${indicator}Active Persona:`\n      : `${indicator}Active Personas (${activePersonas.length}):`;\n\n    let text = `${header}\\n\\n${personaList}`;\n\n    // Issue #642: Restriction summary for active personas with externalRestrictions\n    const restrictedPersonas = activePersonas.filter(\n      p => (p.metadata as unknown as Record<string, unknown>)?.gatekeeper &&\n        ((p.metadata as unknown as Record<string, unknown>).gatekeeper as Record<string, unknown>)?.externalRestrictions\n    );\n    if (restrictedPersonas.length > 0) {\n      const summary = restrictedPersonas.map(p => {\n        const gk = (p.metadata as unknown as Record<string, unknown>).gatekeeper as Record<string, unknown>;\n        const r = gk.externalRestrictions as Record<string, unknown>;\n        return `  **${p.metadata.name}**: ${r.description}`;\n      }).join('\\n');\n      text += `\\n\\n**Active CLI Restrictions:**\\n${summary}\\n> Use \\`get_effective_cli_policies\\` for full details.`;\n    }\n\n    return {\n      content: [{\n        type: \"text\",\n        text\n      }]\n    };\n  }\n\n  /**\n   * Get detailed information about a persona\n   * Extracted from ElementCRUDHandler.ts lines 620-648\n   *\n   * @throws {ElementNotFoundError} When persona does not exist\n   * @see Issue #275 - Handlers return success=true for missing elements\n   */\n  async getElementDetails(name: string): Promise<MCPResponse> {\n    const persona = this.personaManager.findPersona(name);\n    const indicator = this.getPersonaIndicator();\n\n    if (!persona) {\n      throw new ElementNotFoundError('Persona', name);\n    }\n\n    const triggers = persona.metadata.triggers?.join(', ') || 'None';\n    const content = persona.content?.trim() || 'No instructions provided.';\n\n    return {\n      content: [{\n        type: \"text\",\n        text: `${indicator}📋 **${persona.metadata.name}** Details\\n\\n` +\n          `**Description:** ${persona.metadata.description}\\n` +\n          `**File:** ${persona.filename}\\n` +\n          `**Version:** ${persona.metadata.version || '1.0'}\\n` +\n          `**Author:** ${persona.metadata.author || 'Unknown'}\\n` +\n          `**Triggers:** ${triggers}\\n\\n` +\n          `**Full Instructions:**\\n\\`\\`\\`\\n${content}\\n\\`\\`\\``\n      }]\n    };\n  }\n}\n"]}
159
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"PersonaActivationStrategy.js","sourceRoot":"","sources":["../../../src/handlers/strategies/PersonaActivationStrategy.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * PersonaActivationStrategy - Strategy for persona element activation\n *\n * Handles activation, deactivation, and status tracking for persona elements.\n * Uses the PersonaManager's unique API and PersonaIndicatorService for formatting.\n */\n\nimport { PersonaManager } from '../../persona/PersonaManager.js';\nimport { PersonaIndicatorService } from '../../services/PersonaIndicatorService.js';\nimport { ElementNotFoundError } from '../../utils/ErrorHandler.js';\nimport { BaseActivationStrategy } from './BaseActivationStrategy.js';\nimport { ElementActivationStrategy, MCPResponse } from './ElementActivationStrategy.js';\n\nexport class PersonaActivationStrategy extends BaseActivationStrategy implements ElementActivationStrategy {\n  constructor(\n    private readonly personaManager: PersonaManager,\n    private readonly indicatorService: PersonaIndicatorService\n  ) {\n    super();\n  }\n\n  /**\n   * Get the persona indicator prefix\n   */\n  private getPersonaIndicator(): string {\n    return this.indicatorService.getPersonaIndicator();\n  }\n\n  /**\n   * Activate a persona\n   * Extracted from ElementCRUDHandler.ts lines 161-182\n   */\n  async activate(name: string): Promise<MCPResponse> {\n    const result = await this.personaManager.activatePersona(name);\n\n    if (!result.success || !result.persona) {\n      return {\n        content: [{\n          type: \"text\",\n          text: `${this.getPersonaIndicator()}❌ ${result.message}`\n        }]\n      };\n    }\n\n    const persona = result.persona;\n    const instructions = persona.instructions?.trim() || persona.content?.trim() || 'No instructions provided.';\n    const referenceContent = persona.instructions?.trim() ? persona.content?.trim() : '';\n\n    let text = `${this.getPersonaIndicator()}Persona Activated: **${persona.metadata.name}**\\n\\n${persona.metadata.description}\\n\\n**Instructions:**\\n${instructions}`;\n    if (referenceContent) {\n      text += `\\n\\n**Reference:**\\n${referenceContent}`;\n    }\n\n    // Issue #642: Fail-safe warning for CLI restrictions\n    const restrictionWarning = this.formatRestrictionWarning(persona.metadata as unknown as Record<string, unknown>);\n    if (restrictionWarning) {\n      text += restrictionWarning;\n    }\n\n    return {\n      content: [{\n        type: \"text\",\n        text\n      }]\n    };\n  }\n\n  /**\n   * Deactivate a specific persona\n   * Issue #281: Updated to support multiple active personas\n   *\n   * @throws {ElementNotFoundError} When named persona does not exist\n   * @throws {Error} When name parameter is missing\n   * @see Issue #275 - Handlers return success=true for missing elements\n   */\n  async deactivate(name: string): Promise<MCPResponse> {\n    // Issue #275: Require name parameter for consistent error handling\n    if (!name || name === '') {\n      throw new Error('Name parameter is required for deactivate operation');\n    }\n\n    // Issue #275: Verify the named persona exists before deactivating\n    const persona = this.personaManager.findPersona(name);\n    if (!persona) {\n      throw new ElementNotFoundError('Persona', name);\n    }\n\n    // Issue #281: Pass the name to deactivate the specific persona\n    const result = this.personaManager.deactivatePersona(name);\n    const indicator = this.getPersonaIndicator();\n\n    if (!result.success) {\n      return {\n        content: [{\n          type: \"text\",\n          text: `${indicator}❌ ${result.message}`\n        }]\n      };\n    }\n\n    return {\n      content: [{\n        type: \"text\",\n        text: `${indicator}✅ ${result.message}`\n      }]\n    };\n  }\n\n  /**\n   * Get all active personas\n   * Issue #281: Updated to show all active personas (supports multiple)\n   */\n  async getActiveElements(): Promise<MCPResponse> {\n    const activePersonas = this.personaManager.getActivePersonas();\n    const indicator = this.getPersonaIndicator();\n\n    if (activePersonas.length === 0) {\n      return {\n        content: [{\n          type: \"text\",\n          text: `${indicator}No personas are currently active.`\n        }]\n      };\n    }\n\n    const personaList = activePersonas.map(p =>\n      `🔹 **${p.metadata.name}** (${p.unique_id})\\n   ${p.metadata.description}\\n   📁 ${p.metadata.category || 'general'} | 🎭 ${p.metadata.author || 'Unknown'}`\n    ).join('\\n\\n');\n\n    const header = activePersonas.length === 1\n      ? `${indicator}Active Persona:`\n      : `${indicator}Active Personas (${activePersonas.length}):`;\n\n    let text = `${header}\\n\\n${personaList}`;\n\n    // Issue #642: Restriction summary for active personas with externalRestrictions\n    const restrictedPersonas = activePersonas.filter(\n      p => (p.metadata as unknown as Record<string, unknown>)?.gatekeeper &&\n        ((p.metadata as unknown as Record<string, unknown>).gatekeeper as Record<string, unknown>)?.externalRestrictions\n    );\n    if (restrictedPersonas.length > 0) {\n      const summary = restrictedPersonas.map(p => {\n        const gk = (p.metadata as unknown as Record<string, unknown>).gatekeeper as Record<string, unknown>;\n        const r = gk.externalRestrictions as Record<string, unknown>;\n        return `  **${p.metadata.name}**: ${r.description}`;\n      }).join('\\n');\n      text += `\\n\\n**Loaded CLI Restrictions:**\\n${summary}\\n> Use \\`get_effective_cli_policies\\` for full details.`;\n    }\n\n    return {\n      content: [{\n        type: \"text\",\n        text\n      }]\n    };\n  }\n\n  /**\n   * Get detailed information about a persona\n   * Extracted from ElementCRUDHandler.ts lines 620-648\n   *\n   * @throws {ElementNotFoundError} When persona does not exist\n   * @see Issue #275 - Handlers return success=true for missing elements\n   */\n  async getElementDetails(name: string): Promise<MCPResponse> {\n    const persona = this.personaManager.findPersona(name);\n    const indicator = this.getPersonaIndicator();\n\n    if (!persona) {\n      throw new ElementNotFoundError('Persona', name);\n    }\n\n    const triggers = persona.metadata.triggers?.join(', ') || 'None';\n    const content = persona.content?.trim() || 'No instructions provided.';\n\n    return {\n      content: [{\n        type: \"text\",\n        text: `${indicator}📋 **${persona.metadata.name}** Details\\n\\n` +\n          `**Description:** ${persona.metadata.description}\\n` +\n          `**File:** ${persona.filename}\\n` +\n          `**Version:** ${persona.metadata.version || '1.0'}\\n` +\n          `**Author:** ${persona.metadata.author || 'Unknown'}\\n` +\n          `**Triggers:** ${triggers}\\n\\n` +\n          `**Full Instructions:**\\n\\`\\`\\`\\n${content}\\n\\`\\`\\``\n      }]\n    };\n  }\n}\n"]}
@@ -0,0 +1,38 @@
1
+ export interface PermissionHookMarker {
2
+ host: string;
3
+ scriptPath: string;
4
+ settingsPath: string;
5
+ installedAt: string;
6
+ }
7
+ export interface PermissionHookStatus {
8
+ installed: boolean;
9
+ host?: string;
10
+ scriptPath?: string;
11
+ settingsPath?: string;
12
+ }
13
+ export interface InstallPermissionHookResult {
14
+ supported: boolean;
15
+ installed: boolean;
16
+ configured: boolean;
17
+ host: string;
18
+ scriptPath?: string;
19
+ settingsPath?: string;
20
+ markerPath?: string;
21
+ backupPath?: string;
22
+ message: string;
23
+ }
24
+ export interface InstallPermissionHookOptions {
25
+ homeDir?: string;
26
+ sourceScriptPath?: string;
27
+ now?: Date;
28
+ }
29
+ export declare function getPermissionHookScriptPath(homeDir?: string): string;
30
+ export declare function getPermissionHookMarkerPath(homeDir?: string): string;
31
+ export declare function getClaudeHookSettingsPath(homeDir?: string): string;
32
+ export declare function getPermissionHookStatus(homeDir?: string): PermissionHookStatus;
33
+ export declare function ensureClaudePreToolUseHook(parsed: Record<string, unknown>, command: string): {
34
+ changed: boolean;
35
+ parsed: Record<string, unknown>;
36
+ };
37
+ export declare function installPermissionHook(client: string, options?: InstallPermissionHookOptions): Promise<InstallPermissionHookResult>;
38
+ //# sourceMappingURL=permissionHooks.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"permissionHooks.d.ts","sourceRoot":"","sources":["../../src/utils/permissionHooks.ts"],"names":[],"mappings":"AAMA,MAAM,WAAW,oBAAoB;IACnC,IAAI,EAAE,MAAM,CAAC;IACb,UAAU,EAAE,MAAM,CAAC;IACnB,YAAY,EAAE,MAAM,CAAC;IACrB,WAAW,EAAE,MAAM,CAAC;CACrB;AAED,MAAM,WAAW,oBAAoB;IACnC,SAAS,EAAE,OAAO,CAAC;IACnB,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,YAAY,CAAC,EAAE,MAAM,CAAC;CACvB;AAED,MAAM,WAAW,2BAA2B;IAC1C,SAAS,EAAE,OAAO,CAAC;IACnB,SAAS,EAAE,OAAO,CAAC;IACnB,UAAU,EAAE,OAAO,CAAC;IACpB,IAAI,EAAE,MAAM,CAAC;IACb,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,OAAO,EAAE,MAAM,CAAC;CACjB;AAED,MAAM,WAAW,4BAA4B;IAC3C,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,gBAAgB,CAAC,EAAE,MAAM,CAAC;IAC1B,GAAG,CAAC,EAAE,IAAI,CAAC;CACZ;AA4BD,wBAAgB,2BAA2B,CAAC,OAAO,SAAY,GAAG,MAAM,CAEvE;AAED,wBAAgB,2BAA2B,CAAC,OAAO,SAAY,GAAG,MAAM,CAEvE;AAED,wBAAgB,yBAAyB,CAAC,OAAO,SAAY,GAAG,MAAM,CAErE;AAED,wBAAgB,uBAAuB,CAAC,OAAO,SAAY,GAAG,oBAAoB,CA0BjF;AAUD,wBAAgB,0BAA0B,CACxC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAC/B,OAAO,EAAE,MAAM,GACd;IAAE,OAAO,EAAE,OAAO,CAAC;IAAC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAA;CAAE,CAsCvD;AAsDD,wBAAsB,qBAAqB,CACzC,MAAM,EAAE,MAAM,EACd,OAAO,GAAE,4BAAiC,GACzC,OAAO,CAAC,2BAA2B,CAAC,CAiDtC"}
@@ -0,0 +1,194 @@
1
+ import { accessSync, constants as fsConstants, copyFileSync, existsSync, readFileSync, statSync, writeFileSync } from 'node:fs';
2
+ import { chmod, mkdir, readFile, writeFile } from 'node:fs/promises';
3
+ import { dirname, join } from 'node:path';
4
+ import { fileURLToPath } from 'node:url';
5
+ import { homedir } from 'node:os';
6
+ function repoRootFromModule() {
7
+ const currentFile = fileURLToPath(import.meta.url);
8
+ return dirname(dirname(dirname(currentFile)));
9
+ }
10
+ function isMissingFileError(error) {
11
+ return Boolean(error
12
+ && typeof error === 'object'
13
+ && 'code' in error
14
+ && error.code === 'ENOENT');
15
+ }
16
+ function detectIndent(raw) {
17
+ for (const line of raw.split('\n')) {
18
+ if (line.length === 0 || line.startsWith('{') || line.startsWith('}'))
19
+ continue;
20
+ if (line.startsWith('\t'))
21
+ return '\t';
22
+ if (line.startsWith(' ')) {
23
+ const spaces = line.length - line.trimStart().length;
24
+ if (spaces >= 2)
25
+ return spaces;
26
+ }
27
+ }
28
+ return 2;
29
+ }
30
+ export function getPermissionHookScriptPath(homeDir = homedir()) {
31
+ return join(homeDir, '.dollhouse', 'hooks', 'pretooluse-dollhouse.sh');
32
+ }
33
+ export function getPermissionHookMarkerPath(homeDir = homedir()) {
34
+ return join(homeDir, '.dollhouse', 'run', 'hook-installed.json');
35
+ }
36
+ export function getClaudeHookSettingsPath(homeDir = homedir()) {
37
+ return join(homeDir, '.claude', 'settings.json');
38
+ }
39
+ export function getPermissionHookStatus(homeDir = homedir()) {
40
+ const markerPath = getPermissionHookMarkerPath(homeDir);
41
+ 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 };
48
+ }
49
+ if (!existsSync(parsed.scriptPath) || !existsSync(parsed.settingsPath)) {
50
+ return { installed: false };
51
+ }
52
+ return {
53
+ installed: true,
54
+ host: parsed.host,
55
+ scriptPath: parsed.scriptPath,
56
+ settingsPath: parsed.settingsPath,
57
+ };
58
+ }
59
+ catch {
60
+ return { installed: false };
61
+ }
62
+ }
63
+ function normalizeHooksRoot(parsed) {
64
+ const hooksValue = parsed.hooks;
65
+ if (!hooksValue || typeof hooksValue !== 'object' || Array.isArray(hooksValue)) {
66
+ parsed.hooks = {};
67
+ }
68
+ return parsed.hooks;
69
+ }
70
+ export function ensureClaudePreToolUseHook(parsed, command) {
71
+ const hooksRoot = normalizeHooksRoot(parsed);
72
+ const existingEntries = Array.isArray(hooksRoot.PreToolUse)
73
+ ? hooksRoot.PreToolUse.filter((entry) => typeof entry === 'object' && entry !== null)
74
+ : [];
75
+ hooksRoot.PreToolUse = existingEntries;
76
+ const commandExists = existingEntries.some((entry) => {
77
+ const hooks = Array.isArray(entry?.hooks) ? entry.hooks : [];
78
+ return hooks.some((hook) => hook?.type === 'command' && hook?.command === command);
79
+ });
80
+ if (commandExists) {
81
+ return { changed: false, parsed };
82
+ }
83
+ const wildcardEntry = existingEntries.find((entry) => (entry?.matcher === '*' || entry?.matcher === undefined) && Array.isArray(entry?.hooks));
84
+ if (wildcardEntry) {
85
+ const hooks = wildcardEntry.hooks;
86
+ hooks.push({
87
+ type: 'command',
88
+ command,
89
+ });
90
+ }
91
+ else {
92
+ existingEntries.push({
93
+ matcher: '*',
94
+ hooks: [
95
+ {
96
+ type: 'command',
97
+ command,
98
+ },
99
+ ],
100
+ });
101
+ }
102
+ return { changed: true, parsed };
103
+ }
104
+ async function copyHookScript(sourcePath, targetPath) {
105
+ await mkdir(dirname(targetPath), { recursive: true });
106
+ const sourceRaw = await readFile(sourcePath, 'utf-8');
107
+ let targetRaw;
108
+ try {
109
+ targetRaw = await readFile(targetPath, 'utf-8');
110
+ }
111
+ catch (error) {
112
+ if (!isMissingFileError(error)) {
113
+ throw error;
114
+ }
115
+ }
116
+ const changed = targetRaw === undefined || sourceRaw !== targetRaw;
117
+ if (changed) {
118
+ copyFileSync(sourcePath, targetPath);
119
+ }
120
+ else {
121
+ accessSync(targetPath, fsConstants.F_OK);
122
+ }
123
+ await chmod(targetPath, 0o755);
124
+ return changed;
125
+ }
126
+ async function mergeClaudeSettings(settingsPath, command) {
127
+ await mkdir(dirname(settingsPath), { recursive: true });
128
+ let raw = '{}\n';
129
+ try {
130
+ raw = await readFile(settingsPath, 'utf-8');
131
+ }
132
+ catch {
133
+ raw = '{}\n';
134
+ }
135
+ const indent = detectIndent(raw);
136
+ const parsed = raw.trim().length === 0 ? {} : JSON.parse(raw);
137
+ const { changed, parsed: updated } = ensureClaudePreToolUseHook(parsed, command);
138
+ if (!changed) {
139
+ return { changed: false };
140
+ }
141
+ let backupPath;
142
+ if (existsSync(settingsPath)) {
143
+ backupPath = `${settingsPath}.dollhouse.bak`;
144
+ writeFileSync(backupPath, raw, 'utf-8');
145
+ }
146
+ await writeFile(settingsPath, JSON.stringify(updated, null, indent) + '\n', 'utf-8');
147
+ return { changed: true, backupPath };
148
+ }
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
+ };
159
+ }
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);
173
+ await mkdir(dirname(markerPath), { recursive: true });
174
+ const installedAt = (options.now ?? new Date()).toISOString();
175
+ const marker = {
176
+ host: normalizedClient,
177
+ scriptPath: targetScriptPath,
178
+ settingsPath,
179
+ installedAt,
180
+ };
181
+ await writeFile(markerPath, JSON.stringify(marker, null, 2) + '\n', 'utf-8');
182
+ return {
183
+ supported: true,
184
+ installed: true,
185
+ configured: true,
186
+ host: normalizedClient,
187
+ scriptPath: targetScriptPath,
188
+ settingsPath,
189
+ markerPath,
190
+ backupPath: settingsResult.backupPath,
191
+ message: 'Installed Claude Code permission hook and updated settings.json.',
192
+ };
193
+ }
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"]}