@dollhousemcp/mcp-server 2.0.15 → 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 (53) hide show
  1. package/CHANGELOG.md +10 -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/ElementCRUDHandler.d.ts.map +1 -1
  10. package/dist/handlers/ElementCRUDHandler.js +7 -3
  11. package/dist/handlers/element-crud/createElement.d.ts.map +1 -1
  12. package/dist/handlers/element-crud/createElement.js +6 -2
  13. package/dist/handlers/element-crud/editElement.d.ts.map +1 -1
  14. package/dist/handlers/element-crud/editElement.js +6 -2
  15. package/dist/handlers/element-crud/helpers.d.ts +2 -0
  16. package/dist/handlers/element-crud/helpers.d.ts.map +1 -1
  17. package/dist/handlers/element-crud/helpers.js +21 -2
  18. package/dist/handlers/mcp-aql/IntrospectionResolver.d.ts.map +1 -1
  19. package/dist/handlers/mcp-aql/IntrospectionResolver.js +34 -7
  20. package/dist/handlers/mcp-aql/MCPAQLHandler.d.ts.map +1 -1
  21. package/dist/handlers/mcp-aql/MCPAQLHandler.js +51 -20
  22. package/dist/handlers/mcp-aql/OperationSchema.d.ts.map +1 -1
  23. package/dist/handlers/mcp-aql/OperationSchema.js +6 -4
  24. package/dist/handlers/mcp-aql/evaluatePermission.d.ts +2 -1
  25. package/dist/handlers/mcp-aql/evaluatePermission.d.ts.map +1 -1
  26. package/dist/handlers/mcp-aql/evaluatePermission.js +23 -11
  27. package/dist/handlers/mcp-aql/policies/ElementPolicies.d.ts +8 -0
  28. package/dist/handlers/mcp-aql/policies/ElementPolicies.d.ts.map +1 -1
  29. package/dist/handlers/mcp-aql/policies/ElementPolicies.js +26 -1
  30. package/dist/handlers/strategies/BaseActivationStrategy.d.ts.map +1 -1
  31. package/dist/handlers/strategies/BaseActivationStrategy.js +12 -3
  32. package/dist/handlers/strategies/PersonaActivationStrategy.js +2 -2
  33. package/dist/utils/permissionHooks.d.ts +74 -0
  34. package/dist/utils/permissionHooks.d.ts.map +1 -0
  35. package/dist/utils/permissionHooks.js +771 -0
  36. package/dist/web/public/index.html +12 -6
  37. package/dist/web/public/permissions.css +11 -0
  38. package/dist/web/public/permissions.js +78 -35
  39. package/dist/web/public/setup.css +172 -1
  40. package/dist/web/public/setup.js +644 -38
  41. package/dist/web/routes/permissionRoutes.d.ts.map +1 -1
  42. package/dist/web/routes/permissionRoutes.js +95 -27
  43. package/dist/web/routes/setupRoutes.d.ts +4 -0
  44. package/dist/web/routes/setupRoutes.d.ts.map +1 -1
  45. package/dist/web/routes/setupRoutes.js +122 -39
  46. package/package.json +8 -1
  47. package/scripts/pretooluse-codex.sh +6 -0
  48. package/scripts/pretooluse-cursor.sh +6 -0
  49. package/scripts/pretooluse-dollhouse.sh +110 -0
  50. package/scripts/pretooluse-gemini.sh +6 -0
  51. package/scripts/pretooluse-vscode.sh +163 -0
  52. package/scripts/pretooluse-windsurf.sh +168 -0
  53. 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;AA+CD,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 = {
@@ -58,10 +59,20 @@ const platformFormatters = {
58
59
  cursor: formatCursor,
59
60
  windsurf: formatWindsurf,
60
61
  codex: formatCodex,
62
+ vscode: formatClaudeCode,
61
63
  claude_code: formatClaudeCode,
62
64
  };
63
65
  /** Known platform identifiers */
64
66
  export const SUPPORTED_PLATFORMS = Object.keys(platformFormatters);
67
+ function warnOnUnknownPlatform(platform) {
68
+ void import('../../utils/logger.js')
69
+ .then(({ logger }) => {
70
+ logger.warn(`[evaluatePermission] Unknown platform "${platform}", defaulting to claude_code format. Supported: ${SUPPORTED_PLATFORMS.join(', ')}`);
71
+ })
72
+ .catch((error) => {
73
+ console.warn(`[evaluatePermission] Failed to load logger while handling unknown platform "${platform}".`, error);
74
+ });
75
+ }
65
76
  /**
66
77
  * Format permission evaluation response for platform-specific hook scripts.
67
78
  * Each platform expects a different JSON shape from its hook response.
@@ -77,9 +88,7 @@ export function formatPermissionResponse(decision, platform, _input, reason) {
77
88
  case 'claude_code': return formatClaudeCode(decision, reason);
78
89
  default:
79
90
  // 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(() => { });
91
+ warnOnUnknownPlatform(platform);
83
92
  return formatClaudeCode(decision, reason);
84
93
  }
85
94
  }
@@ -97,6 +106,9 @@ export async function evaluatePermission(params, deps) {
97
106
  ? inputRaw
98
107
  : {};
99
108
  const platform = typeof params.platform === 'string' ? params.platform : 'claude_code';
109
+ const sessionId = typeof params.session_id === 'string' && params.session_id.trim() !== ''
110
+ ? params.session_id
111
+ : undefined;
100
112
  // Rate limit
101
113
  const rateStatus = deps.permissionPromptLimiter.checkLimit();
102
114
  if (!rateStatus.allowed) {
@@ -114,7 +126,7 @@ export async function evaluatePermission(params, deps) {
114
126
  // Stage 2: Element policy evaluation
115
127
  let elements;
116
128
  try {
117
- elements = await deps.getActiveElements();
129
+ elements = await deps.getActiveElements(sessionId);
118
130
  }
119
131
  catch (err) {
120
132
  throw new PermissionEvaluationError(`Failed to fetch active elements for policy evaluation: ${err instanceof Error ? err.message : String(err)}`, 'element_fetch', toolName, err);
@@ -129,4 +141,4 @@ export async function evaluatePermission(params, deps) {
129
141
  // Default: allow
130
142
  return formatPermissionResponse('allow', platform, input);
131
143
  }
132
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"evaluatePermission.js","sourceRoot":"","sources":["../../../src/handlers/mcp-aql/evaluatePermission.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;GAcG;AAMH,wEAAwE;AACxE,MAAM,OAAO,yBAA0B,SAAQ,KAAK;IAClC,KAAK,CAA+D;IACpE,QAAQ,CAAS;IAEjC,YACE,OAAe,EACf,KAAmE,EACnE,QAAgB,EAChB,KAAe;QAEf,KAAK,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC;QAC9C,IAAI,CAAC,IAAI,GAAG,2BAA2B,CAAC;QACxC,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;QACnB,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC;IAC3B,CAAC;CACF;AAUD,mEAAmE;AACnE,SAAS,UAAU,CAAC,GAA4B,EAAE,MAAe,EAAE,GAAG,GAAG,QAAQ;IAC/E,OAAO,MAAM,CAAC,CAAC,CAAC,EAAE,GAAG,GAAG,EAAE,CAAC,GAAG,CAAC,EAAE,MAAM,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC;AAClD,CAAC;AAED,4DAA4D;AAC5D,SAAS,YAAY,CAAC,QAAgB,EAAE,MAAe;IACrD,OAAO,UAAU,CAAC,EAAE,QAAQ,EAAE,QAAQ,KAAK,KAAK,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,QAAQ,EAAE,EAAE,MAAM,CAAC,CAAC;AAClF,CAAC;AAED,4DAA4D;AAC5D,SAAS,YAAY,CAAC,QAAgB,EAAE,MAAe;IACrD,OAAO,UAAU,CAAC,EAAE,UAAU,EAAE,QAAQ,EAAE,EAAE,MAAM,CAAC,CAAC;AACtD,CAAC;AAED,6CAA6C;AAC7C,SAAS,cAAc,CAAC,QAAgB,EAAE,MAAe;IACvD,OAAO,UAAU,CAAC,EAAE,OAAO,EAAE,QAAQ,KAAK,OAAO,EAAE,EAAE,MAAM,CAAC,CAAC;AAC/D,CAAC;AAED,+DAA+D;AAC/D,SAAS,WAAW,CAAC,QAAgB,EAAE,MAAe;IACpD,OAAO,EAAE,kBAAkB,EAAE,UAAU,CAAC,EAAE,kBAAkB,EAAE,QAAQ,KAAK,KAAK,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,QAAQ,EAAE,EAAE,MAAM,CAAC,EAAE,CAAC;AACpH,CAAC;AAED,uFAAuF;AACvF,SAAS,gBAAgB,CAAC,QAAgB,EAAE,MAAe;IACzD,IAAI,QAAQ,KAAK,OAAO;QAAE,OAAO,EAAE,QAAQ,EAAE,OAAO,EAAE,CAAC;IACvD,IAAI,QAAQ,KAAK,KAAK;QAAE,OAAO,UAAU,CAAC,EAAE,QAAQ,EAAE,KAAK,EAAE,EAAE,MAAM,EAAE,SAAS,CAAC,CAAC;IAClF,OAAO,UAAU,CAAC,EAAE,QAAQ,EAAE,MAAM,EAAE,EAAE,MAAM,CAAC,CAAC;AAClD,CAAC;AAED,gCAAgC;AAChC,MAAM,kBAAkB,GAAmF;IACzG,MAAM,EAAE,YAAY;IACpB,MAAM,EAAE,YAAY;IACpB,QAAQ,EAAE,cAAc;IACxB,KAAK,EAAE,WAAW;IAClB,WAAW,EAAE,gBAAgB;CAC9B,CAAC;AAEF,iCAAiC;AACjC,MAAM,CAAC,MAAM,mBAAmB,GAAG,MAAM,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC;AAEnE;;;;;GAKG;AACH,MAAM,UAAU,wBAAwB,CACtC,QAAkC,EAClC,QAAgB,EAChB,MAA+B,EAC/B,MAAe;IAEf,QAAQ,QAAQ,EAAE,CAAC;QACjB,KAAK,QAAQ,CAAC,CAAC,OAAO,YAAY,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;QACrD,KAAK,QAAQ,CAAC,CAAC,OAAO,YAAY,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;QACrD,KAAK,UAAU,CAAC,CAAC,OAAO,cAAc,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;QACzD,KAAK,OAAO,CAAC,CAAC,OAAO,WAAW,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;QACnD,KAAK,aAAa,CAAC,CAAC,OAAO,gBAAgB,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;QAC9D;YACE,iEAAiE;YACjE,MAAM,CAAC,uBAAuB,CAAC,CAAC,IAAI,CAAC,CAAC,EAAE,MAAM,EAAE,EAAE,EAAE;gBAClD,MAAM,CAAC,IAAI,CAAC,0CAA0C,QAAQ,mDAAmD,mBAAmB,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YACrJ,CAAC,CAAC,CAAC,KAAK,CAAC,GAAG,EAAE,GAA8B,CAAC,CAAC,CAAC;YAC/C,OAAO,gBAAgB,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;IAC9C,CAAC;AACH,CAAC;AAED;;;;;;GAMG;AACH,MAAM,CAAC,KAAK,UAAU,kBAAkB,CACtC,MAAoE,EACpE,IAA4B;IAE5B,MAAM,QAAQ,GAAG,OAAO,MAAM,CAAC,SAAS,KAAK,QAAQ,CAAC,CAAC,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,CAAC;IAC9E,MAAM,QAAQ,GAAG,MAAM,CAAC,KAAK,CAAC;IAC9B,MAAM,KAAK,GAAG,CAAC,QAAQ,IAAI,OAAO,QAAQ,KAAK,QAAQ,CAAC;QACtD,CAAC,CAAC,QAAmC;QACrC,CAAC,CAAC,EAAE,CAAC;IACP,MAAM,QAAQ,GAAG,OAAO,MAAM,CAAC,QAAQ,KAAK,QAAQ,CAAC,CAAC,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,CAAC,aAAa,CAAC;IAEvF,aAAa;IACb,MAAM,UAAU,GAAG,IAAI,CAAC,uBAAuB,CAAC,UAAU,EAAE,CAAC;IAC7D,IAAI,CAAC,UAAU,CAAC,OAAO,EAAE,CAAC;QACxB,OAAO,wBAAwB,CAAC,MAAM,EAAE,QAAQ,EAAE,KAAK,EAAE,qBAAqB,CAAC,CAAC;IAClF,CAAC;IACD,IAAI,CAAC,uBAAuB,CAAC,YAAY,EAAE,CAAC;IAE5C,iCAAiC;IACjC,MAAM,cAAc,GAAG,IAAI,CAAC,YAAY,CAAC,QAAQ,EAAE,KAAK,CAAC,CAAC;IAC1D,IAAI,cAAc,CAAC,QAAQ,KAAK,OAAO,EAAE,CAAC;QACxC,OAAO,wBAAwB,CAAC,OAAO,EAAE,QAAQ,EAAE,KAAK,CAAC,CAAC;IAC5D,CAAC;IACD,IAAI,cAAc,CAAC,QAAQ,KAAK,MAAM,EAAE,CAAC;QACvC,OAAO,wBAAwB,CAAC,MAAM,EAAE,QAAQ,EAAE,KAAK,EAAE,cAAc,CAAC,MAAM,CAAC,CAAC;IAClF,CAAC;IAED,qCAAqC;IACrC,IAAI,QAAyB,CAAC;IAC9B,IAAI,CAAC;QACH,QAAQ,GAAG,MAAM,IAAI,CAAC,iBAAiB,EAAE,CAAC;IAC5C,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,MAAM,IAAI,yBAAyB,CACjC,0DAA0D,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,EAC5G,eAAe,EAAE,QAAQ,EAAE,GAAG,CAC/B,CAAC;IACJ,CAAC;IACD,MAAM,QAAQ,GAAG,IAAI,CAAC,qBAAqB,CAAC,QAAQ,EAAE,KAAK,EAAE,QAAQ,CAAC,CAAC;IAEvE,IAAI,QAAQ,CAAC,QAAQ,KAAK,MAAM,EAAE,CAAC;QACjC,OAAO,wBAAwB,CAAC,MAAM,EAAE,QAAQ,EAAE,KAAK,EAAE,QAAQ,CAAC,OAAO,CAAC,CAAC;IAC7E,CAAC;IACD,IAAI,QAAQ,CAAC,QAAQ,KAAK,SAAS,EAAE,CAAC;QACpC,OAAO,wBAAwB,CAAC,KAAK,EAAE,QAAQ,EAAE,KAAK,EACpD,QAAQ,CAAC,OAAO,IAAI,0CAA0C,CAAC,CAAC;IACpE,CAAC;IAED,iBAAiB;IACjB,OAAO,wBAAwB,CAAC,OAAO,EAAE,QAAQ,EAAE,KAAK,CAAC,CAAC;AAC5D,CAAC","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"]}
144
+ //# 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  vscode: formatClaudeCode,\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"]}
@@ -81,6 +81,14 @@ export declare function createDecisionFromPolicy(operation: string, result: Elem
81
81
  * @throws Error if the policy is malformed
82
82
  */
83
83
  export declare function parseElementPolicy(metadata: unknown): ElementGatekeeperPolicy | undefined;
84
+ /**
85
+ * Validate authored gatekeeper input before save.
86
+ *
87
+ * Authoring-time validation is stricter than load-time sanitization: it should
88
+ * reject misplaced policy blocks instead of silently saving an element that
89
+ * later appears active but has non-enforceable external restrictions.
90
+ */
91
+ export declare function getGatekeeperAuthoringErrors(record: Record<string, unknown> | undefined): string[];
84
92
  /**
85
93
  * Analyze externalRestrictions patterns for common mistakes and suspicious syntax.
86
94
  *
@@ -1 +1 @@
1
- {"version":3,"file":"ElementPolicies.d.ts","sourceRoot":"","sources":["../../../../src/handlers/mcp-aql/policies/ElementPolicies.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;GAYG;AAEH,OAAO,EACL,eAAe,EACf,KAAK,uBAAuB,EAE5B,KAAK,kBAAkB,EAExB,MAAM,uBAAuB,CAAC;AAM/B;;;GAGG;AACH,MAAM,WAAW,yBAAyB;IACxC,IAAI,EAAE,MAAM,CAAC;IACb,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,UAAU,CAAC,EAAE,uBAAuB,CAAC;IACrC,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC;CACxB;AAED;;;GAGG;AACH,MAAM,WAAW,aAAa;IAC5B,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE,MAAM,CAAC;IACb,QAAQ,EAAE,yBAAyB,CAAC;CACrC;AAED;;;GAGG;AACH,MAAM,WAAW,mBAAmB;IAClC,kEAAkE;IAClE,eAAe,EAAE,eAAe,CAAC;IACjC,oDAAoD;IACpD,aAAa,CAAC,EAAE,MAAM,CAAC;IACvB,kEAAkE;IAClE,aAAa,CAAC,EAAE,OAAO,GAAG,SAAS,GAAG,MAAM,GAAG,mBAAmB,CAAC;IACnE,8DAA8D;IAC9D,YAAY,CAAC,EAAE,OAAO,CAAC;IACvB;;;;OAIG;IACH,mBAAmB,CAAC,EAAE,KAAK,CAAC;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,WAAW,EAAE,eAAe,CAAA;KAAE,CAAC,CAAC;CAC7E;AAED;;;;;;;;GAQG;AACH,wBAAgB,oBAAoB,CAClC,SAAS,EAAE,MAAM,EACjB,cAAc,EAAE,aAAa,EAAE,EAC/B,iBAAiB,CAAC,EAAE,MAAM,GACzB,mBAAmB,CAgGrB;AAED;;;;;;;GAOG;AACH,wBAAgB,wBAAwB,CACtC,SAAS,EAAE,MAAM,EACjB,MAAM,EAAE,mBAAmB,EAC3B,iBAAiB,CAAC,EAAE,MAAM,GACzB,kBAAkB,CA2EpB;AAED;;;;;;GAMG;AACH,wBAAgB,kBAAkB,CAChC,QAAQ,EAAE,OAAO,GAChB,uBAAuB,GAAG,SAAS,CAmHrC;AAgED;;;;;;;;;;;;;;;GAeG;AACH,wBAAgB,oBAAoB,CAClC,QAAQ,EAAE,MAAM,EAAE,EAClB,SAAS,EAAE,MAAM,GAChB,MAAM,EAAE,CAMV;AA+HD;;;;GAIG;AACH,wBAAgB,0BAA0B,CAAC,SAAS,EAAE,MAAM,GAAG,OAAO,CAErE;AAED;;;GAGG;AACH,wBAAgB,yBAAyB,CACvC,cAAc,EAAE,KAAK,CAAC;IAAE,IAAI,EAAE,MAAM,CAAC;IAAC,IAAI,EAAE,MAAM,CAAC;IAAC,QAAQ,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAA;CAAE,CAAC,GACvF;IAAE,IAAI,EAAE,MAAM,CAAC;IAAC,IAAI,EAAE,MAAM,CAAA;CAAE,GAAG,SAAS,CAS5C;AAED;;;GAGG;AACH,wBAAgB,2BAA2B,CACzC,cAAc,EAAE,KAAK,CAAC;IAAE,IAAI,EAAE,MAAM,CAAC;IAAC,IAAI,EAAE,MAAM,CAAC;IAAC,QAAQ,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAA;CAAE,CAAC,GACvF,KAAK,CAAC;IAAE,IAAI,EAAE,MAAM,CAAC;IAAC,IAAI,EAAE,MAAM,CAAA;CAAE,CAAC,CAUvC;AAwCD,wBAAgB,wBAAwB,CACtC,SAAS,EAAE,OAAO,EAClB,WAAW,EAAE,MAAM,EACnB,WAAW,EAAE,MAAM,GAClB,uBAAuB,GAAG,SAAS,CAkCrC"}
1
+ {"version":3,"file":"ElementPolicies.d.ts","sourceRoot":"","sources":["../../../../src/handlers/mcp-aql/policies/ElementPolicies.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;GAYG;AAEH,OAAO,EACL,eAAe,EACf,KAAK,uBAAuB,EAE5B,KAAK,kBAAkB,EAExB,MAAM,uBAAuB,CAAC;AAM/B;;;GAGG;AACH,MAAM,WAAW,yBAAyB;IACxC,IAAI,EAAE,MAAM,CAAC;IACb,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,UAAU,CAAC,EAAE,uBAAuB,CAAC;IACrC,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC;CACxB;AAED;;;GAGG;AACH,MAAM,WAAW,aAAa;IAC5B,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE,MAAM,CAAC;IACb,QAAQ,EAAE,yBAAyB,CAAC;CACrC;AAED;;;GAGG;AACH,MAAM,WAAW,mBAAmB;IAClC,kEAAkE;IAClE,eAAe,EAAE,eAAe,CAAC;IACjC,oDAAoD;IACpD,aAAa,CAAC,EAAE,MAAM,CAAC;IACvB,kEAAkE;IAClE,aAAa,CAAC,EAAE,OAAO,GAAG,SAAS,GAAG,MAAM,GAAG,mBAAmB,CAAC;IACnE,8DAA8D;IAC9D,YAAY,CAAC,EAAE,OAAO,CAAC;IACvB;;;;OAIG;IACH,mBAAmB,CAAC,EAAE,KAAK,CAAC;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,WAAW,EAAE,eAAe,CAAA;KAAE,CAAC,CAAC;CAC7E;AAED;;;;;;;;GAQG;AACH,wBAAgB,oBAAoB,CAClC,SAAS,EAAE,MAAM,EACjB,cAAc,EAAE,aAAa,EAAE,EAC/B,iBAAiB,CAAC,EAAE,MAAM,GACzB,mBAAmB,CAgGrB;AAED;;;;;;;GAOG;AACH,wBAAgB,wBAAwB,CACtC,SAAS,EAAE,MAAM,EACjB,MAAM,EAAE,mBAAmB,EAC3B,iBAAiB,CAAC,EAAE,MAAM,GACzB,kBAAkB,CA2EpB;AAED;;;;;;GAMG;AACH,wBAAgB,kBAAkB,CAChC,QAAQ,EAAE,OAAO,GAChB,uBAAuB,GAAG,SAAS,CAmHrC;AAED;;;;;;GAMG;AACH,wBAAgB,4BAA4B,CAC1C,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,GAAG,SAAS,GAC1C,MAAM,EAAE,CAsBV;AAgED;;;;;;;;;;;;;;;GAeG;AACH,wBAAgB,oBAAoB,CAClC,QAAQ,EAAE,MAAM,EAAE,EAClB,SAAS,EAAE,MAAM,GAChB,MAAM,EAAE,CAMV;AA+HD;;;;GAIG;AACH,wBAAgB,0BAA0B,CAAC,SAAS,EAAE,MAAM,GAAG,OAAO,CAErE;AAED;;;GAGG;AACH,wBAAgB,yBAAyB,CACvC,cAAc,EAAE,KAAK,CAAC;IAAE,IAAI,EAAE,MAAM,CAAC;IAAC,IAAI,EAAE,MAAM,CAAC;IAAC,QAAQ,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAA;CAAE,CAAC,GACvF;IAAE,IAAI,EAAE,MAAM,CAAC;IAAC,IAAI,EAAE,MAAM,CAAA;CAAE,GAAG,SAAS,CAS5C;AAED;;;GAGG;AACH,wBAAgB,2BAA2B,CACzC,cAAc,EAAE,KAAK,CAAC;IAAE,IAAI,EAAE,MAAM,CAAC;IAAC,IAAI,EAAE,MAAM,CAAC;IAAC,QAAQ,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAA;CAAE,CAAC,GACvF,KAAK,CAAC;IAAE,IAAI,EAAE,MAAM,CAAC;IAAC,IAAI,EAAE,MAAM,CAAA;CAAE,CAAC,CAUvC;AAwCD,wBAAgB,wBAAwB,CACtC,SAAS,EAAE,OAAO,EAClB,WAAW,EAAE,MAAM,EACnB,WAAW,EAAE,MAAM,GAClB,uBAAuB,GAAG,SAAS,CAkCrC"}
@@ -304,6 +304,31 @@ export function parseElementPolicy(metadata) {
304
304
  externalRestrictions,
305
305
  };
306
306
  }
307
+ /**
308
+ * Validate authored gatekeeper input before save.
309
+ *
310
+ * Authoring-time validation is stricter than load-time sanitization: it should
311
+ * reject misplaced policy blocks instead of silently saving an element that
312
+ * later appears active but has non-enforceable external restrictions.
313
+ */
314
+ export function getGatekeeperAuthoringErrors(record) {
315
+ if (!record || typeof record !== 'object') {
316
+ return [];
317
+ }
318
+ const errors = [];
319
+ if (Object.hasOwn(record, 'externalRestrictions')) {
320
+ errors.push('Invalid gatekeeper policy: externalRestrictions must be nested under gatekeeper.externalRestrictions');
321
+ }
322
+ if (record.gatekeeper !== undefined) {
323
+ try {
324
+ parseElementPolicy({ gatekeeper: record.gatekeeper });
325
+ }
326
+ catch (error) {
327
+ errors.push(error instanceof Error ? error.message : String(error));
328
+ }
329
+ }
330
+ return errors;
331
+ }
307
332
  /**
308
333
  * Validate that a value is an array of strings.
309
334
  *
@@ -573,4 +598,4 @@ export function sanitizeGatekeeperPolicy(rawPolicy, elementName, elementType) {
573
598
  return undefined;
574
599
  }
575
600
  }
576
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ElementPolicies.js","sourceRoot":"","sources":["../../../../src/handlers/mcp-aql/policies/ElementPolicies.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Element Policies\n *\n * Handles element-based access control for the Gatekeeper.\n * Allows ANY DollhouseMCP element to define policies in its metadata\n * that override or restrict the default operation policies.\n *\n * Policy Resolution Order:\n * 1. Active element deny list (highest priority - blocks operation)\n * 2. Active element confirm list (requires confirmation)\n * 3. Active element allow list (auto-approves)\n * 4. Operation default permission (fallback)\n */\n\nimport {\n  PermissionLevel,\n  type ElementGatekeeperPolicy,\n  type CliApprovalPolicy,\n  type GatekeeperDecision,\n  GatekeeperErrorCode,\n} from '../GatekeeperTypes.js';\nimport { getDefaultPermissionLevel, canOperationBeElevated, getOperationPolicy } from './OperationPolicies.js';\nimport { SecurityMonitor } from '../../../security/securityMonitor.js';\nimport { logger } from '../../../utils/logger.js';\nimport { MAX_GLOB_PATTERN_LENGTH } from '../../../utils/patternMatcher.js';\n\n/**\n * Metadata structure for elements with Gatekeeper policies.\n * The 'gatekeeper' field is optional and contains policy definitions.\n */\nexport interface ElementMetadataWithPolicy {\n  name: string;\n  description?: string;\n  gatekeeper?: ElementGatekeeperPolicy;\n  [key: string]: unknown;\n}\n\n/**\n * Active element for policy evaluation.\n * Represents an element currently active in the session.\n */\nexport interface ActiveElement {\n  type: string;\n  name: string;\n  metadata: ElementMetadataWithPolicy;\n}\n\n/**\n * Result of element policy resolution.\n * Contains the effective permission level and policy source.\n */\nexport interface ElementPolicyResult {\n  /** Effective permission level after element policy application */\n  permissionLevel: PermissionLevel;\n  /** Which element's policy determined this result */\n  sourceElement?: string;\n  /** The specific policy field that matched (allow/confirm/deny) */\n  matchedPolicy?: 'allow' | 'confirm' | 'deny' | 'scope_restriction';\n  /** Whether the operation was blocked by scope restrictions */\n  scopeBlocked?: boolean;\n  /**\n   * Elements that wanted to auto-approve this operation but were overridden\n   * by a higher-priority confirm or deny policy from another element.\n   * Issue #674: allow cannot override confirm.\n   */\n  conflictingElements?: Array<{ name: string; wantedLevel: PermissionLevel }>;\n}\n\n/**\n * Resolve the effective permission level for an operation\n * considering all active elements and their policies.\n *\n * @param operation - The operation to check\n * @param activeElements - Currently active elements with their metadata\n * @param targetElementType - Optional element type being operated on\n * @returns The resolved policy result\n */\nexport function resolveElementPolicy(\n  operation: string,\n  activeElements: ActiveElement[],\n  targetElementType?: string\n): ElementPolicyResult {\n  // Start with the default operation permission level\n  let effectiveLevel = getDefaultPermissionLevel(operation);\n  let sourceElement: string | undefined;\n  let matchedPolicy: ElementPolicyResult['matchedPolicy'];\n  let scopeBlocked = false;\n  // Issue #674: Track whether CONFIRM was set by an element policy (not just the route default)\n  // allow can override the route default but NOT another element's confirm policy\n  let confirmedByElement = false;\n  // Track elements that wanted allow but were overridden by an element-set confirm/deny\n  const conflictingElements: Array<{ name: string; wantedLevel: PermissionLevel }> = [];\n\n  // Check each active element's policy in order\n  for (const element of activeElements) {\n    const policy = element.metadata.gatekeeper;\n    if (!policy) {\n      continue;\n    }\n\n    // 1. Check deny list first (highest priority)\n    if (policy.deny?.includes(operation)) {\n      return {\n        permissionLevel: PermissionLevel.DENY,\n        sourceElement: element.name,\n        matchedPolicy: 'deny',\n        conflictingElements: conflictingElements.length > 0 ? conflictingElements : undefined,\n      };\n    }\n\n    // 2. Check scope restrictions\n    if (targetElementType && policy.scopeRestrictions) {\n      const { allowedTypes, blockedTypes } = policy.scopeRestrictions;\n\n      // If allowedTypes is specified, target must be in the list\n      if (allowedTypes && !allowedTypes.includes(targetElementType)) {\n        scopeBlocked = true;\n        sourceElement = element.name;\n        matchedPolicy = 'scope_restriction';\n        continue; // Check other elements, might have different policy\n      }\n\n      // If blockedTypes is specified, target must NOT be in the list\n      if (blockedTypes?.includes(targetElementType)) {\n        scopeBlocked = true;\n        sourceElement = element.name;\n        matchedPolicy = 'scope_restriction';\n        continue;\n      }\n    }\n\n    // 3. Check confirm list (requires confirmation)\n    if (policy.confirm?.includes(operation)) {\n      // Don't downgrade from DENY or CONFIRM_SINGLE_USE\n      if (effectiveLevel !== PermissionLevel.DENY) {\n        effectiveLevel = PermissionLevel.CONFIRM_SESSION;\n        confirmedByElement = true;\n        sourceElement = element.name;\n        matchedPolicy = 'confirm';\n      }\n    }\n\n    // 4. Check allow list (auto-approves)\n    // Issue #674: allow CAN override the route default, but NOT another element's confirm policy.\n    // Priority hierarchy: element deny > element confirm > element allow > route default\n    if (policy.allow?.includes(operation)) {\n      // Only elevate if the operation allows elevation\n      if (canOperationBeElevated(operation)) {\n        if (!confirmedByElement) {\n          // No element has confirmed this — safe to elevate (overrides route default)\n          effectiveLevel = PermissionLevel.AUTO_APPROVE;\n          sourceElement = element.name;\n          matchedPolicy = 'allow';\n        } else {\n          // Another element's confirm policy takes priority over this allow\n          conflictingElements.push({ name: element.name, wantedLevel: PermissionLevel.AUTO_APPROVE });\n        }\n      }\n    }\n  }\n\n  // If scope was blocked by all elements with restrictions\n  if (scopeBlocked) {\n    return {\n      permissionLevel: PermissionLevel.DENY,\n      sourceElement,\n      matchedPolicy: 'scope_restriction',\n      scopeBlocked: true,\n    };\n  }\n\n  return {\n    permissionLevel: effectiveLevel,\n    sourceElement,\n    matchedPolicy,\n    conflictingElements: conflictingElements.length > 0 ? conflictingElements : undefined,\n  };\n}\n\n/**\n * Create a Gatekeeper decision from element policy resolution.\n *\n * @param operation - The operation that was checked\n * @param result - The element policy resolution result\n * @param targetElementType - Optional element type being operated on\n * @returns A GatekeeperDecision object\n */\nexport function createDecisionFromPolicy(\n  operation: string,\n  result: ElementPolicyResult,\n  targetElementType?: string\n): GatekeeperDecision {\n  const { permissionLevel, sourceElement, scopeBlocked, conflictingElements } = result;\n\n  // Handle DENY\n  if (permissionLevel === PermissionLevel.DENY) {\n    if (scopeBlocked) {\n      return {\n        allowed: false,\n        permissionLevel,\n        errorCode: GatekeeperErrorCode.SCOPE_RESTRICTION,\n        reason: `Operation \"${operation}\" is not allowed on element type \"${targetElementType}\" due to scope restrictions in active element \"${sourceElement}\"`,\n        suggestion: `Deactivate the element \"${sourceElement}\" or use a different element type`,\n        policySource: 'element_policy',\n      };\n    }\n\n    return {\n      allowed: false,\n      permissionLevel,\n      errorCode: GatekeeperErrorCode.ELEMENT_POLICY_VIOLATION,\n      reason: `Operation \"${operation}\" is blocked by active element \"${sourceElement}\"'s deny policy`,\n      suggestion: `Deactivate the element \"${sourceElement}\" to proceed with this operation`,\n      policySource: 'element_policy',\n    };\n  }\n\n  // Handle confirmation required\n  if (\n    permissionLevel === PermissionLevel.CONFIRM_SESSION ||\n    permissionLevel === PermissionLevel.CONFIRM_SINGLE_USE\n  ) {\n    // Build human-readable rationale explaining WHY confirmation is needed\n    const policy = getOperationPolicy(operation);\n    const levelLabel = permissionLevel === PermissionLevel.CONFIRM_SINGLE_USE\n      ? 'requires confirmation each time'\n      : 'requires confirmation once per session';\n\n    let rationale: string;\n    if (sourceElement && result.matchedPolicy === 'confirm') {\n      // Element policy elevated this operation's confirmation requirement\n      const policyDetail = policy?.rationale ? ` ${policy.rationale}` : '';\n      rationale = `Active element \"${sourceElement}\" policy requires confirmation for this operation.${policyDetail}`;\n    } else if (policy?.rationale) {\n      // Route-level default with documented rationale\n      rationale = policy.rationale;\n    } else {\n      // Endpoint-level default, no specific rationale\n      rationale = 'This operation modifies data and requires human approval before proceeding';\n    }\n\n    // Issue #674: Surface elements that wanted to auto-approve but were overridden\n    const conflictNote = conflictingElements && conflictingElements.length > 0\n      ? ` (Note: ${conflictingElements.map(e => `\"${e.name}\"`).join(', ')} would auto-approve this but ${conflictingElements.length === 1 ? 'is' : 'are'} overridden by the confirm policy.)`\n      : '';\n\n    return {\n      allowed: false,\n      permissionLevel,\n      errorCode: GatekeeperErrorCode.CONFIRMATION_REQUIRED,\n      reason: `Operation \"${operation}\" ${levelLabel}. ${rationale}${conflictNote}`,\n      suggestion: 'Use the confirmation dialog to approve this operation',\n      confirmationPending: true,\n      policySource: sourceElement ? 'element_policy' : 'operation_default',\n    };\n  }\n\n  // Handle AUTO_APPROVE\n  return {\n    allowed: true,\n    permissionLevel,\n    reason: sourceElement\n      ? `Operation \"${operation}\" auto-approved by element \"${sourceElement}\"`\n      : `Operation \"${operation}\" auto-approved by default policy`,\n    policySource: sourceElement ? 'element_policy' : 'operation_default',\n  };\n}\n\n/**\n * Parse and validate a Gatekeeper policy from element metadata.\n *\n * @param metadata - The element metadata to parse\n * @returns The parsed policy, or undefined if no policy is defined\n * @throws Error if the policy is malformed\n */\nexport function parseElementPolicy(\n  metadata: unknown\n): ElementGatekeeperPolicy | undefined {\n  if (!metadata || typeof metadata !== 'object') {\n    return undefined;\n  }\n\n  const meta = metadata as Record<string, unknown>;\n  const gatekeeper = meta.gatekeeper;\n\n  if (!gatekeeper) {\n    return undefined;\n  }\n\n  if (typeof gatekeeper !== 'object' || gatekeeper === null) {\n    throw new Error('Invalid gatekeeper policy: must be an object');\n  }\n\n  const policy = gatekeeper as Record<string, unknown>;\n\n  // Validate and extract allow list\n  const allow = validateStringArray(policy.allow, 'allow');\n  const confirm = validateStringArray(policy.confirm, 'confirm');\n  const deny = validateStringArray(policy.deny, 'deny');\n\n  // Validate scope restrictions\n  let scopeRestrictions: ElementGatekeeperPolicy['scopeRestrictions'];\n  if (policy.scopeRestrictions) {\n    if (typeof policy.scopeRestrictions !== 'object' || policy.scopeRestrictions === null) {\n      throw new Error('Invalid gatekeeper policy: scopeRestrictions must be an object');\n    }\n    const sr = policy.scopeRestrictions as Record<string, unknown>;\n    scopeRestrictions = {\n      allowedTypes: validateStringArray(sr.allowedTypes, 'scopeRestrictions.allowedTypes'),\n      blockedTypes: validateStringArray(sr.blockedTypes, 'scopeRestrictions.blockedTypes'),\n    };\n  }\n\n  // Validate external restrictions (Issue #625 Phase 2)\n  let externalRestrictions: ElementGatekeeperPolicy['externalRestrictions'];\n  if (policy.externalRestrictions) {\n    if (typeof policy.externalRestrictions !== 'object' || policy.externalRestrictions === null) {\n      throw new Error('Invalid gatekeeper policy: externalRestrictions must be an object');\n    }\n    const er = policy.externalRestrictions as Record<string, unknown>;\n    if (!er.description || typeof er.description !== 'string') {\n      throw new Error('Invalid gatekeeper policy: externalRestrictions.description is required and must be a non-empty string');\n    }\n    const denyPatterns = validateStringArray(er.denyPatterns, 'externalRestrictions.denyPatterns');\n    const confirmPatterns = validateStringArray(er.confirmPatterns, 'externalRestrictions.confirmPatterns');\n    const allowPatterns = validateStringArray(er.allowPatterns, 'externalRestrictions.allowPatterns');\n    if (denyPatterns) validatePatternStrings(denyPatterns, 'externalRestrictions.denyPatterns');\n    if (confirmPatterns) validatePatternStrings(confirmPatterns, 'externalRestrictions.confirmPatterns');\n    if (allowPatterns) validatePatternStrings(allowPatterns, 'externalRestrictions.allowPatterns');\n    // Validate approvalPolicy (Issue #625 Phase 3)\n    let approvalPolicy: CliApprovalPolicy | undefined = undefined;\n    if (er.approvalPolicy !== undefined) {\n      if (typeof er.approvalPolicy !== 'object' || er.approvalPolicy === null) {\n        throw new Error('Invalid gatekeeper policy: externalRestrictions.approvalPolicy must be an object');\n      }\n      const ap = er.approvalPolicy as Record<string, unknown>;\n\n      // Validate requireApproval\n      let requireApproval: ('moderate' | 'dangerous')[] | undefined;\n      if (ap.requireApproval !== undefined) {\n        if (!Array.isArray(ap.requireApproval)) {\n          throw new Error('Invalid gatekeeper policy: externalRestrictions.approvalPolicy.requireApproval must be an array');\n        }\n        const validLevels = new Set(['moderate', 'dangerous']);\n        for (const level of ap.requireApproval) {\n          if (typeof level !== 'string' || !validLevels.has(level)) {\n            throw new Error(`Invalid gatekeeper policy: externalRestrictions.approvalPolicy.requireApproval contains invalid value \"${level}\". Must be \"moderate\" or \"dangerous\".`);\n          }\n        }\n        requireApproval = ap.requireApproval as ('moderate' | 'dangerous')[];\n      }\n\n      // Validate defaultScope\n      let defaultScope: 'single' | 'tool_session' | undefined;\n      if (ap.defaultScope !== undefined) {\n        if (ap.defaultScope !== 'single' && ap.defaultScope !== 'tool_session') {\n          throw new Error(`Invalid gatekeeper policy: externalRestrictions.approvalPolicy.defaultScope must be \"single\" or \"tool_session\", got \"${ap.defaultScope}\"`);\n        }\n        defaultScope = ap.defaultScope;\n      }\n\n      // Validate ttlSeconds\n      let ttlSeconds: number | undefined;\n      if (ap.ttlSeconds !== undefined) {\n        if (typeof ap.ttlSeconds !== 'number' || !Number.isFinite(ap.ttlSeconds)) {\n          throw new Error('Invalid gatekeeper policy: externalRestrictions.approvalPolicy.ttlSeconds must be a number');\n        }\n        if (ap.ttlSeconds < 30 || ap.ttlSeconds > 3600) {\n          throw new Error(`Invalid gatekeeper policy: externalRestrictions.approvalPolicy.ttlSeconds must be between 30 and 3600, got ${ap.ttlSeconds}`);\n        }\n        ttlSeconds = ap.ttlSeconds;\n      }\n\n      approvalPolicy = { requireApproval, defaultScope, ttlSeconds };\n    }\n\n    externalRestrictions = {\n      description: er.description as string,\n      denyPatterns,\n      confirmPatterns,\n      allowPatterns,\n      approvalPolicy,\n    };\n  }\n\n  return {\n    allow,\n    confirm,\n    deny,\n    scopeRestrictions,\n    externalRestrictions,\n  };\n}\n\n/**\n * Validate that a value is an array of strings.\n *\n * @param value - The value to validate\n * @param fieldName - The field name for error messages\n * @returns The validated array, or undefined\n */\nfunction validateStringArray(\n  value: unknown,\n  fieldName: string\n): string[] | undefined {\n  if (value === undefined) {\n    return undefined;\n  }\n\n  if (!Array.isArray(value)) {\n    throw new Error(`Invalid gatekeeper policy: ${fieldName} must be an array`);\n  }\n\n  if (!value.every((item): item is string => typeof item === 'string')) {\n    throw new Error(`Invalid gatekeeper policy: ${fieldName} must contain only strings`);\n  }\n\n  return value;\n}\n\n/**\n * Validate pattern strings for length and non-emptiness.\n *\n * @param patterns - Array of pattern strings to validate\n * @param fieldName - Field name for error messages\n * @throws Error if any pattern is empty or exceeds MAX_GLOB_PATTERN_LENGTH\n */\nfunction validatePatternStrings(\n  patterns: string[],\n  fieldName: string\n): void {\n  for (const pattern of patterns) {\n    if (!pattern) {\n      throw new Error(`Invalid gatekeeper policy: ${fieldName} contains an empty pattern string`);\n    }\n    if (pattern.length > MAX_GLOB_PATTERN_LENGTH) {\n      throw new Error(`Invalid gatekeeper policy: ${fieldName} pattern exceeds maximum length of ${MAX_GLOB_PATTERN_LENGTH} characters`);\n    }\n  }\n}\n\n/**\n * Known CLI tool names used in externalRestrictions patterns.\n * Patterns should start with one of these followed by ':' to be effective.\n */\nconst KNOWN_TOOL_PREFIXES = new Set([\n  'Bash', 'Edit', 'Write', 'Read', 'Glob', 'Grep',\n  'WebFetch', 'WebSearch', 'NotebookEdit',\n]);\n\n/**\n * Regex characters that have no meaning in glob syntax.\n * Users who include these likely intend regex but the matcher only supports * and ?.\n */\nconst REGEX_SYNTAX_PATTERN = /[|(){}[\\]\\\\+^$]/;\n\n/**\n * Analyze externalRestrictions patterns for common mistakes and suspicious syntax.\n *\n * Returns non-fatal warnings to help LLMs and users write effective patterns.\n * Does NOT throw — validation errors are handled by {@link validatePatternStrings}.\n *\n * Checks performed (Issue #1664):\n * - Missing tool prefix (pattern doesn't start with ToolName:)\n * - Overly broad patterns (bare `*` or `ToolName:*`)\n * - Regex syntax that won't work in glob matching\n * - Leading/trailing whitespace\n *\n * @param patterns - Array of pattern strings to analyze\n * @param fieldName - Field name for warning messages\n * @returns Array of warning messages (empty if no issues found)\n */\nexport function analyzePatternSyntax(\n  patterns: string[],\n  fieldName: string\n): string[] {\n  const warnings: string[] = [];\n  for (const pattern of patterns) {\n    analyzeOnePattern(pattern, fieldName, warnings);\n  }\n  return warnings;\n}\n\n/**\n * Analyze a single pattern string and append any warnings found.\n * Extracted from {@link analyzePatternSyntax} to reduce cognitive complexity.\n */\nfunction analyzeOnePattern(\n  pattern: string,\n  fieldName: string,\n  warnings: string[]\n): void {\n  // Check leading/trailing whitespace\n  if (pattern !== pattern.trim()) {\n    warnings.push(\n      `${fieldName} pattern '${pattern}' has leading/trailing whitespace — this may prevent expected matches`\n    );\n  }\n\n  // Check for bare wildcard (matches everything)\n  if (pattern === '*') {\n    warnings.push(\n      `${fieldName} pattern '*' matches everything — this is likely unintentional`\n    );\n    return;\n  }\n\n  // Check for tool prefix and related issues\n  const colonIndex = pattern.indexOf(':');\n  if (colonIndex === -1) {\n    warnings.push(\n      `${fieldName} pattern '${pattern}' has no tool prefix (e.g., 'Bash:', 'Edit:'). ` +\n      `Patterns are matched against 'ToolName:input' strings and will not match without a prefix.`\n    );\n  } else {\n    checkToolPrefix(pattern, colonIndex, fieldName, warnings);\n  }\n\n  // Check for regex syntax that won't work in glob\n  const regexChars = new Set<string>();\n  for (const ch of pattern) {\n    if (REGEX_SYNTAX_PATTERN.test(ch)) regexChars.add(ch);\n  }\n  if (regexChars.size > 0) {\n    warnings.push(\n      `${fieldName} pattern '${pattern}' contains regex syntax '${[...regexChars].join(', ')}' — ` +\n      `only glob wildcards (* and ?) are supported`\n    );\n  }\n}\n\n/**\n * Check tool prefix validity and broadness for a pattern that contains ':'.\n */\nfunction checkToolPrefix(\n  pattern: string,\n  colonIndex: number,\n  fieldName: string,\n  warnings: string[]\n): void {\n  const prefix = pattern.slice(0, colonIndex);\n  const afterColon = pattern.slice(colonIndex + 1);\n\n  // Check for unknown tool prefix\n  if (!KNOWN_TOOL_PREFIXES.has(prefix) && !prefix.startsWith('mcp_')) {\n    warnings.push(\n      `${fieldName} pattern '${pattern}' uses unknown tool prefix '${prefix}'. ` +\n      `Known prefixes: ${[...KNOWN_TOOL_PREFIXES].join(', ')}. MCP tools use 'mcp_' prefix.`\n    );\n  }\n\n  // Check for overly broad ToolName:* pattern\n  if (afterColon === '*') {\n    warnings.push(\n      `${fieldName} pattern '${pattern}' matches ALL ${prefix} operations — verify this is intentional`\n    );\n  }\n}\n\n/**\n * Validate and sanitize a gatekeeper policy during element deserialization.\n *\n * Returns a validated {@link ElementGatekeeperPolicy} if the raw data is\n * structurally valid, or `undefined` if the data is absent, falsy, or\n * malformed. Malformed policies are logged as security events and stripped\n * so they never reach the enforcement pipeline.\n *\n * Use this at every deserialization boundary (manager `parseMetadata` /\n * `sanitizeMetadata` / `parseMemoryFile`) to guarantee that only well-formed\n * policies survive loading.\n *\n * @param rawPolicy - The raw gatekeeper value from parsed YAML/JSON\n * @param elementName - Element name for diagnostic logging\n * @param elementType - Element type for diagnostic logging\n * @returns Validated policy or undefined\n *\n * @since Issue #524 — Runtime validation for all element types\n */\n/**\n * Gatekeeper infrastructure operations — two related sets with distinct purposes.\n *\n * UNGATABLE_OPERATIONS: Operations that must NEVER appear in element policy lists.\n * These are pure internal plumbing — gating them serves no security purpose and\n * breaks critical flows (verification, CLI approval, permission evaluation).\n * Stripped from ALL policy lists (allow, confirm, deny) during sanitization.\n *\n * GATEKEEPER_INFRA_OPERATIONS: Operations that skip Layer 2 (element policy\n * resolution) during primary enforcement. This is a SUPERSET of UNGATABLE_OPERATIONS\n * — it adds confirm_operation, which IS a valid policy target (deny = nuclear sandbox)\n * but must not be gated through the normal enforce() path (that creates the\n * cascading confirmation loop from Issue #758). confirm_operation's element policies\n * are enforced through a separate check in the confirm handler instead.\n *\n * Relationship: GATEKEEPER_INFRA_OPERATIONS = UNGATABLE_OPERATIONS + confirm_operation\n * This derivation is explicit in code to prevent the sets from drifting apart.\n */\nconst UNGATABLE_OPERATIONS = new Set([\n  'verify_challenge',\n  'approve_cli_permission',\n  'permission_prompt',\n]);\n\n/** Derived from UNGATABLE_OPERATIONS + confirm_operation. See block comment above. */\nconst GATEKEEPER_INFRA_OPERATIONS = new Set([\n  ...UNGATABLE_OPERATIONS,\n  'confirm_operation',\n]);\n\n/**\n * Check if an operation is a gatekeeper infrastructure operation that should\n * skip element policy evaluation in the primary enforcement path.\n * Exported for use by MCPAQLHandler. Issue #758.\n */\nexport function isGatekeeperInfraOperation(operation: string): boolean {\n  return GATEKEEPER_INFRA_OPERATIONS.has(operation);\n}\n\n/**\n * Check if any active elements deny confirm_operation (nuclear sandbox).\n * Returns the denying element name if found, undefined otherwise.\n */\nexport function findConfirmDenyingElement(\n  activeElements: Array<{ name: string; type: string; metadata: Record<string, unknown> }>\n): { name: string; type: string } | undefined {\n  for (const element of activeElements) {\n    const gatekeeper = element.metadata?.gatekeeper as Record<string, unknown> | undefined;\n    const denyList = gatekeeper?.deny;\n    if (Array.isArray(denyList) && denyList.includes('confirm_operation')) {\n      return { name: element.name, type: element.type };\n    }\n  }\n  return undefined;\n}\n\n/**\n * Check if any active elements have confirm_operation in their confirm list (advisory).\n * Returns the element names that request additional scrutiny.\n */\nexport function findConfirmAdvisoryElements(\n  activeElements: Array<{ name: string; type: string; metadata: Record<string, unknown> }>\n): Array<{ name: string; type: string }> {\n  const advisories: Array<{ name: string; type: string }> = [];\n  for (const element of activeElements) {\n    const gatekeeper = element.metadata?.gatekeeper as Record<string, unknown> | undefined;\n    const confirmList = gatekeeper?.confirm;\n    if (Array.isArray(confirmList) && confirmList.includes('confirm_operation')) {\n      advisories.push({ name: element.name, type: element.type });\n    }\n  }\n  return advisories;\n}\n\n/**\n * Strip ungatable operations from a policy list and log warnings.\n * Returns the filtered list (may be empty).\n *\n * Note: confirm_operation is NOT stripped from deny lists — it's a legitimate\n * sandbox mechanism. It IS stripped from allow/confirm lists since those are\n * handled through advisory messaging, not through enforce().\n */\nfunction stripUngatable(\n  operations: string[] | undefined,\n  listName: string,\n  elementName: string,\n  elementType: string,\n): string[] | undefined {\n  if (!operations?.length) return operations;\n\n  // confirm_operation in deny lists = nuclear sandbox (preserve)\n  // confirm_operation in confirm lists = advisory marker (preserve — inert in enforce path,\n  //   read at runtime by findConfirmAdvisoryElements(). Safe because skipElementPolicies\n  //   prevents resolveElementPolicy() from ever evaluating confirm_operation as an operation.)\n  // confirm_operation in allow lists = redundant with route default AUTO_APPROVE (strip)\n  const opsToStrip = listName === 'allow'\n    ? new Set([...UNGATABLE_OPERATIONS, 'confirm_operation'])\n    : UNGATABLE_OPERATIONS;\n\n  const stripped = operations.filter(op => {\n    if (opsToStrip.has(op)) {\n      logger.warn(\n        `[Gatekeeper] Stripped \"${op}\" from ${listName} list in ${elementType} \"${elementName}\" — ` +\n        `gatekeeper infrastructure operations cannot be gated by element policies (Issue #758)`\n      );\n      return false;\n    }\n    return true;\n  });\n  return stripped.length > 0 ? stripped : undefined;\n}\n\nexport function sanitizeGatekeeperPolicy(\n  rawPolicy: unknown,\n  elementName: string,\n  elementType: string,\n): ElementGatekeeperPolicy | undefined {\n  if (!rawPolicy || typeof rawPolicy !== 'object') {\n    return undefined;\n  }\n\n  try {\n    // Wrap in a metadata envelope so parseElementPolicy can extract it\n    const validated = parseElementPolicy({ gatekeeper: rawPolicy });\n    if (validated) {\n      // Issue #758: Strip gatekeeper infrastructure operations from element policies\n      // to prevent cascading confirmation loops\n      validated.allow = stripUngatable(validated.allow, 'allow', elementName, elementType);\n      validated.confirm = stripUngatable(validated.confirm, 'confirm', elementName, elementType);\n      validated.deny = stripUngatable(validated.deny, 'deny', elementName, elementType);\n\n      logger.debug(`Loaded gatekeeper policy for ${elementType} \"${elementName}\"`, {\n        allow: validated.allow?.length ?? 0,\n        confirm: validated.confirm?.length ?? 0,\n        deny: validated.deny?.length ?? 0,\n        hasScopeRestrictions: !!validated.scopeRestrictions,\n      });\n    }\n    return validated;\n  } catch (error) {\n    const message = error instanceof Error ? error.message : String(error);\n    SecurityMonitor.logSecurityEvent({\n      type: 'YAML_PARSING_WARNING',\n      severity: 'MEDIUM',\n      source: `${elementType}.deserialize`,\n      details: `Malformed gatekeeper policy in \"${elementName}\" stripped during load: ${message}`,\n    });\n    logger.warn(`Stripped malformed gatekeeper policy from ${elementType} \"${elementName}\": ${message}`);\n    return undefined;\n  }\n}\n"]}
601
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ElementPolicies.js","sourceRoot":"","sources":["../../../../src/handlers/mcp-aql/policies/ElementPolicies.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Element Policies\n *\n * Handles element-based access control for the Gatekeeper.\n * Allows ANY DollhouseMCP element to define policies in its metadata\n * that override or restrict the default operation policies.\n *\n * Policy Resolution Order:\n * 1. Active element deny list (highest priority - blocks operation)\n * 2. Active element confirm list (requires confirmation)\n * 3. Active element allow list (auto-approves)\n * 4. Operation default permission (fallback)\n */\n\nimport {\n  PermissionLevel,\n  type ElementGatekeeperPolicy,\n  type CliApprovalPolicy,\n  type GatekeeperDecision,\n  GatekeeperErrorCode,\n} from '../GatekeeperTypes.js';\nimport { getDefaultPermissionLevel, canOperationBeElevated, getOperationPolicy } from './OperationPolicies.js';\nimport { SecurityMonitor } from '../../../security/securityMonitor.js';\nimport { logger } from '../../../utils/logger.js';\nimport { MAX_GLOB_PATTERN_LENGTH } from '../../../utils/patternMatcher.js';\n\n/**\n * Metadata structure for elements with Gatekeeper policies.\n * The 'gatekeeper' field is optional and contains policy definitions.\n */\nexport interface ElementMetadataWithPolicy {\n  name: string;\n  description?: string;\n  gatekeeper?: ElementGatekeeperPolicy;\n  [key: string]: unknown;\n}\n\n/**\n * Active element for policy evaluation.\n * Represents an element currently active in the session.\n */\nexport interface ActiveElement {\n  type: string;\n  name: string;\n  metadata: ElementMetadataWithPolicy;\n}\n\n/**\n * Result of element policy resolution.\n * Contains the effective permission level and policy source.\n */\nexport interface ElementPolicyResult {\n  /** Effective permission level after element policy application */\n  permissionLevel: PermissionLevel;\n  /** Which element's policy determined this result */\n  sourceElement?: string;\n  /** The specific policy field that matched (allow/confirm/deny) */\n  matchedPolicy?: 'allow' | 'confirm' | 'deny' | 'scope_restriction';\n  /** Whether the operation was blocked by scope restrictions */\n  scopeBlocked?: boolean;\n  /**\n   * Elements that wanted to auto-approve this operation but were overridden\n   * by a higher-priority confirm or deny policy from another element.\n   * Issue #674: allow cannot override confirm.\n   */\n  conflictingElements?: Array<{ name: string; wantedLevel: PermissionLevel }>;\n}\n\n/**\n * Resolve the effective permission level for an operation\n * considering all active elements and their policies.\n *\n * @param operation - The operation to check\n * @param activeElements - Currently active elements with their metadata\n * @param targetElementType - Optional element type being operated on\n * @returns The resolved policy result\n */\nexport function resolveElementPolicy(\n  operation: string,\n  activeElements: ActiveElement[],\n  targetElementType?: string\n): ElementPolicyResult {\n  // Start with the default operation permission level\n  let effectiveLevel = getDefaultPermissionLevel(operation);\n  let sourceElement: string | undefined;\n  let matchedPolicy: ElementPolicyResult['matchedPolicy'];\n  let scopeBlocked = false;\n  // Issue #674: Track whether CONFIRM was set by an element policy (not just the route default)\n  // allow can override the route default but NOT another element's confirm policy\n  let confirmedByElement = false;\n  // Track elements that wanted allow but were overridden by an element-set confirm/deny\n  const conflictingElements: Array<{ name: string; wantedLevel: PermissionLevel }> = [];\n\n  // Check each active element's policy in order\n  for (const element of activeElements) {\n    const policy = element.metadata.gatekeeper;\n    if (!policy) {\n      continue;\n    }\n\n    // 1. Check deny list first (highest priority)\n    if (policy.deny?.includes(operation)) {\n      return {\n        permissionLevel: PermissionLevel.DENY,\n        sourceElement: element.name,\n        matchedPolicy: 'deny',\n        conflictingElements: conflictingElements.length > 0 ? conflictingElements : undefined,\n      };\n    }\n\n    // 2. Check scope restrictions\n    if (targetElementType && policy.scopeRestrictions) {\n      const { allowedTypes, blockedTypes } = policy.scopeRestrictions;\n\n      // If allowedTypes is specified, target must be in the list\n      if (allowedTypes && !allowedTypes.includes(targetElementType)) {\n        scopeBlocked = true;\n        sourceElement = element.name;\n        matchedPolicy = 'scope_restriction';\n        continue; // Check other elements, might have different policy\n      }\n\n      // If blockedTypes is specified, target must NOT be in the list\n      if (blockedTypes?.includes(targetElementType)) {\n        scopeBlocked = true;\n        sourceElement = element.name;\n        matchedPolicy = 'scope_restriction';\n        continue;\n      }\n    }\n\n    // 3. Check confirm list (requires confirmation)\n    if (policy.confirm?.includes(operation)) {\n      // Don't downgrade from DENY or CONFIRM_SINGLE_USE\n      if (effectiveLevel !== PermissionLevel.DENY) {\n        effectiveLevel = PermissionLevel.CONFIRM_SESSION;\n        confirmedByElement = true;\n        sourceElement = element.name;\n        matchedPolicy = 'confirm';\n      }\n    }\n\n    // 4. Check allow list (auto-approves)\n    // Issue #674: allow CAN override the route default, but NOT another element's confirm policy.\n    // Priority hierarchy: element deny > element confirm > element allow > route default\n    if (policy.allow?.includes(operation)) {\n      // Only elevate if the operation allows elevation\n      if (canOperationBeElevated(operation)) {\n        if (!confirmedByElement) {\n          // No element has confirmed this — safe to elevate (overrides route default)\n          effectiveLevel = PermissionLevel.AUTO_APPROVE;\n          sourceElement = element.name;\n          matchedPolicy = 'allow';\n        } else {\n          // Another element's confirm policy takes priority over this allow\n          conflictingElements.push({ name: element.name, wantedLevel: PermissionLevel.AUTO_APPROVE });\n        }\n      }\n    }\n  }\n\n  // If scope was blocked by all elements with restrictions\n  if (scopeBlocked) {\n    return {\n      permissionLevel: PermissionLevel.DENY,\n      sourceElement,\n      matchedPolicy: 'scope_restriction',\n      scopeBlocked: true,\n    };\n  }\n\n  return {\n    permissionLevel: effectiveLevel,\n    sourceElement,\n    matchedPolicy,\n    conflictingElements: conflictingElements.length > 0 ? conflictingElements : undefined,\n  };\n}\n\n/**\n * Create a Gatekeeper decision from element policy resolution.\n *\n * @param operation - The operation that was checked\n * @param result - The element policy resolution result\n * @param targetElementType - Optional element type being operated on\n * @returns A GatekeeperDecision object\n */\nexport function createDecisionFromPolicy(\n  operation: string,\n  result: ElementPolicyResult,\n  targetElementType?: string\n): GatekeeperDecision {\n  const { permissionLevel, sourceElement, scopeBlocked, conflictingElements } = result;\n\n  // Handle DENY\n  if (permissionLevel === PermissionLevel.DENY) {\n    if (scopeBlocked) {\n      return {\n        allowed: false,\n        permissionLevel,\n        errorCode: GatekeeperErrorCode.SCOPE_RESTRICTION,\n        reason: `Operation \"${operation}\" is not allowed on element type \"${targetElementType}\" due to scope restrictions in active element \"${sourceElement}\"`,\n        suggestion: `Deactivate the element \"${sourceElement}\" or use a different element type`,\n        policySource: 'element_policy',\n      };\n    }\n\n    return {\n      allowed: false,\n      permissionLevel,\n      errorCode: GatekeeperErrorCode.ELEMENT_POLICY_VIOLATION,\n      reason: `Operation \"${operation}\" is blocked by active element \"${sourceElement}\"'s deny policy`,\n      suggestion: `Deactivate the element \"${sourceElement}\" to proceed with this operation`,\n      policySource: 'element_policy',\n    };\n  }\n\n  // Handle confirmation required\n  if (\n    permissionLevel === PermissionLevel.CONFIRM_SESSION ||\n    permissionLevel === PermissionLevel.CONFIRM_SINGLE_USE\n  ) {\n    // Build human-readable rationale explaining WHY confirmation is needed\n    const policy = getOperationPolicy(operation);\n    const levelLabel = permissionLevel === PermissionLevel.CONFIRM_SINGLE_USE\n      ? 'requires confirmation each time'\n      : 'requires confirmation once per session';\n\n    let rationale: string;\n    if (sourceElement && result.matchedPolicy === 'confirm') {\n      // Element policy elevated this operation's confirmation requirement\n      const policyDetail = policy?.rationale ? ` ${policy.rationale}` : '';\n      rationale = `Active element \"${sourceElement}\" policy requires confirmation for this operation.${policyDetail}`;\n    } else if (policy?.rationale) {\n      // Route-level default with documented rationale\n      rationale = policy.rationale;\n    } else {\n      // Endpoint-level default, no specific rationale\n      rationale = 'This operation modifies data and requires human approval before proceeding';\n    }\n\n    // Issue #674: Surface elements that wanted to auto-approve but were overridden\n    const conflictNote = conflictingElements && conflictingElements.length > 0\n      ? ` (Note: ${conflictingElements.map(e => `\"${e.name}\"`).join(', ')} would auto-approve this but ${conflictingElements.length === 1 ? 'is' : 'are'} overridden by the confirm policy.)`\n      : '';\n\n    return {\n      allowed: false,\n      permissionLevel,\n      errorCode: GatekeeperErrorCode.CONFIRMATION_REQUIRED,\n      reason: `Operation \"${operation}\" ${levelLabel}. ${rationale}${conflictNote}`,\n      suggestion: 'Use the confirmation dialog to approve this operation',\n      confirmationPending: true,\n      policySource: sourceElement ? 'element_policy' : 'operation_default',\n    };\n  }\n\n  // Handle AUTO_APPROVE\n  return {\n    allowed: true,\n    permissionLevel,\n    reason: sourceElement\n      ? `Operation \"${operation}\" auto-approved by element \"${sourceElement}\"`\n      : `Operation \"${operation}\" auto-approved by default policy`,\n    policySource: sourceElement ? 'element_policy' : 'operation_default',\n  };\n}\n\n/**\n * Parse and validate a Gatekeeper policy from element metadata.\n *\n * @param metadata - The element metadata to parse\n * @returns The parsed policy, or undefined if no policy is defined\n * @throws Error if the policy is malformed\n */\nexport function parseElementPolicy(\n  metadata: unknown\n): ElementGatekeeperPolicy | undefined {\n  if (!metadata || typeof metadata !== 'object') {\n    return undefined;\n  }\n\n  const meta = metadata as Record<string, unknown>;\n  const gatekeeper = meta.gatekeeper;\n\n  if (!gatekeeper) {\n    return undefined;\n  }\n\n  if (typeof gatekeeper !== 'object' || gatekeeper === null) {\n    throw new Error('Invalid gatekeeper policy: must be an object');\n  }\n\n  const policy = gatekeeper as Record<string, unknown>;\n\n  // Validate and extract allow list\n  const allow = validateStringArray(policy.allow, 'allow');\n  const confirm = validateStringArray(policy.confirm, 'confirm');\n  const deny = validateStringArray(policy.deny, 'deny');\n\n  // Validate scope restrictions\n  let scopeRestrictions: ElementGatekeeperPolicy['scopeRestrictions'];\n  if (policy.scopeRestrictions) {\n    if (typeof policy.scopeRestrictions !== 'object' || policy.scopeRestrictions === null) {\n      throw new Error('Invalid gatekeeper policy: scopeRestrictions must be an object');\n    }\n    const sr = policy.scopeRestrictions as Record<string, unknown>;\n    scopeRestrictions = {\n      allowedTypes: validateStringArray(sr.allowedTypes, 'scopeRestrictions.allowedTypes'),\n      blockedTypes: validateStringArray(sr.blockedTypes, 'scopeRestrictions.blockedTypes'),\n    };\n  }\n\n  // Validate external restrictions (Issue #625 Phase 2)\n  let externalRestrictions: ElementGatekeeperPolicy['externalRestrictions'];\n  if (policy.externalRestrictions) {\n    if (typeof policy.externalRestrictions !== 'object' || policy.externalRestrictions === null) {\n      throw new Error('Invalid gatekeeper policy: externalRestrictions must be an object');\n    }\n    const er = policy.externalRestrictions as Record<string, unknown>;\n    if (!er.description || typeof er.description !== 'string') {\n      throw new Error('Invalid gatekeeper policy: externalRestrictions.description is required and must be a non-empty string');\n    }\n    const denyPatterns = validateStringArray(er.denyPatterns, 'externalRestrictions.denyPatterns');\n    const confirmPatterns = validateStringArray(er.confirmPatterns, 'externalRestrictions.confirmPatterns');\n    const allowPatterns = validateStringArray(er.allowPatterns, 'externalRestrictions.allowPatterns');\n    if (denyPatterns) validatePatternStrings(denyPatterns, 'externalRestrictions.denyPatterns');\n    if (confirmPatterns) validatePatternStrings(confirmPatterns, 'externalRestrictions.confirmPatterns');\n    if (allowPatterns) validatePatternStrings(allowPatterns, 'externalRestrictions.allowPatterns');\n    // Validate approvalPolicy (Issue #625 Phase 3)\n    let approvalPolicy: CliApprovalPolicy | undefined = undefined;\n    if (er.approvalPolicy !== undefined) {\n      if (typeof er.approvalPolicy !== 'object' || er.approvalPolicy === null) {\n        throw new Error('Invalid gatekeeper policy: externalRestrictions.approvalPolicy must be an object');\n      }\n      const ap = er.approvalPolicy as Record<string, unknown>;\n\n      // Validate requireApproval\n      let requireApproval: ('moderate' | 'dangerous')[] | undefined;\n      if (ap.requireApproval !== undefined) {\n        if (!Array.isArray(ap.requireApproval)) {\n          throw new Error('Invalid gatekeeper policy: externalRestrictions.approvalPolicy.requireApproval must be an array');\n        }\n        const validLevels = new Set(['moderate', 'dangerous']);\n        for (const level of ap.requireApproval) {\n          if (typeof level !== 'string' || !validLevels.has(level)) {\n            throw new Error(`Invalid gatekeeper policy: externalRestrictions.approvalPolicy.requireApproval contains invalid value \"${level}\". Must be \"moderate\" or \"dangerous\".`);\n          }\n        }\n        requireApproval = ap.requireApproval as ('moderate' | 'dangerous')[];\n      }\n\n      // Validate defaultScope\n      let defaultScope: 'single' | 'tool_session' | undefined;\n      if (ap.defaultScope !== undefined) {\n        if (ap.defaultScope !== 'single' && ap.defaultScope !== 'tool_session') {\n          throw new Error(`Invalid gatekeeper policy: externalRestrictions.approvalPolicy.defaultScope must be \"single\" or \"tool_session\", got \"${ap.defaultScope}\"`);\n        }\n        defaultScope = ap.defaultScope;\n      }\n\n      // Validate ttlSeconds\n      let ttlSeconds: number | undefined;\n      if (ap.ttlSeconds !== undefined) {\n        if (typeof ap.ttlSeconds !== 'number' || !Number.isFinite(ap.ttlSeconds)) {\n          throw new Error('Invalid gatekeeper policy: externalRestrictions.approvalPolicy.ttlSeconds must be a number');\n        }\n        if (ap.ttlSeconds < 30 || ap.ttlSeconds > 3600) {\n          throw new Error(`Invalid gatekeeper policy: externalRestrictions.approvalPolicy.ttlSeconds must be between 30 and 3600, got ${ap.ttlSeconds}`);\n        }\n        ttlSeconds = ap.ttlSeconds;\n      }\n\n      approvalPolicy = { requireApproval, defaultScope, ttlSeconds };\n    }\n\n    externalRestrictions = {\n      description: er.description as string,\n      denyPatterns,\n      confirmPatterns,\n      allowPatterns,\n      approvalPolicy,\n    };\n  }\n\n  return {\n    allow,\n    confirm,\n    deny,\n    scopeRestrictions,\n    externalRestrictions,\n  };\n}\n\n/**\n * Validate authored gatekeeper input before save.\n *\n * Authoring-time validation is stricter than load-time sanitization: it should\n * reject misplaced policy blocks instead of silently saving an element that\n * later appears active but has non-enforceable external restrictions.\n */\nexport function getGatekeeperAuthoringErrors(\n  record: Record<string, unknown> | undefined\n): string[] {\n  if (!record || typeof record !== 'object') {\n    return [];\n  }\n\n  const errors: string[] = [];\n\n  if (Object.hasOwn(record, 'externalRestrictions')) {\n    errors.push(\n      'Invalid gatekeeper policy: externalRestrictions must be nested under gatekeeper.externalRestrictions'\n    );\n  }\n\n  if (record.gatekeeper !== undefined) {\n    try {\n      parseElementPolicy({ gatekeeper: record.gatekeeper });\n    } catch (error) {\n      errors.push(error instanceof Error ? error.message : String(error));\n    }\n  }\n\n  return errors;\n}\n\n/**\n * Validate that a value is an array of strings.\n *\n * @param value - The value to validate\n * @param fieldName - The field name for error messages\n * @returns The validated array, or undefined\n */\nfunction validateStringArray(\n  value: unknown,\n  fieldName: string\n): string[] | undefined {\n  if (value === undefined) {\n    return undefined;\n  }\n\n  if (!Array.isArray(value)) {\n    throw new Error(`Invalid gatekeeper policy: ${fieldName} must be an array`);\n  }\n\n  if (!value.every((item): item is string => typeof item === 'string')) {\n    throw new Error(`Invalid gatekeeper policy: ${fieldName} must contain only strings`);\n  }\n\n  return value;\n}\n\n/**\n * Validate pattern strings for length and non-emptiness.\n *\n * @param patterns - Array of pattern strings to validate\n * @param fieldName - Field name for error messages\n * @throws Error if any pattern is empty or exceeds MAX_GLOB_PATTERN_LENGTH\n */\nfunction validatePatternStrings(\n  patterns: string[],\n  fieldName: string\n): void {\n  for (const pattern of patterns) {\n    if (!pattern) {\n      throw new Error(`Invalid gatekeeper policy: ${fieldName} contains an empty pattern string`);\n    }\n    if (pattern.length > MAX_GLOB_PATTERN_LENGTH) {\n      throw new Error(`Invalid gatekeeper policy: ${fieldName} pattern exceeds maximum length of ${MAX_GLOB_PATTERN_LENGTH} characters`);\n    }\n  }\n}\n\n/**\n * Known CLI tool names used in externalRestrictions patterns.\n * Patterns should start with one of these followed by ':' to be effective.\n */\nconst KNOWN_TOOL_PREFIXES = new Set([\n  'Bash', 'Edit', 'Write', 'Read', 'Glob', 'Grep',\n  'WebFetch', 'WebSearch', 'NotebookEdit',\n]);\n\n/**\n * Regex characters that have no meaning in glob syntax.\n * Users who include these likely intend regex but the matcher only supports * and ?.\n */\nconst REGEX_SYNTAX_PATTERN = /[|(){}[\\]\\\\+^$]/;\n\n/**\n * Analyze externalRestrictions patterns for common mistakes and suspicious syntax.\n *\n * Returns non-fatal warnings to help LLMs and users write effective patterns.\n * Does NOT throw — validation errors are handled by {@link validatePatternStrings}.\n *\n * Checks performed (Issue #1664):\n * - Missing tool prefix (pattern doesn't start with ToolName:)\n * - Overly broad patterns (bare `*` or `ToolName:*`)\n * - Regex syntax that won't work in glob matching\n * - Leading/trailing whitespace\n *\n * @param patterns - Array of pattern strings to analyze\n * @param fieldName - Field name for warning messages\n * @returns Array of warning messages (empty if no issues found)\n */\nexport function analyzePatternSyntax(\n  patterns: string[],\n  fieldName: string\n): string[] {\n  const warnings: string[] = [];\n  for (const pattern of patterns) {\n    analyzeOnePattern(pattern, fieldName, warnings);\n  }\n  return warnings;\n}\n\n/**\n * Analyze a single pattern string and append any warnings found.\n * Extracted from {@link analyzePatternSyntax} to reduce cognitive complexity.\n */\nfunction analyzeOnePattern(\n  pattern: string,\n  fieldName: string,\n  warnings: string[]\n): void {\n  // Check leading/trailing whitespace\n  if (pattern !== pattern.trim()) {\n    warnings.push(\n      `${fieldName} pattern '${pattern}' has leading/trailing whitespace — this may prevent expected matches`\n    );\n  }\n\n  // Check for bare wildcard (matches everything)\n  if (pattern === '*') {\n    warnings.push(\n      `${fieldName} pattern '*' matches everything — this is likely unintentional`\n    );\n    return;\n  }\n\n  // Check for tool prefix and related issues\n  const colonIndex = pattern.indexOf(':');\n  if (colonIndex === -1) {\n    warnings.push(\n      `${fieldName} pattern '${pattern}' has no tool prefix (e.g., 'Bash:', 'Edit:'). ` +\n      `Patterns are matched against 'ToolName:input' strings and will not match without a prefix.`\n    );\n  } else {\n    checkToolPrefix(pattern, colonIndex, fieldName, warnings);\n  }\n\n  // Check for regex syntax that won't work in glob\n  const regexChars = new Set<string>();\n  for (const ch of pattern) {\n    if (REGEX_SYNTAX_PATTERN.test(ch)) regexChars.add(ch);\n  }\n  if (regexChars.size > 0) {\n    warnings.push(\n      `${fieldName} pattern '${pattern}' contains regex syntax '${[...regexChars].join(', ')}' — ` +\n      `only glob wildcards (* and ?) are supported`\n    );\n  }\n}\n\n/**\n * Check tool prefix validity and broadness for a pattern that contains ':'.\n */\nfunction checkToolPrefix(\n  pattern: string,\n  colonIndex: number,\n  fieldName: string,\n  warnings: string[]\n): void {\n  const prefix = pattern.slice(0, colonIndex);\n  const afterColon = pattern.slice(colonIndex + 1);\n\n  // Check for unknown tool prefix\n  if (!KNOWN_TOOL_PREFIXES.has(prefix) && !prefix.startsWith('mcp_')) {\n    warnings.push(\n      `${fieldName} pattern '${pattern}' uses unknown tool prefix '${prefix}'. ` +\n      `Known prefixes: ${[...KNOWN_TOOL_PREFIXES].join(', ')}. MCP tools use 'mcp_' prefix.`\n    );\n  }\n\n  // Check for overly broad ToolName:* pattern\n  if (afterColon === '*') {\n    warnings.push(\n      `${fieldName} pattern '${pattern}' matches ALL ${prefix} operations — verify this is intentional`\n    );\n  }\n}\n\n/**\n * Validate and sanitize a gatekeeper policy during element deserialization.\n *\n * Returns a validated {@link ElementGatekeeperPolicy} if the raw data is\n * structurally valid, or `undefined` if the data is absent, falsy, or\n * malformed. Malformed policies are logged as security events and stripped\n * so they never reach the enforcement pipeline.\n *\n * Use this at every deserialization boundary (manager `parseMetadata` /\n * `sanitizeMetadata` / `parseMemoryFile`) to guarantee that only well-formed\n * policies survive loading.\n *\n * @param rawPolicy - The raw gatekeeper value from parsed YAML/JSON\n * @param elementName - Element name for diagnostic logging\n * @param elementType - Element type for diagnostic logging\n * @returns Validated policy or undefined\n *\n * @since Issue #524 — Runtime validation for all element types\n */\n/**\n * Gatekeeper infrastructure operations — two related sets with distinct purposes.\n *\n * UNGATABLE_OPERATIONS: Operations that must NEVER appear in element policy lists.\n * These are pure internal plumbing — gating them serves no security purpose and\n * breaks critical flows (verification, CLI approval, permission evaluation).\n * Stripped from ALL policy lists (allow, confirm, deny) during sanitization.\n *\n * GATEKEEPER_INFRA_OPERATIONS: Operations that skip Layer 2 (element policy\n * resolution) during primary enforcement. This is a SUPERSET of UNGATABLE_OPERATIONS\n * — it adds confirm_operation, which IS a valid policy target (deny = nuclear sandbox)\n * but must not be gated through the normal enforce() path (that creates the\n * cascading confirmation loop from Issue #758). confirm_operation's element policies\n * are enforced through a separate check in the confirm handler instead.\n *\n * Relationship: GATEKEEPER_INFRA_OPERATIONS = UNGATABLE_OPERATIONS + confirm_operation\n * This derivation is explicit in code to prevent the sets from drifting apart.\n */\nconst UNGATABLE_OPERATIONS = new Set([\n  'verify_challenge',\n  'approve_cli_permission',\n  'permission_prompt',\n]);\n\n/** Derived from UNGATABLE_OPERATIONS + confirm_operation. See block comment above. */\nconst GATEKEEPER_INFRA_OPERATIONS = new Set([\n  ...UNGATABLE_OPERATIONS,\n  'confirm_operation',\n]);\n\n/**\n * Check if an operation is a gatekeeper infrastructure operation that should\n * skip element policy evaluation in the primary enforcement path.\n * Exported for use by MCPAQLHandler. Issue #758.\n */\nexport function isGatekeeperInfraOperation(operation: string): boolean {\n  return GATEKEEPER_INFRA_OPERATIONS.has(operation);\n}\n\n/**\n * Check if any active elements deny confirm_operation (nuclear sandbox).\n * Returns the denying element name if found, undefined otherwise.\n */\nexport function findConfirmDenyingElement(\n  activeElements: Array<{ name: string; type: string; metadata: Record<string, unknown> }>\n): { name: string; type: string } | undefined {\n  for (const element of activeElements) {\n    const gatekeeper = element.metadata?.gatekeeper as Record<string, unknown> | undefined;\n    const denyList = gatekeeper?.deny;\n    if (Array.isArray(denyList) && denyList.includes('confirm_operation')) {\n      return { name: element.name, type: element.type };\n    }\n  }\n  return undefined;\n}\n\n/**\n * Check if any active elements have confirm_operation in their confirm list (advisory).\n * Returns the element names that request additional scrutiny.\n */\nexport function findConfirmAdvisoryElements(\n  activeElements: Array<{ name: string; type: string; metadata: Record<string, unknown> }>\n): Array<{ name: string; type: string }> {\n  const advisories: Array<{ name: string; type: string }> = [];\n  for (const element of activeElements) {\n    const gatekeeper = element.metadata?.gatekeeper as Record<string, unknown> | undefined;\n    const confirmList = gatekeeper?.confirm;\n    if (Array.isArray(confirmList) && confirmList.includes('confirm_operation')) {\n      advisories.push({ name: element.name, type: element.type });\n    }\n  }\n  return advisories;\n}\n\n/**\n * Strip ungatable operations from a policy list and log warnings.\n * Returns the filtered list (may be empty).\n *\n * Note: confirm_operation is NOT stripped from deny lists — it's a legitimate\n * sandbox mechanism. It IS stripped from allow/confirm lists since those are\n * handled through advisory messaging, not through enforce().\n */\nfunction stripUngatable(\n  operations: string[] | undefined,\n  listName: string,\n  elementName: string,\n  elementType: string,\n): string[] | undefined {\n  if (!operations?.length) return operations;\n\n  // confirm_operation in deny lists = nuclear sandbox (preserve)\n  // confirm_operation in confirm lists = advisory marker (preserve — inert in enforce path,\n  //   read at runtime by findConfirmAdvisoryElements(). Safe because skipElementPolicies\n  //   prevents resolveElementPolicy() from ever evaluating confirm_operation as an operation.)\n  // confirm_operation in allow lists = redundant with route default AUTO_APPROVE (strip)\n  const opsToStrip = listName === 'allow'\n    ? new Set([...UNGATABLE_OPERATIONS, 'confirm_operation'])\n    : UNGATABLE_OPERATIONS;\n\n  const stripped = operations.filter(op => {\n    if (opsToStrip.has(op)) {\n      logger.warn(\n        `[Gatekeeper] Stripped \"${op}\" from ${listName} list in ${elementType} \"${elementName}\" — ` +\n        `gatekeeper infrastructure operations cannot be gated by element policies (Issue #758)`\n      );\n      return false;\n    }\n    return true;\n  });\n  return stripped.length > 0 ? stripped : undefined;\n}\n\nexport function sanitizeGatekeeperPolicy(\n  rawPolicy: unknown,\n  elementName: string,\n  elementType: string,\n): ElementGatekeeperPolicy | undefined {\n  if (!rawPolicy || typeof rawPolicy !== 'object') {\n    return undefined;\n  }\n\n  try {\n    // Wrap in a metadata envelope so parseElementPolicy can extract it\n    const validated = parseElementPolicy({ gatekeeper: rawPolicy });\n    if (validated) {\n      // Issue #758: Strip gatekeeper infrastructure operations from element policies\n      // to prevent cascading confirmation loops\n      validated.allow = stripUngatable(validated.allow, 'allow', elementName, elementType);\n      validated.confirm = stripUngatable(validated.confirm, 'confirm', elementName, elementType);\n      validated.deny = stripUngatable(validated.deny, 'deny', elementName, elementType);\n\n      logger.debug(`Loaded gatekeeper policy for ${elementType} \"${elementName}\"`, {\n        allow: validated.allow?.length ?? 0,\n        confirm: validated.confirm?.length ?? 0,\n        deny: validated.deny?.length ?? 0,\n        hasScopeRestrictions: !!validated.scopeRestrictions,\n      });\n    }\n    return validated;\n  } catch (error) {\n    const message = error instanceof Error ? error.message : String(error);\n    SecurityMonitor.logSecurityEvent({\n      type: 'YAML_PARSING_WARNING',\n      severity: 'MEDIUM',\n      source: `${elementType}.deserialize`,\n      details: `Malformed gatekeeper policy in \"${elementName}\" stripped during load: ${message}`,\n    });\n    logger.warn(`Stripped malformed gatekeeper policy from ${elementType} \"${elementName}\": ${message}`);\n    return undefined;\n  }\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":"AAAA;;;;;;GAMG;AAEH,OAAO,EAAE,mBAAmB,IAAI,UAAU,EAAE,MAAM,4BAA4B,CAAC;AAC/E,OAAO,EAAE,oBAAoB,EAAE,MAAM,6BAA6B,CAAC;AAGnE;;GAEG;AACH,MAAM,OAAgB,sBAAsB;IAC1C;;;;;OAKG;IACO,KAAK,CAAC,mBAAmB,CAAC,IAAY,EAAE,WAAkB;QAClE,OAAO,UAAU,CAAC,IAAI,EAAE,WAAW,CAAC,CAAC;IACvC,CAAC;IAED;;;;OAIG;IACO,mBAAmB,CAAC,OAAe;QAC3C,OAAO;YACL,OAAO,EAAE,CAAC;oBACR,IAAI,EAAE,MAAM;oBACZ,IAAI,EAAE,OAAO;iBACd,CAAC;SACH,CAAC;IACJ,CAAC;IAED;;;;OAIG;IACO,qBAAqB,CAAC,OAAe;QAC7C,OAAO;YACL,OAAO,EAAE,CAAC;oBACR,IAAI,EAAE,MAAM;oBACZ,IAAI,EAAE,OAAO;iBACd,CAAC;SACH,CAAC;IACJ,CAAC;IAED;;;;;;;;;;OAUG;IACO,kBAAkB,CAAC,IAAY,EAAE,IAAY;QACrD,MAAM,IAAI,oBAAoB,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;IAC7C,CAAC;IAED;;;;OAIG;IACO,sBAAsB,CAAC,IAAY,EAAE,IAAY;QACzD,sFAAsF;QACtF,OAAO,IAAI,CAAC,mBAAmB,CAAC,KAAK,IAAI,KAAK,IAAI,aAAa,CAAC,CAAC;IACnE,CAAC;IAED;;;;;;;;OAQG;IACO,wBAAwB,CAAC,QAAiC;QAClE,MAAM,UAAU,GAAG,QAAQ,EAAE,UAAiD,CAAC;QAC/E,MAAM,YAAY,GAAG,UAAU,EAAE,oBAA2D,CAAC;QAC7F,IAAI,CAAC,YAAY;YAAE,OAAO,EAAE,CAAC;QAE7B,MAAM,KAAK,GAAa,CAAC,qCAAqC,CAAC,CAAC;QAChE,IAAI,YAAY,CAAC,WAAW,EAAE,CAAC;YAC7B,KAAK,CAAC,IAAI,CAAC,KAAK,YAAY,CAAC,WAAW,EAAE,CAAC,CAAC;QAC9C,CAAC;QACD,MAAM,YAAY,GAAG,YAAY,CAAC,YAAoC,CAAC;QACvE,IAAI,YAAY,EAAE,MAAM,EAAE,CAAC;YACzB,MAAM,KAAK,GAAG,YAAY,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAClD,KAAK,CAAC,IAAI,CAAC,aAAa,KAAK,GAAG,YAAY,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;QAC1E,CAAC;QACD,MAAM,aAAa,GAAG,YAAY,CAAC,aAAqC,CAAC;QACzE,IAAI,aAAa,EAAE,MAAM,EAAE,CAAC;YAC1B,MAAM,KAAK,GAAG,aAAa,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YACnD,KAAK,CAAC,IAAI,CAAC,mBAAmB,KAAK,GAAG,aAAa,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;QACjF,CAAC;QACD,MAAM,cAAc,GAAG,YAAY,CAAC,cAAqD,CAAC;QAC1F,MAAM,eAAe,GAAG,cAAc,EAAE,eAAuC,CAAC;QAChF,IAAI,eAAe,EAAE,MAAM,EAAE,CAAC;YAC5B,KAAK,CAAC,IAAI,CAAC,4BAA4B,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;QAClF,CAAC;QACD,KAAK,CAAC,IAAI,CAAC,kEAAkE,CAAC,CAAC;QAC/E,KAAK,CAAC,IAAI,CAAC,kHAAkH,CAAC,CAAC;QAC/H,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAC1B,CAAC;CACF","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"]}