@dollhousemcp/mcp-server 2.0.17 → 2.0.19

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 (78) hide show
  1. package/CHANGELOG.md +10 -0
  2. package/dist/elements/BaseElement.d.ts +1 -0
  3. package/dist/elements/BaseElement.d.ts.map +1 -1
  4. package/dist/elements/BaseElement.js +7 -1
  5. package/dist/elements/agents/AgentManager.js +2 -2
  6. package/dist/elements/base/ElementFileOperations.js +2 -2
  7. package/dist/elements/ensembles/EnsembleManager.js +3 -3
  8. package/dist/elements/memories/MemoryManager.js +2 -2
  9. package/dist/elements/skills/SkillManager.js +2 -2
  10. package/dist/elements/templates/TemplateManager.js +2 -2
  11. package/dist/generated/version.d.ts +2 -2
  12. package/dist/generated/version.js +3 -3
  13. package/dist/handlers/ElementCRUDHandler.d.ts +30 -0
  14. package/dist/handlers/ElementCRUDHandler.d.ts.map +1 -1
  15. package/dist/handlers/ElementCRUDHandler.js +142 -2
  16. package/dist/handlers/mcp-aql/MCPAQLHandler.d.ts +4 -0
  17. package/dist/handlers/mcp-aql/MCPAQLHandler.d.ts.map +1 -1
  18. package/dist/handlers/mcp-aql/MCPAQLHandler.js +132 -14
  19. package/dist/handlers/mcp-aql/OperationRouter.d.ts.map +1 -1
  20. package/dist/handlers/mcp-aql/OperationRouter.js +6 -1
  21. package/dist/handlers/mcp-aql/OperationSchema.d.ts.map +1 -1
  22. package/dist/handlers/mcp-aql/OperationSchema.js +17 -1
  23. package/dist/handlers/mcp-aql/policies/AgentToolPolicyTranslator.d.ts.map +1 -1
  24. package/dist/handlers/mcp-aql/policies/AgentToolPolicyTranslator.js +2 -1
  25. package/dist/handlers/mcp-aql/policies/ElementPolicies.d.ts +9 -1
  26. package/dist/handlers/mcp-aql/policies/ElementPolicies.d.ts.map +1 -1
  27. package/dist/handlers/mcp-aql/policies/ElementPolicies.js +64 -4
  28. package/dist/handlers/mcp-aql/policies/OperationPolicies.d.ts.map +1 -1
  29. package/dist/handlers/mcp-aql/policies/OperationPolicies.js +6 -1
  30. package/dist/handlers/mcp-aql/policies/ToolClassification.d.ts.map +1 -1
  31. package/dist/handlers/mcp-aql/policies/ToolClassification.js +2 -1
  32. package/dist/handlers/strategies/AgentActivationStrategy.d.ts.map +1 -1
  33. package/dist/handlers/strategies/AgentActivationStrategy.js +5 -1
  34. package/dist/handlers/strategies/BaseActivationStrategy.d.ts +1 -0
  35. package/dist/handlers/strategies/BaseActivationStrategy.d.ts.map +1 -1
  36. package/dist/handlers/strategies/BaseActivationStrategy.js +15 -1
  37. package/dist/handlers/strategies/EnsembleActivationStrategy.d.ts.map +1 -1
  38. package/dist/handlers/strategies/EnsembleActivationStrategy.js +5 -1
  39. package/dist/handlers/strategies/MemoryActivationStrategy.d.ts.map +1 -1
  40. package/dist/handlers/strategies/MemoryActivationStrategy.js +5 -1
  41. package/dist/handlers/strategies/PersonaActivationStrategy.d.ts.map +1 -1
  42. package/dist/handlers/strategies/PersonaActivationStrategy.js +5 -1
  43. package/dist/handlers/strategies/SkillActivationStrategy.d.ts.map +1 -1
  44. package/dist/handlers/strategies/SkillActivationStrategy.js +5 -1
  45. package/dist/handlers/strategies/TemplateActivationStrategy.d.ts.map +1 -1
  46. package/dist/handlers/strategies/TemplateActivationStrategy.js +7 -2
  47. package/dist/persona/PersonaElement.js +2 -2
  48. package/dist/server/tools/MCPAQLTools.js +2 -1
  49. package/dist/services/SerializationService.d.ts.map +1 -1
  50. package/dist/services/SerializationService.js +7 -1
  51. package/dist/types/elements/IElement.d.ts +9 -0
  52. package/dist/types/elements/IElement.d.ts.map +1 -1
  53. package/dist/types/elements/IElement.js +1 -1
  54. package/dist/web/console/IngestRoutes.d.ts +6 -0
  55. package/dist/web/console/IngestRoutes.d.ts.map +1 -1
  56. package/dist/web/console/IngestRoutes.js +38 -9
  57. package/dist/web/console/LeaderElection.d.ts +39 -0
  58. package/dist/web/console/LeaderElection.d.ts.map +1 -1
  59. package/dist/web/console/LeaderElection.js +147 -29
  60. package/dist/web/console/LeaderForwardingSink.d.ts.map +1 -1
  61. package/dist/web/console/LeaderForwardingSink.js +5 -1
  62. package/dist/web/console/PromotionManager.d.ts.map +1 -1
  63. package/dist/web/console/PromotionManager.js +3 -11
  64. package/dist/web/public/app.js +62 -1
  65. package/dist/web/public/index.html +19 -17
  66. package/dist/web/public/permissions.css +190 -2
  67. package/dist/web/public/permissions.js +171 -30
  68. package/dist/web/public/sessions.js +111 -0
  69. package/dist/web/public/setup.js +131 -60
  70. package/dist/web/routes/permissionRoutes.d.ts.map +1 -1
  71. package/dist/web/routes/permissionRoutes.js +77 -5
  72. package/dist/web/routes/setupRoutes.d.ts.map +1 -1
  73. package/dist/web/routes/setupRoutes.js +16 -2
  74. package/dist/web/server.d.ts.map +1 -1
  75. package/dist/web/server.js +12 -10
  76. package/package.json +1 -1
  77. package/scripts/pretooluse-dollhouse.sh +39 -1
  78. package/server.json +2 -2
@@ -1 +1 @@
1
- {"version":3,"file":"AgentToolPolicyTranslator.d.ts","sourceRoot":"","sources":["../../../../src/handlers/mcp-aql/policies/AgentToolPolicyTranslator.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;GAcG;AAEH,OAAO,KAAK,EAAE,uBAAuB,EAAE,MAAM,uBAAuB,CAAC;AACrE,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,mCAAmC,CAAC;AA+CzE;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAgCG;AACH,wBAAgB,2BAA2B,CACzC,UAAU,EAAE,eAAe,GAC1B,uBAAuB,GAAG,SAAS,CA4CrC"}
1
+ {"version":3,"file":"AgentToolPolicyTranslator.d.ts","sourceRoot":"","sources":["../../../../src/handlers/mcp-aql/policies/AgentToolPolicyTranslator.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;GAcG;AAEH,OAAO,KAAK,EAAE,uBAAuB,EAAE,MAAM,uBAAuB,CAAC;AACrE,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,mCAAmC,CAAC;AAgDzE;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAgCG;AACH,wBAAgB,2BAA2B,CACzC,UAAU,EAAE,eAAe,GAC1B,uBAAuB,GAAG,SAAS,CA4CrC"}
@@ -37,6 +37,7 @@ const EXEMPT_OPERATIONS = new Set([
37
37
  // Safety system — agent must interact with Gatekeeper/verification
38
38
  'confirm_operation',
39
39
  'verify_challenge',
40
+ 'release_deadlock',
40
41
  'permission_prompt', // Issue #625: CLI-level permission delegation
41
42
  'approve_cli_permission', // Issue #625 Phase 3: CLI approval workflow
42
43
  'get_pending_cli_approvals', // Issue #625 Phase 3: CLI approval visibility
@@ -128,4 +129,4 @@ export function translateToolConfigToPolicy(toolConfig) {
128
129
  }
129
130
  return { deny: Array.from(denySet).sort((a, b) => a.localeCompare(b)) };
130
131
  }
131
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"AgentToolPolicyTranslator.js","sourceRoot":"","sources":["../../../../src/handlers/mcp-aql/policies/AgentToolPolicyTranslator.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Agent Tool Policy Translator (Issue #449)\n *\n * Translates an agent's `tools` configuration (allowed/denied endpoint names)\n * into an `ElementGatekeeperPolicy` that the Gatekeeper can enforce.\n *\n * This bridges the informational `AgentToolConfig` with the enforceable\n * Gatekeeper policy system, giving agent tool restrictions programmatic teeth.\n *\n * **Policy precedence:** This translator is only used when an agent has no\n * explicit `gatekeeper` policy. If both `gatekeeper` and `tools` are present,\n * the explicit `gatekeeper` policy takes precedence (see MCPAQLHandler).\n *\n * @module AgentToolPolicyTranslator\n */\n\nimport type { ElementGatekeeperPolicy } from '../GatekeeperTypes.js';\nimport type { AgentToolConfig } from '../../../elements/agents/types.js';\nimport { getOperationsForEndpoint } from '../OperationRouter.js';\nimport type { CRUDEndpoint } from '../OperationRouter.js';\n\n/**\n * Operations that are exempt from tools-based deny synthesis.\n * These must always be available for agent lifecycle and safety flows.\n *\n * - **Execution lifecycle**: An agent must be able to manage its own execution\n *   (execute, get state, update state, complete, continue).\n * - **Safety system**: An agent must interact with Gatekeeper confirmation\n *   and danger zone verification flows.\n */\nconst EXEMPT_OPERATIONS = new Set([\n  // Execution lifecycle — agent must manage its own execution\n  'execute_agent',\n  'get_execution_state',\n  'record_execution_step',\n  'complete_execution',\n  'continue_execution',\n  'abort_execution', // Issue #249\n  'get_gathered_data', // Issue #68\n  'prepare_handoff', // Issue #69\n  'resume_from_handoff', // Issue #69\n  // Safety system — agent must interact with Gatekeeper/verification\n  'confirm_operation',\n  'verify_challenge',\n  'permission_prompt', // Issue #625: CLI-level permission delegation\n  'approve_cli_permission', // Issue #625 Phase 3: CLI approval workflow\n  'get_pending_cli_approvals', // Issue #625 Phase 3: CLI approval visibility\n]);\n\n/**\n * Maps MCP-AQL tool names (as used in `AgentToolConfig.allowed/denied`)\n * to their corresponding CRUD endpoint types in the OperationRouter.\n *\n * Tool names follow the format `mcp_aql_{endpoint}` and map to one of\n * the five CRUD endpoints: CREATE, READ, UPDATE, DELETE, EXECUTE.\n */\nconst ENDPOINT_TOOL_MAP = new Map<string, CRUDEndpoint>([\n  ['mcp_aql_create', 'CREATE'],\n  ['mcp_aql_read', 'READ'],\n  ['mcp_aql_update', 'UPDATE'],\n  ['mcp_aql_delete', 'DELETE'],\n  ['mcp_aql_execute', 'EXECUTE'],\n]);\n\n/**\n * Translates an {@link AgentToolConfig} into an {@link ElementGatekeeperPolicy}.\n *\n * The translation works as follows:\n * - If `tools.allowed` is specified, all operations **not** in the allowed\n *   endpoints are added to the deny list (allowlist → denylist inversion).\n * - If `tools.denied` is specified, all operations from the denied endpoints\n *   are added to the deny list directly.\n * - Both `allowed` and `denied` can be specified simultaneously; their effects\n *   are cumulative (union of denied operations).\n *\n * Lifecycle and safety operations ({@link EXEMPT_OPERATIONS}) are **never**\n * included in the synthesized deny list, regardless of the tool config.\n *\n * @param toolConfig - The agent's tool configuration containing allowed/denied endpoint names\n * @returns A synthesized {@link ElementGatekeeperPolicy} with a deny list,\n *          or `undefined` if the config produces no restrictions\n *\n * @example\n * ```typescript\n * // Agent that can only read — all other endpoints denied\n * translateToolConfigToPolicy({ allowed: ['mcp_aql_read'] });\n * // → { deny: ['addEntry', 'clear', 'create_element', 'delete_element', 'edit_element', ...] }\n *\n * // Agent that cannot delete — only delete operations denied\n * translateToolConfigToPolicy({ allowed: ['mcp_aql_create', 'mcp_aql_read', 'mcp_aql_update', 'mcp_aql_execute'] });\n * // → { deny: ['clear', 'delete_element'] }\n *\n * // No restrictions — returns undefined\n * translateToolConfigToPolicy({ allowed: ['mcp_aql_create', 'mcp_aql_read', 'mcp_aql_update', 'mcp_aql_delete', 'mcp_aql_execute'] });\n * // → undefined\n * ```\n */\nexport function translateToolConfigToPolicy(\n  toolConfig: AgentToolConfig\n): ElementGatekeeperPolicy | undefined {\n  const denySet = new Set<string>();\n\n  if (toolConfig.allowed && toolConfig.allowed.length > 0) {\n    // Compute the set of allowed operations from allowed endpoints\n    const allowedOps = new Set<string>();\n    for (const toolName of toolConfig.allowed) {\n      const endpoint = ENDPOINT_TOOL_MAP.get(toolName);\n      if (endpoint) {\n        for (const op of getOperationsForEndpoint(endpoint)) {\n          allowedOps.add(op);\n        }\n      }\n    }\n\n    // Everything NOT in the allowed set gets denied (except exempt operations)\n    for (const endpoint of ENDPOINT_TOOL_MAP.values()) {\n      for (const op of getOperationsForEndpoint(endpoint)) {\n        if (!allowedOps.has(op) && !EXEMPT_OPERATIONS.has(op)) {\n          denySet.add(op);\n        }\n      }\n    }\n  }\n\n  if (toolConfig.denied && toolConfig.denied.length > 0) {\n    // Add all operations from denied endpoints\n    for (const toolName of toolConfig.denied) {\n      const endpoint = ENDPOINT_TOOL_MAP.get(toolName);\n      if (endpoint) {\n        for (const op of getOperationsForEndpoint(endpoint)) {\n          if (!EXEMPT_OPERATIONS.has(op)) {\n            denySet.add(op);\n          }\n        }\n      }\n    }\n  }\n\n  if (denySet.size === 0) {\n    return undefined;\n  }\n\n  return { deny: Array.from(denySet).sort((a, b) => a.localeCompare(b)) };\n}\n"]}
132
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"AgentToolPolicyTranslator.js","sourceRoot":"","sources":["../../../../src/handlers/mcp-aql/policies/AgentToolPolicyTranslator.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Agent Tool Policy Translator (Issue #449)\n *\n * Translates an agent's `tools` configuration (allowed/denied endpoint names)\n * into an `ElementGatekeeperPolicy` that the Gatekeeper can enforce.\n *\n * This bridges the informational `AgentToolConfig` with the enforceable\n * Gatekeeper policy system, giving agent tool restrictions programmatic teeth.\n *\n * **Policy precedence:** This translator is only used when an agent has no\n * explicit `gatekeeper` policy. If both `gatekeeper` and `tools` are present,\n * the explicit `gatekeeper` policy takes precedence (see MCPAQLHandler).\n *\n * @module AgentToolPolicyTranslator\n */\n\nimport type { ElementGatekeeperPolicy } from '../GatekeeperTypes.js';\nimport type { AgentToolConfig } from '../../../elements/agents/types.js';\nimport { getOperationsForEndpoint } from '../OperationRouter.js';\nimport type { CRUDEndpoint } from '../OperationRouter.js';\n\n/**\n * Operations that are exempt from tools-based deny synthesis.\n * These must always be available for agent lifecycle and safety flows.\n *\n * - **Execution lifecycle**: An agent must be able to manage its own execution\n *   (execute, get state, update state, complete, continue).\n * - **Safety system**: An agent must interact with Gatekeeper confirmation\n *   and danger zone verification flows.\n */\nconst EXEMPT_OPERATIONS = new Set([\n  // Execution lifecycle — agent must manage its own execution\n  'execute_agent',\n  'get_execution_state',\n  'record_execution_step',\n  'complete_execution',\n  'continue_execution',\n  'abort_execution', // Issue #249\n  'get_gathered_data', // Issue #68\n  'prepare_handoff', // Issue #69\n  'resume_from_handoff', // Issue #69\n  // Safety system — agent must interact with Gatekeeper/verification\n  'confirm_operation',\n  'verify_challenge',\n  'release_deadlock',\n  'permission_prompt', // Issue #625: CLI-level permission delegation\n  'approve_cli_permission', // Issue #625 Phase 3: CLI approval workflow\n  'get_pending_cli_approvals', // Issue #625 Phase 3: CLI approval visibility\n]);\n\n/**\n * Maps MCP-AQL tool names (as used in `AgentToolConfig.allowed/denied`)\n * to their corresponding CRUD endpoint types in the OperationRouter.\n *\n * Tool names follow the format `mcp_aql_{endpoint}` and map to one of\n * the five CRUD endpoints: CREATE, READ, UPDATE, DELETE, EXECUTE.\n */\nconst ENDPOINT_TOOL_MAP = new Map<string, CRUDEndpoint>([\n  ['mcp_aql_create', 'CREATE'],\n  ['mcp_aql_read', 'READ'],\n  ['mcp_aql_update', 'UPDATE'],\n  ['mcp_aql_delete', 'DELETE'],\n  ['mcp_aql_execute', 'EXECUTE'],\n]);\n\n/**\n * Translates an {@link AgentToolConfig} into an {@link ElementGatekeeperPolicy}.\n *\n * The translation works as follows:\n * - If `tools.allowed` is specified, all operations **not** in the allowed\n *   endpoints are added to the deny list (allowlist → denylist inversion).\n * - If `tools.denied` is specified, all operations from the denied endpoints\n *   are added to the deny list directly.\n * - Both `allowed` and `denied` can be specified simultaneously; their effects\n *   are cumulative (union of denied operations).\n *\n * Lifecycle and safety operations ({@link EXEMPT_OPERATIONS}) are **never**\n * included in the synthesized deny list, regardless of the tool config.\n *\n * @param toolConfig - The agent's tool configuration containing allowed/denied endpoint names\n * @returns A synthesized {@link ElementGatekeeperPolicy} with a deny list,\n *          or `undefined` if the config produces no restrictions\n *\n * @example\n * ```typescript\n * // Agent that can only read — all other endpoints denied\n * translateToolConfigToPolicy({ allowed: ['mcp_aql_read'] });\n * // → { deny: ['addEntry', 'clear', 'create_element', 'delete_element', 'edit_element', ...] }\n *\n * // Agent that cannot delete — only delete operations denied\n * translateToolConfigToPolicy({ allowed: ['mcp_aql_create', 'mcp_aql_read', 'mcp_aql_update', 'mcp_aql_execute'] });\n * // → { deny: ['clear', 'delete_element'] }\n *\n * // No restrictions — returns undefined\n * translateToolConfigToPolicy({ allowed: ['mcp_aql_create', 'mcp_aql_read', 'mcp_aql_update', 'mcp_aql_delete', 'mcp_aql_execute'] });\n * // → undefined\n * ```\n */\nexport function translateToolConfigToPolicy(\n  toolConfig: AgentToolConfig\n): ElementGatekeeperPolicy | undefined {\n  const denySet = new Set<string>();\n\n  if (toolConfig.allowed && toolConfig.allowed.length > 0) {\n    // Compute the set of allowed operations from allowed endpoints\n    const allowedOps = new Set<string>();\n    for (const toolName of toolConfig.allowed) {\n      const endpoint = ENDPOINT_TOOL_MAP.get(toolName);\n      if (endpoint) {\n        for (const op of getOperationsForEndpoint(endpoint)) {\n          allowedOps.add(op);\n        }\n      }\n    }\n\n    // Everything NOT in the allowed set gets denied (except exempt operations)\n    for (const endpoint of ENDPOINT_TOOL_MAP.values()) {\n      for (const op of getOperationsForEndpoint(endpoint)) {\n        if (!allowedOps.has(op) && !EXEMPT_OPERATIONS.has(op)) {\n          denySet.add(op);\n        }\n      }\n    }\n  }\n\n  if (toolConfig.denied && toolConfig.denied.length > 0) {\n    // Add all operations from denied endpoints\n    for (const toolName of toolConfig.denied) {\n      const endpoint = ENDPOINT_TOOL_MAP.get(toolName);\n      if (endpoint) {\n        for (const op of getOperationsForEndpoint(endpoint)) {\n          if (!EXEMPT_OPERATIONS.has(op)) {\n            denySet.add(op);\n          }\n        }\n      }\n    }\n  }\n\n  if (denySet.size === 0) {\n    return undefined;\n  }\n\n  return { deny: Array.from(denySet).sort((a, b) => a.localeCompare(b)) };\n}\n"]}
@@ -31,6 +31,11 @@ export interface ActiveElement {
31
31
  name: string;
32
32
  metadata: ElementMetadataWithPolicy;
33
33
  }
34
+ export interface GatekeeperPolicyDiagnostics {
35
+ valid: false;
36
+ enforceable: false;
37
+ message: string;
38
+ }
34
39
  /**
35
40
  * Result of element policy resolution.
36
41
  * Contains the effective permission level and policy source.
@@ -136,5 +141,8 @@ export declare function findConfirmAdvisoryElements(activeElements: Array<{
136
141
  name: string;
137
142
  type: string;
138
143
  }>;
139
- export declare function sanitizeGatekeeperPolicy(rawPolicy: unknown, elementName: string, elementType: string): ElementGatekeeperPolicy | undefined;
144
+ export declare function sanitizeGatekeeperPolicy(rawPolicy: unknown, elementName: string, elementType: string, diagnosticsTarget?: Record<string, unknown>): ElementGatekeeperPolicy | undefined;
145
+ export declare function attachGatekeeperDiagnostics(target: unknown, message: string): void;
146
+ export declare function clearGatekeeperDiagnostics(target: unknown): void;
147
+ export declare function getGatekeeperDiagnostics(target: unknown): GatekeeperPolicyDiagnostics | undefined;
140
148
  //# sourceMappingURL=ElementPolicies.d.ts.map
@@ -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;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"}
1
+ {"version":3,"file":"ElementPolicies.d.ts","sourceRoot":"","sources":["../../../../src/handlers/mcp-aql/policies/ElementPolicies.ts"],"names":[],"mappings":"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"}
@@ -490,6 +490,7 @@ function checkToolPrefix(pattern, colonIndex, fieldName, warnings) {
490
490
  */
491
491
  const UNGATABLE_OPERATIONS = new Set([
492
492
  'verify_challenge',
493
+ 'release_deadlock',
493
494
  'approve_cli_permission',
494
495
  'permission_prompt',
495
496
  ]);
@@ -564,13 +565,14 @@ function stripUngatable(operations, listName, elementName, elementType) {
564
565
  });
565
566
  return stripped.length > 0 ? stripped : undefined;
566
567
  }
567
- export function sanitizeGatekeeperPolicy(rawPolicy, elementName, elementType) {
568
- if (!rawPolicy || typeof rawPolicy !== 'object') {
568
+ export function sanitizeGatekeeperPolicy(rawPolicy, elementName, elementType, diagnosticsTarget) {
569
+ if (rawPolicy === undefined || rawPolicy === null) {
569
570
  return undefined;
570
571
  }
571
572
  try {
572
573
  // Wrap in a metadata envelope so parseElementPolicy can extract it
573
574
  const validated = parseElementPolicy({ gatekeeper: rawPolicy });
575
+ clearGatekeeperDiagnostics(diagnosticsTarget ?? rawPolicy);
574
576
  if (validated) {
575
577
  // Issue #758: Strip gatekeeper infrastructure operations from element policies
576
578
  // to prevent cascading confirmation loops
@@ -587,7 +589,7 @@ export function sanitizeGatekeeperPolicy(rawPolicy, elementName, elementType) {
587
589
  return validated;
588
590
  }
589
591
  catch (error) {
590
- const message = error instanceof Error ? error.message : String(error);
592
+ const message = formatGatekeeperDiagnosticMessage(error instanceof Error ? error.message : String(error));
591
593
  SecurityMonitor.logSecurityEvent({
592
594
  type: 'YAML_PARSING_WARNING',
593
595
  severity: 'MEDIUM',
@@ -595,7 +597,65 @@ export function sanitizeGatekeeperPolicy(rawPolicy, elementName, elementType) {
595
597
  details: `Malformed gatekeeper policy in "${elementName}" stripped during load: ${message}`,
596
598
  });
597
599
  logger.warn(`Stripped malformed gatekeeper policy from ${elementType} "${elementName}": ${message}`);
600
+ attachGatekeeperDiagnostics(diagnosticsTarget ?? rawPolicy, message);
598
601
  return undefined;
599
602
  }
600
603
  }
601
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ElementPolicies.js","sourceRoot":"","sources":["../../../../src/handlers/mcp-aql/policies/ElementPolicies.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;GAYG;AAEH,OAAO,EACL,eAAe,EAIf,mBAAmB,GACpB,MAAM,uBAAuB,CAAC;AAC/B,OAAO,EAAE,yBAAyB,EAAE,sBAAsB,EAAE,kBAAkB,EAAE,MAAM,wBAAwB,CAAC;AAC/G,OAAO,EAAE,eAAe,EAAE,MAAM,sCAAsC,CAAC;AACvE,OAAO,EAAE,MAAM,EAAE,MAAM,0BAA0B,CAAC;AAClD,OAAO,EAAE,uBAAuB,EAAE,MAAM,kCAAkC,CAAC;AA4C3E;;;;;;;;GAQG;AACH,MAAM,UAAU,oBAAoB,CAClC,SAAiB,EACjB,cAA+B,EAC/B,iBAA0B;IAE1B,oDAAoD;IACpD,IAAI,cAAc,GAAG,yBAAyB,CAAC,SAAS,CAAC,CAAC;IAC1D,IAAI,aAAiC,CAAC;IACtC,IAAI,aAAmD,CAAC;IACxD,IAAI,YAAY,GAAG,KAAK,CAAC;IACzB,8FAA8F;IAC9F,gFAAgF;IAChF,IAAI,kBAAkB,GAAG,KAAK,CAAC;IAC/B,sFAAsF;IACtF,MAAM,mBAAmB,GAA0D,EAAE,CAAC;IAEtF,8CAA8C;IAC9C,KAAK,MAAM,OAAO,IAAI,cAAc,EAAE,CAAC;QACrC,MAAM,MAAM,GAAG,OAAO,CAAC,QAAQ,CAAC,UAAU,CAAC;QAC3C,IAAI,CAAC,MAAM,EAAE,CAAC;YACZ,SAAS;QACX,CAAC;QAED,8CAA8C;QAC9C,IAAI,MAAM,CAAC,IAAI,EAAE,QAAQ,CAAC,SAAS,CAAC,EAAE,CAAC;YACrC,OAAO;gBACL,eAAe,EAAE,eAAe,CAAC,IAAI;gBACrC,aAAa,EAAE,OAAO,CAAC,IAAI;gBAC3B,aAAa,EAAE,MAAM;gBACrB,mBAAmB,EAAE,mBAAmB,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,mBAAmB,CAAC,CAAC,CAAC,SAAS;aACtF,CAAC;QACJ,CAAC;QAED,8BAA8B;QAC9B,IAAI,iBAAiB,IAAI,MAAM,CAAC,iBAAiB,EAAE,CAAC;YAClD,MAAM,EAAE,YAAY,EAAE,YAAY,EAAE,GAAG,MAAM,CAAC,iBAAiB,CAAC;YAEhE,2DAA2D;YAC3D,IAAI,YAAY,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,iBAAiB,CAAC,EAAE,CAAC;gBAC9D,YAAY,GAAG,IAAI,CAAC;gBACpB,aAAa,GAAG,OAAO,CAAC,IAAI,CAAC;gBAC7B,aAAa,GAAG,mBAAmB,CAAC;gBACpC,SAAS,CAAC,oDAAoD;YAChE,CAAC;YAED,+DAA+D;YAC/D,IAAI,YAAY,EAAE,QAAQ,CAAC,iBAAiB,CAAC,EAAE,CAAC;gBAC9C,YAAY,GAAG,IAAI,CAAC;gBACpB,aAAa,GAAG,OAAO,CAAC,IAAI,CAAC;gBAC7B,aAAa,GAAG,mBAAmB,CAAC;gBACpC,SAAS;YACX,CAAC;QACH,CAAC;QAED,gDAAgD;QAChD,IAAI,MAAM,CAAC,OAAO,EAAE,QAAQ,CAAC,SAAS,CAAC,EAAE,CAAC;YACxC,kDAAkD;YAClD,IAAI,cAAc,KAAK,eAAe,CAAC,IAAI,EAAE,CAAC;gBAC5C,cAAc,GAAG,eAAe,CAAC,eAAe,CAAC;gBACjD,kBAAkB,GAAG,IAAI,CAAC;gBAC1B,aAAa,GAAG,OAAO,CAAC,IAAI,CAAC;gBAC7B,aAAa,GAAG,SAAS,CAAC;YAC5B,CAAC;QACH,CAAC;QAED,sCAAsC;QACtC,8FAA8F;QAC9F,qFAAqF;QACrF,IAAI,MAAM,CAAC,KAAK,EAAE,QAAQ,CAAC,SAAS,CAAC,EAAE,CAAC;YACtC,iDAAiD;YACjD,IAAI,sBAAsB,CAAC,SAAS,CAAC,EAAE,CAAC;gBACtC,IAAI,CAAC,kBAAkB,EAAE,CAAC;oBACxB,4EAA4E;oBAC5E,cAAc,GAAG,eAAe,CAAC,YAAY,CAAC;oBAC9C,aAAa,GAAG,OAAO,CAAC,IAAI,CAAC;oBAC7B,aAAa,GAAG,OAAO,CAAC;gBAC1B,CAAC;qBAAM,CAAC;oBACN,kEAAkE;oBAClE,mBAAmB,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,OAAO,CAAC,IAAI,EAAE,WAAW,EAAE,eAAe,CAAC,YAAY,EAAE,CAAC,CAAC;gBAC9F,CAAC;YACH,CAAC;QACH,CAAC;IACH,CAAC;IAED,yDAAyD;IACzD,IAAI,YAAY,EAAE,CAAC;QACjB,OAAO;YACL,eAAe,EAAE,eAAe,CAAC,IAAI;YACrC,aAAa;YACb,aAAa,EAAE,mBAAmB;YAClC,YAAY,EAAE,IAAI;SACnB,CAAC;IACJ,CAAC;IAED,OAAO;QACL,eAAe,EAAE,cAAc;QAC/B,aAAa;QACb,aAAa;QACb,mBAAmB,EAAE,mBAAmB,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,mBAAmB,CAAC,CAAC,CAAC,SAAS;KACtF,CAAC;AACJ,CAAC;AAED;;;;;;;GAOG;AACH,MAAM,UAAU,wBAAwB,CACtC,SAAiB,EACjB,MAA2B,EAC3B,iBAA0B;IAE1B,MAAM,EAAE,eAAe,EAAE,aAAa,EAAE,YAAY,EAAE,mBAAmB,EAAE,GAAG,MAAM,CAAC;IAErF,cAAc;IACd,IAAI,eAAe,KAAK,eAAe,CAAC,IAAI,EAAE,CAAC;QAC7C,IAAI,YAAY,EAAE,CAAC;YACjB,OAAO;gBACL,OAAO,EAAE,KAAK;gBACd,eAAe;gBACf,SAAS,EAAE,mBAAmB,CAAC,iBAAiB;gBAChD,MAAM,EAAE,cAAc,SAAS,qCAAqC,iBAAiB,kDAAkD,aAAa,GAAG;gBACvJ,UAAU,EAAE,2BAA2B,aAAa,mCAAmC;gBACvF,YAAY,EAAE,gBAAgB;aAC/B,CAAC;QACJ,CAAC;QAED,OAAO;YACL,OAAO,EAAE,KAAK;YACd,eAAe;YACf,SAAS,EAAE,mBAAmB,CAAC,wBAAwB;YACvD,MAAM,EAAE,cAAc,SAAS,mCAAmC,aAAa,iBAAiB;YAChG,UAAU,EAAE,2BAA2B,aAAa,kCAAkC;YACtF,YAAY,EAAE,gBAAgB;SAC/B,CAAC;IACJ,CAAC;IAED,+BAA+B;IAC/B,IACE,eAAe,KAAK,eAAe,CAAC,eAAe;QACnD,eAAe,KAAK,eAAe,CAAC,kBAAkB,EACtD,CAAC;QACD,uEAAuE;QACvE,MAAM,MAAM,GAAG,kBAAkB,CAAC,SAAS,CAAC,CAAC;QAC7C,MAAM,UAAU,GAAG,eAAe,KAAK,eAAe,CAAC,kBAAkB;YACvE,CAAC,CAAC,iCAAiC;YACnC,CAAC,CAAC,wCAAwC,CAAC;QAE7C,IAAI,SAAiB,CAAC;QACtB,IAAI,aAAa,IAAI,MAAM,CAAC,aAAa,KAAK,SAAS,EAAE,CAAC;YACxD,oEAAoE;YACpE,MAAM,YAAY,GAAG,MAAM,EAAE,SAAS,CAAC,CAAC,CAAC,IAAI,MAAM,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;YACrE,SAAS,GAAG,mBAAmB,aAAa,qDAAqD,YAAY,EAAE,CAAC;QAClH,CAAC;aAAM,IAAI,MAAM,EAAE,SAAS,EAAE,CAAC;YAC7B,gDAAgD;YAChD,SAAS,GAAG,MAAM,CAAC,SAAS,CAAC;QAC/B,CAAC;aAAM,CAAC;YACN,gDAAgD;YAChD,SAAS,GAAG,4EAA4E,CAAC;QAC3F,CAAC;QAED,+EAA+E;QAC/E,MAAM,YAAY,GAAG,mBAAmB,IAAI,mBAAmB,CAAC,MAAM,GAAG,CAAC;YACxE,CAAC,CAAC,WAAW,mBAAmB,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,IAAI,GAAG,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,gCAAgC,mBAAmB,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,qCAAqC;YACvL,CAAC,CAAC,EAAE,CAAC;QAEP,OAAO;YACL,OAAO,EAAE,KAAK;YACd,eAAe;YACf,SAAS,EAAE,mBAAmB,CAAC,qBAAqB;YACpD,MAAM,EAAE,cAAc,SAAS,KAAK,UAAU,KAAK,SAAS,GAAG,YAAY,EAAE;YAC7E,UAAU,EAAE,uDAAuD;YACnE,mBAAmB,EAAE,IAAI;YACzB,YAAY,EAAE,aAAa,CAAC,CAAC,CAAC,gBAAgB,CAAC,CAAC,CAAC,mBAAmB;SACrE,CAAC;IACJ,CAAC;IAED,sBAAsB;IACtB,OAAO;QACL,OAAO,EAAE,IAAI;QACb,eAAe;QACf,MAAM,EAAE,aAAa;YACnB,CAAC,CAAC,cAAc,SAAS,+BAA+B,aAAa,GAAG;YACxE,CAAC,CAAC,cAAc,SAAS,mCAAmC;QAC9D,YAAY,EAAE,aAAa,CAAC,CAAC,CAAC,gBAAgB,CAAC,CAAC,CAAC,mBAAmB;KACrE,CAAC;AACJ,CAAC;AAED;;;;;;GAMG;AACH,MAAM,UAAU,kBAAkB,CAChC,QAAiB;IAEjB,IAAI,CAAC,QAAQ,IAAI,OAAO,QAAQ,KAAK,QAAQ,EAAE,CAAC;QAC9C,OAAO,SAAS,CAAC;IACnB,CAAC;IAED,MAAM,IAAI,GAAG,QAAmC,CAAC;IACjD,MAAM,UAAU,GAAG,IAAI,CAAC,UAAU,CAAC;IAEnC,IAAI,CAAC,UAAU,EAAE,CAAC;QAChB,OAAO,SAAS,CAAC;IACnB,CAAC;IAED,IAAI,OAAO,UAAU,KAAK,QAAQ,IAAI,UAAU,KAAK,IAAI,EAAE,CAAC;QAC1D,MAAM,IAAI,KAAK,CAAC,8CAA8C,CAAC,CAAC;IAClE,CAAC;IAED,MAAM,MAAM,GAAG,UAAqC,CAAC;IAErD,kCAAkC;IAClC,MAAM,KAAK,GAAG,mBAAmB,CAAC,MAAM,CAAC,KAAK,EAAE,OAAO,CAAC,CAAC;IACzD,MAAM,OAAO,GAAG,mBAAmB,CAAC,MAAM,CAAC,OAAO,EAAE,SAAS,CAAC,CAAC;IAC/D,MAAM,IAAI,GAAG,mBAAmB,CAAC,MAAM,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC;IAEtD,8BAA8B;IAC9B,IAAI,iBAA+D,CAAC;IACpE,IAAI,MAAM,CAAC,iBAAiB,EAAE,CAAC;QAC7B,IAAI,OAAO,MAAM,CAAC,iBAAiB,KAAK,QAAQ,IAAI,MAAM,CAAC,iBAAiB,KAAK,IAAI,EAAE,CAAC;YACtF,MAAM,IAAI,KAAK,CAAC,gEAAgE,CAAC,CAAC;QACpF,CAAC;QACD,MAAM,EAAE,GAAG,MAAM,CAAC,iBAA4C,CAAC;QAC/D,iBAAiB,GAAG;YAClB,YAAY,EAAE,mBAAmB,CAAC,EAAE,CAAC,YAAY,EAAE,gCAAgC,CAAC;YACpF,YAAY,EAAE,mBAAmB,CAAC,EAAE,CAAC,YAAY,EAAE,gCAAgC,CAAC;SACrF,CAAC;IACJ,CAAC;IAED,sDAAsD;IACtD,IAAI,oBAAqE,CAAC;IAC1E,IAAI,MAAM,CAAC,oBAAoB,EAAE,CAAC;QAChC,IAAI,OAAO,MAAM,CAAC,oBAAoB,KAAK,QAAQ,IAAI,MAAM,CAAC,oBAAoB,KAAK,IAAI,EAAE,CAAC;YAC5F,MAAM,IAAI,KAAK,CAAC,mEAAmE,CAAC,CAAC;QACvF,CAAC;QACD,MAAM,EAAE,GAAG,MAAM,CAAC,oBAA+C,CAAC;QAClE,IAAI,CAAC,EAAE,CAAC,WAAW,IAAI,OAAO,EAAE,CAAC,WAAW,KAAK,QAAQ,EAAE,CAAC;YAC1D,MAAM,IAAI,KAAK,CAAC,wGAAwG,CAAC,CAAC;QAC5H,CAAC;QACD,MAAM,YAAY,GAAG,mBAAmB,CAAC,EAAE,CAAC,YAAY,EAAE,mCAAmC,CAAC,CAAC;QAC/F,MAAM,eAAe,GAAG,mBAAmB,CAAC,EAAE,CAAC,eAAe,EAAE,sCAAsC,CAAC,CAAC;QACxG,MAAM,aAAa,GAAG,mBAAmB,CAAC,EAAE,CAAC,aAAa,EAAE,oCAAoC,CAAC,CAAC;QAClG,IAAI,YAAY;YAAE,sBAAsB,CAAC,YAAY,EAAE,mCAAmC,CAAC,CAAC;QAC5F,IAAI,eAAe;YAAE,sBAAsB,CAAC,eAAe,EAAE,sCAAsC,CAAC,CAAC;QACrG,IAAI,aAAa;YAAE,sBAAsB,CAAC,aAAa,EAAE,oCAAoC,CAAC,CAAC;QAC/F,+CAA+C;QAC/C,IAAI,cAAc,GAAkC,SAAS,CAAC;QAC9D,IAAI,EAAE,CAAC,cAAc,KAAK,SAAS,EAAE,CAAC;YACpC,IAAI,OAAO,EAAE,CAAC,cAAc,KAAK,QAAQ,IAAI,EAAE,CAAC,cAAc,KAAK,IAAI,EAAE,CAAC;gBACxE,MAAM,IAAI,KAAK,CAAC,kFAAkF,CAAC,CAAC;YACtG,CAAC;YACD,MAAM,EAAE,GAAG,EAAE,CAAC,cAAyC,CAAC;YAExD,2BAA2B;YAC3B,IAAI,eAAyD,CAAC;YAC9D,IAAI,EAAE,CAAC,eAAe,KAAK,SAAS,EAAE,CAAC;gBACrC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,EAAE,CAAC,eAAe,CAAC,EAAE,CAAC;oBACvC,MAAM,IAAI,KAAK,CAAC,iGAAiG,CAAC,CAAC;gBACrH,CAAC;gBACD,MAAM,WAAW,GAAG,IAAI,GAAG,CAAC,CAAC,UAAU,EAAE,WAAW,CAAC,CAAC,CAAC;gBACvD,KAAK,MAAM,KAAK,IAAI,EAAE,CAAC,eAAe,EAAE,CAAC;oBACvC,IAAI,OAAO,KAAK,KAAK,QAAQ,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC;wBACzD,MAAM,IAAI,KAAK,CAAC,0GAA0G,KAAK,uCAAuC,CAAC,CAAC;oBAC1K,CAAC;gBACH,CAAC;gBACD,eAAe,GAAG,EAAE,CAAC,eAA+C,CAAC;YACvE,CAAC;YAED,wBAAwB;YACxB,IAAI,YAAmD,CAAC;YACxD,IAAI,EAAE,CAAC,YAAY,KAAK,SAAS,EAAE,CAAC;gBAClC,IAAI,EAAE,CAAC,YAAY,KAAK,QAAQ,IAAI,EAAE,CAAC,YAAY,KAAK,cAAc,EAAE,CAAC;oBACvE,MAAM,IAAI,KAAK,CAAC,wHAAwH,EAAE,CAAC,YAAY,GAAG,CAAC,CAAC;gBAC9J,CAAC;gBACD,YAAY,GAAG,EAAE,CAAC,YAAY,CAAC;YACjC,CAAC;YAED,sBAAsB;YACtB,IAAI,UAA8B,CAAC;YACnC,IAAI,EAAE,CAAC,UAAU,KAAK,SAAS,EAAE,CAAC;gBAChC,IAAI,OAAO,EAAE,CAAC,UAAU,KAAK,QAAQ,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,EAAE,CAAC,UAAU,CAAC,EAAE,CAAC;oBACzE,MAAM,IAAI,KAAK,CAAC,4FAA4F,CAAC,CAAC;gBAChH,CAAC;gBACD,IAAI,EAAE,CAAC,UAAU,GAAG,EAAE,IAAI,EAAE,CAAC,UAAU,GAAG,IAAI,EAAE,CAAC;oBAC/C,MAAM,IAAI,KAAK,CAAC,8GAA8G,EAAE,CAAC,UAAU,EAAE,CAAC,CAAC;gBACjJ,CAAC;gBACD,UAAU,GAAG,EAAE,CAAC,UAAU,CAAC;YAC7B,CAAC;YAED,cAAc,GAAG,EAAE,eAAe,EAAE,YAAY,EAAE,UAAU,EAAE,CAAC;QACjE,CAAC;QAED,oBAAoB,GAAG;YACrB,WAAW,EAAE,EAAE,CAAC,WAAqB;YACrC,YAAY;YACZ,eAAe;YACf,aAAa;YACb,cAAc;SACf,CAAC;IACJ,CAAC;IAED,OAAO;QACL,KAAK;QACL,OAAO;QACP,IAAI;QACJ,iBAAiB;QACjB,oBAAoB;KACrB,CAAC;AACJ,CAAC;AAED;;;;;;GAMG;AACH,MAAM,UAAU,4BAA4B,CAC1C,MAA2C;IAE3C,IAAI,CAAC,MAAM,IAAI,OAAO,MAAM,KAAK,QAAQ,EAAE,CAAC;QAC1C,OAAO,EAAE,CAAC;IACZ,CAAC;IAED,MAAM,MAAM,GAAa,EAAE,CAAC;IAE5B,IAAI,MAAM,CAAC,MAAM,CAAC,MAAM,EAAE,sBAAsB,CAAC,EAAE,CAAC;QAClD,MAAM,CAAC,IAAI,CACT,sGAAsG,CACvG,CAAC;IACJ,CAAC;IAED,IAAI,MAAM,CAAC,UAAU,KAAK,SAAS,EAAE,CAAC;QACpC,IAAI,CAAC;YACH,kBAAkB,CAAC,EAAE,UAAU,EAAE,MAAM,CAAC,UAAU,EAAE,CAAC,CAAC;QACxD,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,MAAM,CAAC,IAAI,CAAC,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC;QACtE,CAAC;IACH,CAAC;IAED,OAAO,MAAM,CAAC;AAChB,CAAC;AAED;;;;;;GAMG;AACH,SAAS,mBAAmB,CAC1B,KAAc,EACd,SAAiB;IAEjB,IAAI,KAAK,KAAK,SAAS,EAAE,CAAC;QACxB,OAAO,SAAS,CAAC;IACnB,CAAC;IAED,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC;QAC1B,MAAM,IAAI,KAAK,CAAC,8BAA8B,SAAS,mBAAmB,CAAC,CAAC;IAC9E,CAAC;IAED,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,IAAI,EAAkB,EAAE,CAAC,OAAO,IAAI,KAAK,QAAQ,CAAC,EAAE,CAAC;QACrE,MAAM,IAAI,KAAK,CAAC,8BAA8B,SAAS,4BAA4B,CAAC,CAAC;IACvF,CAAC;IAED,OAAO,KAAK,CAAC;AACf,CAAC;AAED;;;;;;GAMG;AACH,SAAS,sBAAsB,CAC7B,QAAkB,EAClB,SAAiB;IAEjB,KAAK,MAAM,OAAO,IAAI,QAAQ,EAAE,CAAC;QAC/B,IAAI,CAAC,OAAO,EAAE,CAAC;YACb,MAAM,IAAI,KAAK,CAAC,8BAA8B,SAAS,mCAAmC,CAAC,CAAC;QAC9F,CAAC;QACD,IAAI,OAAO,CAAC,MAAM,GAAG,uBAAuB,EAAE,CAAC;YAC7C,MAAM,IAAI,KAAK,CAAC,8BAA8B,SAAS,sCAAsC,uBAAuB,aAAa,CAAC,CAAC;QACrI,CAAC;IACH,CAAC;AACH,CAAC;AAED;;;GAGG;AACH,MAAM,mBAAmB,GAAG,IAAI,GAAG,CAAC;IAClC,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM;IAC/C,UAAU,EAAE,WAAW,EAAE,cAAc;CACxC,CAAC,CAAC;AAEH;;;GAGG;AACH,MAAM,oBAAoB,GAAG,iBAAiB,CAAC;AAE/C;;;;;;;;;;;;;;;GAeG;AACH,MAAM,UAAU,oBAAoB,CAClC,QAAkB,EAClB,SAAiB;IAEjB,MAAM,QAAQ,GAAa,EAAE,CAAC;IAC9B,KAAK,MAAM,OAAO,IAAI,QAAQ,EAAE,CAAC;QAC/B,iBAAiB,CAAC,OAAO,EAAE,SAAS,EAAE,QAAQ,CAAC,CAAC;IAClD,CAAC;IACD,OAAO,QAAQ,CAAC;AAClB,CAAC;AAED;;;GAGG;AACH,SAAS,iBAAiB,CACxB,OAAe,EACf,SAAiB,EACjB,QAAkB;IAElB,oCAAoC;IACpC,IAAI,OAAO,KAAK,OAAO,CAAC,IAAI,EAAE,EAAE,CAAC;QAC/B,QAAQ,CAAC,IAAI,CACX,GAAG,SAAS,aAAa,OAAO,uEAAuE,CACxG,CAAC;IACJ,CAAC;IAED,+CAA+C;IAC/C,IAAI,OAAO,KAAK,GAAG,EAAE,CAAC;QACpB,QAAQ,CAAC,IAAI,CACX,GAAG,SAAS,gEAAgE,CAC7E,CAAC;QACF,OAAO;IACT,CAAC;IAED,2CAA2C;IAC3C,MAAM,UAAU,GAAG,OAAO,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;IACxC,IAAI,UAAU,KAAK,CAAC,CAAC,EAAE,CAAC;QACtB,QAAQ,CAAC,IAAI,CACX,GAAG,SAAS,aAAa,OAAO,iDAAiD;YACjF,4FAA4F,CAC7F,CAAC;IACJ,CAAC;SAAM,CAAC;QACN,eAAe,CAAC,OAAO,EAAE,UAAU,EAAE,SAAS,EAAE,QAAQ,CAAC,CAAC;IAC5D,CAAC;IAED,iDAAiD;IACjD,MAAM,UAAU,GAAG,IAAI,GAAG,EAAU,CAAC;IACrC,KAAK,MAAM,EAAE,IAAI,OAAO,EAAE,CAAC;QACzB,IAAI,oBAAoB,CAAC,IAAI,CAAC,EAAE,CAAC;YAAE,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;IACxD,CAAC;IACD,IAAI,UAAU,CAAC,IAAI,GAAG,CAAC,EAAE,CAAC;QACxB,QAAQ,CAAC,IAAI,CACX,GAAG,SAAS,aAAa,OAAO,4BAA4B,CAAC,GAAG,UAAU,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM;YAC5F,6CAA6C,CAC9C,CAAC;IACJ,CAAC;AACH,CAAC;AAED;;GAEG;AACH,SAAS,eAAe,CACtB,OAAe,EACf,UAAkB,EAClB,SAAiB,EACjB,QAAkB;IAElB,MAAM,MAAM,GAAG,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,UAAU,CAAC,CAAC;IAC5C,MAAM,UAAU,GAAG,OAAO,CAAC,KAAK,CAAC,UAAU,GAAG,CAAC,CAAC,CAAC;IAEjD,gCAAgC;IAChC,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,MAAM,CAAC,EAAE,CAAC;QACnE,QAAQ,CAAC,IAAI,CACX,GAAG,SAAS,aAAa,OAAO,+BAA+B,MAAM,KAAK;YAC1E,mBAAmB,CAAC,GAAG,mBAAmB,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,gCAAgC,CACvF,CAAC;IACJ,CAAC;IAED,4CAA4C;IAC5C,IAAI,UAAU,KAAK,GAAG,EAAE,CAAC;QACvB,QAAQ,CAAC,IAAI,CACX,GAAG,SAAS,aAAa,OAAO,iBAAiB,MAAM,0CAA0C,CAClG,CAAC;IACJ,CAAC;AACH,CAAC;AAED;;;;;;;;;;;;;;;;;;GAkBG;AACH;;;;;;;;;;;;;;;;;GAiBG;AACH,MAAM,oBAAoB,GAAG,IAAI,GAAG,CAAC;IACnC,kBAAkB;IAClB,wBAAwB;IACxB,mBAAmB;CACpB,CAAC,CAAC;AAEH,sFAAsF;AACtF,MAAM,2BAA2B,GAAG,IAAI,GAAG,CAAC;IAC1C,GAAG,oBAAoB;IACvB,mBAAmB;CACpB,CAAC,CAAC;AAEH;;;;GAIG;AACH,MAAM,UAAU,0BAA0B,CAAC,SAAiB;IAC1D,OAAO,2BAA2B,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;AACpD,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,yBAAyB,CACvC,cAAwF;IAExF,KAAK,MAAM,OAAO,IAAI,cAAc,EAAE,CAAC;QACrC,MAAM,UAAU,GAAG,OAAO,CAAC,QAAQ,EAAE,UAAiD,CAAC;QACvF,MAAM,QAAQ,GAAG,UAAU,EAAE,IAAI,CAAC;QAClC,IAAI,KAAK,CAAC,OAAO,CAAC,QAAQ,CAAC,IAAI,QAAQ,CAAC,QAAQ,CAAC,mBAAmB,CAAC,EAAE,CAAC;YACtE,OAAO,EAAE,IAAI,EAAE,OAAO,CAAC,IAAI,EAAE,IAAI,EAAE,OAAO,CAAC,IAAI,EAAE,CAAC;QACpD,CAAC;IACH,CAAC;IACD,OAAO,SAAS,CAAC;AACnB,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,2BAA2B,CACzC,cAAwF;IAExF,MAAM,UAAU,GAA0C,EAAE,CAAC;IAC7D,KAAK,MAAM,OAAO,IAAI,cAAc,EAAE,CAAC;QACrC,MAAM,UAAU,GAAG,OAAO,CAAC,QAAQ,EAAE,UAAiD,CAAC;QACvF,MAAM,WAAW,GAAG,UAAU,EAAE,OAAO,CAAC;QACxC,IAAI,KAAK,CAAC,OAAO,CAAC,WAAW,CAAC,IAAI,WAAW,CAAC,QAAQ,CAAC,mBAAmB,CAAC,EAAE,CAAC;YAC5E,UAAU,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,OAAO,CAAC,IAAI,EAAE,IAAI,EAAE,OAAO,CAAC,IAAI,EAAE,CAAC,CAAC;QAC9D,CAAC;IACH,CAAC;IACD,OAAO,UAAU,CAAC;AACpB,CAAC;AAED;;;;;;;GAOG;AACH,SAAS,cAAc,CACrB,UAAgC,EAChC,QAAgB,EAChB,WAAmB,EACnB,WAAmB;IAEnB,IAAI,CAAC,UAAU,EAAE,MAAM;QAAE,OAAO,UAAU,CAAC;IAE3C,+DAA+D;IAC/D,0FAA0F;IAC1F,uFAAuF;IACvF,6FAA6F;IAC7F,uFAAuF;IACvF,MAAM,UAAU,GAAG,QAAQ,KAAK,OAAO;QACrC,CAAC,CAAC,IAAI,GAAG,CAAC,CAAC,GAAG,oBAAoB,EAAE,mBAAmB,CAAC,CAAC;QACzD,CAAC,CAAC,oBAAoB,CAAC;IAEzB,MAAM,QAAQ,GAAG,UAAU,CAAC,MAAM,CAAC,EAAE,CAAC,EAAE;QACtC,IAAI,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,CAAC;YACvB,MAAM,CAAC,IAAI,CACT,0BAA0B,EAAE,UAAU,QAAQ,YAAY,WAAW,KAAK,WAAW,MAAM;gBAC3F,uFAAuF,CACxF,CAAC;YACF,OAAO,KAAK,CAAC;QACf,CAAC;QACD,OAAO,IAAI,CAAC;IACd,CAAC,CAAC,CAAC;IACH,OAAO,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,SAAS,CAAC;AACpD,CAAC;AAED,MAAM,UAAU,wBAAwB,CACtC,SAAkB,EAClB,WAAmB,EACnB,WAAmB;IAEnB,IAAI,CAAC,SAAS,IAAI,OAAO,SAAS,KAAK,QAAQ,EAAE,CAAC;QAChD,OAAO,SAAS,CAAC;IACnB,CAAC;IAED,IAAI,CAAC;QACH,mEAAmE;QACnE,MAAM,SAAS,GAAG,kBAAkB,CAAC,EAAE,UAAU,EAAE,SAAS,EAAE,CAAC,CAAC;QAChE,IAAI,SAAS,EAAE,CAAC;YACd,+EAA+E;YAC/E,0CAA0C;YAC1C,SAAS,CAAC,KAAK,GAAG,cAAc,CAAC,SAAS,CAAC,KAAK,EAAE,OAAO,EAAE,WAAW,EAAE,WAAW,CAAC,CAAC;YACrF,SAAS,CAAC,OAAO,GAAG,cAAc,CAAC,SAAS,CAAC,OAAO,EAAE,SAAS,EAAE,WAAW,EAAE,WAAW,CAAC,CAAC;YAC3F,SAAS,CAAC,IAAI,GAAG,cAAc,CAAC,SAAS,CAAC,IAAI,EAAE,MAAM,EAAE,WAAW,EAAE,WAAW,CAAC,CAAC;YAElF,MAAM,CAAC,KAAK,CAAC,gCAAgC,WAAW,KAAK,WAAW,GAAG,EAAE;gBAC3E,KAAK,EAAE,SAAS,CAAC,KAAK,EAAE,MAAM,IAAI,CAAC;gBACnC,OAAO,EAAE,SAAS,CAAC,OAAO,EAAE,MAAM,IAAI,CAAC;gBACvC,IAAI,EAAE,SAAS,CAAC,IAAI,EAAE,MAAM,IAAI,CAAC;gBACjC,oBAAoB,EAAE,CAAC,CAAC,SAAS,CAAC,iBAAiB;aACpD,CAAC,CAAC;QACL,CAAC;QACD,OAAO,SAAS,CAAC;IACnB,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,MAAM,OAAO,GAAG,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;QACvE,eAAe,CAAC,gBAAgB,CAAC;YAC/B,IAAI,EAAE,sBAAsB;YAC5B,QAAQ,EAAE,QAAQ;YAClB,MAAM,EAAE,GAAG,WAAW,cAAc;YACpC,OAAO,EAAE,mCAAmC,WAAW,2BAA2B,OAAO,EAAE;SAC5F,CAAC,CAAC;QACH,MAAM,CAAC,IAAI,CAAC,6CAA6C,WAAW,KAAK,WAAW,MAAM,OAAO,EAAE,CAAC,CAAC;QACrG,OAAO,SAAS,CAAC;IACnB,CAAC;AACH,CAAC","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"]}
604
+ function formatGatekeeperDiagnosticMessage(message) {
605
+ const normalized = message.trim();
606
+ const guidance = getGatekeeperFixGuidance(normalized);
607
+ return guidance ? `${normalized} Fix: ${guidance}` : normalized;
608
+ }
609
+ function getGatekeeperFixGuidance(message) {
610
+ if (message.includes('externalRestrictions must be nested under gatekeeper.externalRestrictions')) {
611
+ return 'Move externalRestrictions under gatekeeper.externalRestrictions and keep allowPatterns, confirmPatterns, and denyPatterns inside that nested object.';
612
+ }
613
+ if (message.includes('externalRestrictions.description is required')) {
614
+ return 'Add gatekeeper.externalRestrictions.description with a short explanation, for example description: "Read-only shell policy".';
615
+ }
616
+ if (message.includes('must be an array') || message.includes('must contain only strings')) {
617
+ return 'Use YAML arrays of strings, for example denyPatterns: ["Bash:rm *"] or allow: ["read_*"].';
618
+ }
619
+ if (message.includes('scopeRestrictions must be an object')) {
620
+ return 'Use scopeRestrictions as an object, for example scopeRestrictions: { allowedTypes: ["persona", "skill"] }.';
621
+ }
622
+ if (message.includes('must be an object')) {
623
+ return 'Use gatekeeper as an object, for example gatekeeper: { deny: ["delete_element"] }.';
624
+ }
625
+ return 'Compare the element against the gatekeeper examples from introspection or the security docs, then reactivate it after fixing the structure.';
626
+ }
627
+ export function attachGatekeeperDiagnostics(target, message) {
628
+ if (!target || typeof target !== 'object') {
629
+ return;
630
+ }
631
+ target.gatekeeperDiagnostics = {
632
+ valid: false,
633
+ enforceable: false,
634
+ message,
635
+ };
636
+ }
637
+ export function clearGatekeeperDiagnostics(target) {
638
+ if (!target || typeof target !== 'object') {
639
+ return;
640
+ }
641
+ delete target.gatekeeperDiagnostics;
642
+ }
643
+ export function getGatekeeperDiagnostics(target) {
644
+ if (!target || typeof target !== 'object') {
645
+ return undefined;
646
+ }
647
+ const diagnostics = target.gatekeeperDiagnostics;
648
+ if (!diagnostics || typeof diagnostics !== 'object') {
649
+ return undefined;
650
+ }
651
+ const record = diagnostics;
652
+ if (record.valid === false && record.enforceable === false && typeof record.message === 'string' && record.message.trim() !== '') {
653
+ return {
654
+ valid: false,
655
+ enforceable: false,
656
+ message: record.message,
657
+ };
658
+ }
659
+ return undefined;
660
+ }
661
+ //# 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\nexport interface GatekeeperPolicyDiagnostics {\n  valid: false;\n  enforceable: false;\n  message: string;\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  'release_deadlock',\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  diagnosticsTarget?: Record<string, unknown>,\n): ElementGatekeeperPolicy | undefined {\n  if (rawPolicy === undefined || rawPolicy === null) {\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    clearGatekeeperDiagnostics(diagnosticsTarget ?? 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 = formatGatekeeperDiagnosticMessage(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    attachGatekeeperDiagnostics(diagnosticsTarget ?? rawPolicy, message);\n    return undefined;\n  }\n}\n\nfunction formatGatekeeperDiagnosticMessage(message: string): string {\n  const normalized = message.trim();\n  const guidance = getGatekeeperFixGuidance(normalized);\n  return guidance ? `${normalized} Fix: ${guidance}` : normalized;\n}\n\nfunction getGatekeeperFixGuidance(message: string): string | undefined {\n  if (message.includes('externalRestrictions must be nested under gatekeeper.externalRestrictions')) {\n    return 'Move externalRestrictions under gatekeeper.externalRestrictions and keep allowPatterns, confirmPatterns, and denyPatterns inside that nested object.';\n  }\n\n  if (message.includes('externalRestrictions.description is required')) {\n    return 'Add gatekeeper.externalRestrictions.description with a short explanation, for example description: \"Read-only shell policy\".';\n  }\n\n  if (message.includes('must be an array') || message.includes('must contain only strings')) {\n    return 'Use YAML arrays of strings, for example denyPatterns: [\"Bash:rm *\"] or allow: [\"read_*\"].';\n  }\n\n  if (message.includes('scopeRestrictions must be an object')) {\n    return 'Use scopeRestrictions as an object, for example scopeRestrictions: { allowedTypes: [\"persona\", \"skill\"] }.';\n  }\n\n  if (message.includes('must be an object')) {\n    return 'Use gatekeeper as an object, for example gatekeeper: { deny: [\"delete_element\"] }.';\n  }\n\n  return 'Compare the element against the gatekeeper examples from introspection or the security docs, then reactivate it after fixing the structure.';\n}\n\nexport function attachGatekeeperDiagnostics(target: unknown, message: string): void {\n  if (!target || typeof target !== 'object') {\n    return;\n  }\n\n  (target as Record<string, unknown>).gatekeeperDiagnostics = {\n    valid: false,\n    enforceable: false,\n    message,\n  } satisfies GatekeeperPolicyDiagnostics;\n}\n\nexport function clearGatekeeperDiagnostics(target: unknown): void {\n  if (!target || typeof target !== 'object') {\n    return;\n  }\n\n  delete (target as Record<string, unknown>).gatekeeperDiagnostics;\n}\n\nexport function getGatekeeperDiagnostics(target: unknown): GatekeeperPolicyDiagnostics | undefined {\n  if (!target || typeof target !== 'object') {\n    return undefined;\n  }\n\n  const diagnostics = (target as Record<string, unknown>).gatekeeperDiagnostics;\n  if (!diagnostics || typeof diagnostics !== 'object') {\n    return undefined;\n  }\n\n  const record = diagnostics as Record<string, unknown>;\n  if (record.valid === false && record.enforceable === false && typeof record.message === 'string' && record.message.trim() !== '') {\n    return {\n      valid: false,\n      enforceable: false,\n      message: record.message,\n    };\n  }\n\n  return undefined;\n}\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"OperationPolicies.d.ts","sourceRoot":"","sources":["../../../../src/handlers/mcp-aql/policies/OperationPolicies.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA2CG;AAEH,OAAO,EAAE,eAAe,EAAE,KAAK,eAAe,EAAE,MAAM,uBAAuB,CAAC;AAC9E,OAAO,EAA8B,KAAK,YAAY,EAAE,MAAM,uBAAuB,CAAC;AAgBtF;;;;;GAKG;AACH,wBAAgB,uBAAuB,CAAC,QAAQ,EAAE,YAAY,GAAG,eAAe,CAE/E;AAED;;;;;;;;;GASG;AACH,eAAO,MAAM,0BAA0B,EAAE,MAAM,CAAC,MAAM,EAAE,eAAe,CAyFtE,CAAC;AAEF;;;GAGG;AACH,eAAO,MAAM,kBAAkB,iCAA6B,CAAC;AAE7D;;;;;GAKG;AACH,wBAAgB,kBAAkB,CAAC,SAAS,EAAE,MAAM,GAAG,eAAe,GAAG,SAAS,CAEjF;AAED;;;;;;;;;;GAUG;AACH,wBAAgB,yBAAyB,CAAC,SAAS,EAAE,MAAM,GAAG,eAAe,CAe5E;AAED;;;;;;GAMG;AACH,wBAAgB,sBAAsB,CAAC,SAAS,EAAE,MAAM,GAAG,OAAO,CAIjE;AAED;;;;;;GAMG;AACH,wBAAgB,oBAAoB,CAAC,KAAK,EAAE,eAAe,GAAG,MAAM,EAAE,CAUrE;AAED;;;GAGG;AACH,wBAAgB,yBAAyB,IAAI,MAAM,EAAE,CAEpD;AAED;;;GAGG;AACH,wBAAgB,iCAAiC,IAAI,MAAM,EAAE,CAK5D"}
1
+ {"version":3,"file":"OperationPolicies.d.ts","sourceRoot":"","sources":["../../../../src/handlers/mcp-aql/policies/OperationPolicies.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA2CG;AAEH,OAAO,EAAE,eAAe,EAAE,KAAK,eAAe,EAAE,MAAM,uBAAuB,CAAC;AAC9E,OAAO,EAA8B,KAAK,YAAY,EAAE,MAAM,uBAAuB,CAAC;AAgBtF;;;;;GAKG;AACH,wBAAgB,uBAAuB,CAAC,QAAQ,EAAE,YAAY,GAAG,eAAe,CAE/E;AAED;;;;;;;;;GASG;AACH,eAAO,MAAM,0BAA0B,EAAE,MAAM,CAAC,MAAM,EAAE,eAAe,CA8FtE,CAAC;AAEF;;;GAGG;AACH,eAAO,MAAM,kBAAkB,iCAA6B,CAAC;AAE7D;;;;;GAKG;AACH,wBAAgB,kBAAkB,CAAC,SAAS,EAAE,MAAM,GAAG,eAAe,GAAG,SAAS,CAEjF;AAED;;;;;;;;;;GAUG;AACH,wBAAgB,yBAAyB,CAAC,SAAS,EAAE,MAAM,GAAG,eAAe,CAe5E;AAED;;;;;;GAMG;AACH,wBAAgB,sBAAsB,CAAC,SAAS,EAAE,MAAM,GAAG,OAAO,CAIjE;AAED;;;;;;GAMG;AACH,wBAAgB,oBAAoB,CAAC,KAAK,EAAE,eAAe,GAAG,MAAM,EAAE,CAUrE;AAED;;;GAGG;AACH,wBAAgB,yBAAyB,IAAI,MAAM,EAAE,CAEpD;AAED;;;GAGG;AACH,wBAAgB,iCAAiC,IAAI,MAAM,EAAE,CAK5D"}
@@ -83,6 +83,11 @@ export const OPERATION_POLICY_OVERRIDES = {
83
83
  defaultLevel: PermissionLevel.AUTO_APPROVE,
84
84
  rationale: 'Verification flow — must be auto-approved to avoid requiring confirmation to verify',
85
85
  },
86
+ release_deadlock: {
87
+ defaultLevel: PermissionLevel.AUTO_APPROVE,
88
+ rationale: 'Deadlock relief flow — requires out-of-band verification and must remain reachable even when confirmations are blocked',
89
+ canBeElevated: false,
90
+ },
86
91
  beetlejuice_beetlejuice_beetlejuice: {
87
92
  defaultLevel: PermissionLevel.AUTO_APPROVE,
88
93
  rationale: 'Test fixture for danger zone verification — must be auto-approved to avoid requiring confirmation to test',
@@ -244,4 +249,4 @@ export function getConfirmationRequiredOperations() {
244
249
  ...getOperationsAtLevel(PermissionLevel.CONFIRM_SINGLE_USE),
245
250
  ];
246
251
  }
247
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"OperationPolicies.js","sourceRoot":"","sources":["../../../../src/handlers/mcp-aql/policies/OperationPolicies.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Operation Policies\n *\n * Default permission levels are derived from the operation's endpoint routing:\n * - READ    → AUTO_APPROVE (read-only, no side effects)\n * - CREATE  → CONFIRM_SESSION (additive state changes, safe once approved)\n * - UPDATE  → CONFIRM_SINGLE_USE (modifying existing data, each instance reviewed)\n * - DELETE  → CONFIRM_SINGLE_USE (destructive, each instance reviewed)\n * - EXECUTE → CONFIRM_SINGLE_USE (unpredictable side effects)\n *\n * OPERATION_POLICY_OVERRIDES contains only operations that deviate from their\n * endpoint default — e.g., operations that need stricter policies, special\n * canBeElevated flags, or looser defaults than their endpoint implies.\n *\n * Resolution hierarchy:\n * 1. Element policy (Layer 2 — active element allow/confirm/deny lists)\n * 2. Explicit override in OPERATION_POLICY_OVERRIDES\n * 3. Endpoint default (derived from OperationRouter)\n * 4. Secure fallback: CONFIRM_SINGLE_USE for unknown operations\n *\n * Examples:\n *\n *   activate_element → READ endpoint → AUTO_APPROVE (no override needed)\n *     Routing to READ is sufficient. No entry in OPERATION_POLICY_OVERRIDES.\n *\n *   create_element → CREATE endpoint → CONFIRM_SESSION (no override needed)\n *     Endpoint default handles it. First create confirms, rest of session is smooth.\n *\n *   verify_challenge → CREATE endpoint → overridden to AUTO_APPROVE\n *     On CREATE (default CONFIRM_SESSION) but must be frictionless to avoid\n *     requiring confirmation to complete a verification flow.\n *\n *   delete_element → DELETE endpoint → CONFIRM_SINGLE_USE + canBeElevated: false\n *     Matches endpoint default level, but override locks canBeElevated so no\n *     element policy can silently auto-approve deletions.\n *\n *   record_execution_step → CREATE endpoint → CONFIRM_SESSION (no override needed)\n *     Moved from EXECUTE to CREATE so it inherits CONFIRM_SESSION, matching\n *     the polling loop's need for session-level frictionless approval.\n *\n *   A persona with gatekeeper: { confirm: ['create_element'] } → CONFIRM_SESSION\n *     Element policy (Layer 2) can tighten or loosen within what the operation allows.\n *     This is a runtime override via element activation, not a code change.\n */\n\nimport { PermissionLevel, type OperationPolicy } from '../GatekeeperTypes.js';\nimport { getRoute, OPERATION_ROUTES, type CRUDEndpoint } from '../OperationRouter.js';\n\n/**\n * Map CRUDE endpoints to their default permission levels.\n *\n * This is the single source of truth for what each endpoint implies.\n * If an operation is routed to READ, it gets AUTO_APPROVE unless overridden.\n */\nconst ENDPOINT_DEFAULT_LEVELS: Record<CRUDEndpoint, PermissionLevel> = {\n  READ: PermissionLevel.AUTO_APPROVE,\n  CREATE: PermissionLevel.CONFIRM_SESSION,\n  UPDATE: PermissionLevel.CONFIRM_SINGLE_USE,\n  DELETE: PermissionLevel.CONFIRM_SINGLE_USE,\n  EXECUTE: PermissionLevel.CONFIRM_SINGLE_USE,\n};\n\n/**\n * Get the default permission level for a CRUDE endpoint.\n *\n * @param endpoint - The CRUDE endpoint\n * @returns The default permission level for that endpoint\n */\nexport function getEndpointDefaultLevel(endpoint: CRUDEndpoint): PermissionLevel {\n  return ENDPOINT_DEFAULT_LEVELS[endpoint];\n}\n\n/**\n * Explicit policy overrides for operations that deviate from their endpoint default.\n *\n * Only add entries here when an operation needs:\n * - A different permission level than its endpoint implies\n * - A canBeElevated: false restriction\n * - A specific rationale that differs from the endpoint's general rationale\n *\n * Operations NOT listed here inherit their permission level from their endpoint.\n */\nexport const OPERATION_POLICY_OVERRIDES: Record<string, OperationPolicy> = {\n  // ===== CREATE endpoint overrides =====\n  // These are on CREATE (default CONFIRM_SESSION) but need AUTO_APPROVE\n  verify_challenge: {\n    defaultLevel: PermissionLevel.AUTO_APPROVE,\n    rationale: 'Verification flow — must be auto-approved to avoid requiring confirmation to verify',\n  },\n  beetlejuice_beetlejuice_beetlejuice: {\n    defaultLevel: PermissionLevel.AUTO_APPROVE,\n    rationale: 'Test fixture for danger zone verification — must be auto-approved to avoid requiring confirmation to test',\n  },\n\n  // ===== DELETE endpoint overrides =====\n  // These match the endpoint default (CONFIRM_SINGLE_USE) but need canBeElevated: false\n  delete_element: {\n    defaultLevel: PermissionLevel.CONFIRM_SINGLE_USE,\n    rationale: 'Destructive operation, permanently removes data',\n    canBeElevated: false,\n  },\n  clear: {\n    defaultLevel: PermissionLevel.CONFIRM_SINGLE_USE,\n    rationale: 'Destructive operation, clears all memory entries',\n    canBeElevated: false,\n  },\n  clear_github_auth: {\n    defaultLevel: PermissionLevel.CONFIRM_SINGLE_USE,\n    rationale: 'Destructive operation, removes authentication credentials',\n    canBeElevated: false,\n  },\n\n  // ===== EXECUTE endpoint overrides =====\n  // These are on EXECUTE (default CONFIRM_SINGLE_USE) but need different levels\n\n  // Must be auto-approved to avoid infinite confirmation loops\n  confirm_operation: {\n    defaultLevel: PermissionLevel.AUTO_APPROVE,\n    rationale: 'Gatekeeper confirmation flow — must be auto-approved to avoid requiring confirmation to confirm',\n  },\n  // Issue #625: CLI-level permission delegation\n  permission_prompt: {\n    defaultLevel: PermissionLevel.AUTO_APPROVE,\n    rationale: 'Permission evaluation flow — must be auto-approved to avoid infinite permission loops',\n    canBeElevated: false,\n  },\n  // Issue #625 Phase 3: CLI approval workflow\n  approve_cli_permission: {\n    defaultLevel: PermissionLevel.AUTO_APPROVE,\n    rationale: 'CLI approval flow — must be auto-approved to avoid requiring confirmation to approve',\n    canBeElevated: false,\n  },\n\n  // Note: get_execution_state and get_gathered_data moved to READ endpoint —\n  // they inherit AUTO_APPROVE from the endpoint default, no override needed.\n  // record_execution_step moved to CREATE endpoint —\n  // it inherits CONFIRM_SESSION from the endpoint default, no override needed.\n\n  // Session-confirmable lifecycle operations (less friction than per-use)\n  complete_execution: {\n    defaultLevel: PermissionLevel.CONFIRM_SESSION,\n    rationale: 'Signals execution completion, part of active execution',\n    canBeElevated: true,\n  },\n  continue_execution: {\n    defaultLevel: PermissionLevel.CONFIRM_SESSION,\n    rationale: 'Resumes execution from saved state',\n    canBeElevated: true,\n  },\n  prepare_handoff: {\n    defaultLevel: PermissionLevel.CONFIRM_SESSION,\n    rationale: 'Prepares session handoff snapshot, read-heavy with serialization',\n    canBeElevated: true,\n  },\n\n  // Non-elevatable execute operations (match endpoint default but lock elevation)\n  execute_agent: {\n    defaultLevel: PermissionLevel.CONFIRM_SINGLE_USE,\n    rationale: 'Executes agent with unpredictable side effects',\n    canBeElevated: false,\n  },\n  abort_execution: {\n    defaultLevel: PermissionLevel.CONFIRM_SINGLE_USE,\n    rationale: 'Terminates agent execution, rejecting further operations for the goalId',\n    canBeElevated: false,\n  },\n  resume_from_handoff: {\n    defaultLevel: PermissionLevel.CONFIRM_SINGLE_USE,\n    rationale: 'Resumes execution from external handoff data, requires validation',\n    canBeElevated: false,\n  },\n};\n\n/**\n * @deprecated Use OPERATION_POLICY_OVERRIDES instead.\n * Kept as an alias for backward compatibility with code that imports OPERATION_POLICIES.\n */\nexport const OPERATION_POLICIES = OPERATION_POLICY_OVERRIDES;\n\n/**\n * Get the explicit policy override for an operation, if one exists.\n *\n * @param operation - The operation name\n * @returns The override policy, or undefined if no override exists\n */\nexport function getOperationPolicy(operation: string): OperationPolicy | undefined {\n  return OPERATION_POLICY_OVERRIDES[operation];\n}\n\n/**\n * Get the default permission level for an operation.\n *\n * Resolution order:\n * 1. Explicit override in OPERATION_POLICY_OVERRIDES\n * 2. Endpoint default from OperationRouter\n * 3. CONFIRM_SINGLE_USE for unknown operations (secure fallback)\n *\n * @param operation - The operation name\n * @returns The effective default permission level\n */\nexport function getDefaultPermissionLevel(operation: string): PermissionLevel {\n  // 1. Check for explicit override\n  const override = OPERATION_POLICY_OVERRIDES[operation];\n  if (override) {\n    return override.defaultLevel;\n  }\n\n  // 2. Derive from endpoint routing\n  const route = getRoute(operation);\n  if (route) {\n    return ENDPOINT_DEFAULT_LEVELS[route.endpoint];\n  }\n\n  // 3. Secure fallback for unknown operations\n  return PermissionLevel.CONFIRM_SINGLE_USE;\n}\n\n/**\n * Check if an operation can have its permission level elevated.\n * Some destructive operations cannot be elevated to AUTO_APPROVE.\n *\n * @param operation - The operation name\n * @returns true if the operation can be elevated\n */\nexport function canOperationBeElevated(operation: string): boolean {\n  const override = OPERATION_POLICY_OVERRIDES[operation];\n  // Default to allowing elevation for operations without explicit overrides\n  return override?.canBeElevated ?? true;\n}\n\n/**\n * Get all operations at a specific effective permission level.\n * Considers both explicit overrides and endpoint-derived defaults.\n *\n * @param level - The permission level to filter by\n * @returns Array of operation names at that level\n */\nexport function getOperationsAtLevel(level: PermissionLevel): string[] {\n  const results: string[] = [];\n\n  for (const operation of Object.keys(OPERATION_ROUTES)) {\n    if (getDefaultPermissionLevel(operation) === level) {\n      results.push(operation);\n    }\n  }\n\n  return results;\n}\n\n/**\n * Get all auto-approved operations.\n * These are safe to execute without any confirmation.\n */\nexport function getAutoApprovedOperations(): string[] {\n  return getOperationsAtLevel(PermissionLevel.AUTO_APPROVE);\n}\n\n/**\n * Get all operations requiring confirmation.\n * These need user approval before execution.\n */\nexport function getConfirmationRequiredOperations(): string[] {\n  return [\n    ...getOperationsAtLevel(PermissionLevel.CONFIRM_SESSION),\n    ...getOperationsAtLevel(PermissionLevel.CONFIRM_SINGLE_USE),\n  ];\n}\n"]}
252
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"OperationPolicies.js","sourceRoot":"","sources":["../../../../src/handlers/mcp-aql/policies/OperationPolicies.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA2CG;AAEH,OAAO,EAAE,eAAe,EAAwB,MAAM,uBAAuB,CAAC;AAC9E,OAAO,EAAE,QAAQ,EAAE,gBAAgB,EAAqB,MAAM,uBAAuB,CAAC;AAEtF;;;;;GAKG;AACH,MAAM,uBAAuB,GAA0C;IACrE,IAAI,EAAE,eAAe,CAAC,YAAY;IAClC,MAAM,EAAE,eAAe,CAAC,eAAe;IACvC,MAAM,EAAE,eAAe,CAAC,kBAAkB;IAC1C,MAAM,EAAE,eAAe,CAAC,kBAAkB;IAC1C,OAAO,EAAE,eAAe,CAAC,kBAAkB;CAC5C,CAAC;AAEF;;;;;GAKG;AACH,MAAM,UAAU,uBAAuB,CAAC,QAAsB;IAC5D,OAAO,uBAAuB,CAAC,QAAQ,CAAC,CAAC;AAC3C,CAAC;AAED;;;;;;;;;GASG;AACH,MAAM,CAAC,MAAM,0BAA0B,GAAoC;IACzE,wCAAwC;IACxC,sEAAsE;IACtE,gBAAgB,EAAE;QAChB,YAAY,EAAE,eAAe,CAAC,YAAY;QAC1C,SAAS,EAAE,qFAAqF;KACjG;IACD,gBAAgB,EAAE;QAChB,YAAY,EAAE,eAAe,CAAC,YAAY;QAC1C,SAAS,EAAE,wHAAwH;QACnI,aAAa,EAAE,KAAK;KACrB;IACD,mCAAmC,EAAE;QACnC,YAAY,EAAE,eAAe,CAAC,YAAY;QAC1C,SAAS,EAAE,2GAA2G;KACvH;IAED,wCAAwC;IACxC,sFAAsF;IACtF,cAAc,EAAE;QACd,YAAY,EAAE,eAAe,CAAC,kBAAkB;QAChD,SAAS,EAAE,iDAAiD;QAC5D,aAAa,EAAE,KAAK;KACrB;IACD,KAAK,EAAE;QACL,YAAY,EAAE,eAAe,CAAC,kBAAkB;QAChD,SAAS,EAAE,kDAAkD;QAC7D,aAAa,EAAE,KAAK;KACrB;IACD,iBAAiB,EAAE;QACjB,YAAY,EAAE,eAAe,CAAC,kBAAkB;QAChD,SAAS,EAAE,2DAA2D;QACtE,aAAa,EAAE,KAAK;KACrB;IAED,yCAAyC;IACzC,8EAA8E;IAE9E,6DAA6D;IAC7D,iBAAiB,EAAE;QACjB,YAAY,EAAE,eAAe,CAAC,YAAY;QAC1C,SAAS,EAAE,iGAAiG;KAC7G;IACD,8CAA8C;IAC9C,iBAAiB,EAAE;QACjB,YAAY,EAAE,eAAe,CAAC,YAAY;QAC1C,SAAS,EAAE,uFAAuF;QAClG,aAAa,EAAE,KAAK;KACrB;IACD,4CAA4C;IAC5C,sBAAsB,EAAE;QACtB,YAAY,EAAE,eAAe,CAAC,YAAY;QAC1C,SAAS,EAAE,sFAAsF;QACjG,aAAa,EAAE,KAAK;KACrB;IAED,2EAA2E;IAC3E,2EAA2E;IAC3E,mDAAmD;IACnD,6EAA6E;IAE7E,wEAAwE;IACxE,kBAAkB,EAAE;QAClB,YAAY,EAAE,eAAe,CAAC,eAAe;QAC7C,SAAS,EAAE,wDAAwD;QACnE,aAAa,EAAE,IAAI;KACpB;IACD,kBAAkB,EAAE;QAClB,YAAY,EAAE,eAAe,CAAC,eAAe;QAC7C,SAAS,EAAE,oCAAoC;QAC/C,aAAa,EAAE,IAAI;KACpB;IACD,eAAe,EAAE;QACf,YAAY,EAAE,eAAe,CAAC,eAAe;QAC7C,SAAS,EAAE,kEAAkE;QAC7E,aAAa,EAAE,IAAI;KACpB;IAED,gFAAgF;IAChF,aAAa,EAAE;QACb,YAAY,EAAE,eAAe,CAAC,kBAAkB;QAChD,SAAS,EAAE,gDAAgD;QAC3D,aAAa,EAAE,KAAK;KACrB;IACD,eAAe,EAAE;QACf,YAAY,EAAE,eAAe,CAAC,kBAAkB;QAChD,SAAS,EAAE,yEAAyE;QACpF,aAAa,EAAE,KAAK;KACrB;IACD,mBAAmB,EAAE;QACnB,YAAY,EAAE,eAAe,CAAC,kBAAkB;QAChD,SAAS,EAAE,mEAAmE;QAC9E,aAAa,EAAE,KAAK;KACrB;CACF,CAAC;AAEF;;;GAGG;AACH,MAAM,CAAC,MAAM,kBAAkB,GAAG,0BAA0B,CAAC;AAE7D;;;;;GAKG;AACH,MAAM,UAAU,kBAAkB,CAAC,SAAiB;IAClD,OAAO,0BAA0B,CAAC,SAAS,CAAC,CAAC;AAC/C,CAAC;AAED;;;;;;;;;;GAUG;AACH,MAAM,UAAU,yBAAyB,CAAC,SAAiB;IACzD,iCAAiC;IACjC,MAAM,QAAQ,GAAG,0BAA0B,CAAC,SAAS,CAAC,CAAC;IACvD,IAAI,QAAQ,EAAE,CAAC;QACb,OAAO,QAAQ,CAAC,YAAY,CAAC;IAC/B,CAAC;IAED,kCAAkC;IAClC,MAAM,KAAK,GAAG,QAAQ,CAAC,SAAS,CAAC,CAAC;IAClC,IAAI,KAAK,EAAE,CAAC;QACV,OAAO,uBAAuB,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC;IACjD,CAAC;IAED,4CAA4C;IAC5C,OAAO,eAAe,CAAC,kBAAkB,CAAC;AAC5C,CAAC;AAED;;;;;;GAMG;AACH,MAAM,UAAU,sBAAsB,CAAC,SAAiB;IACtD,MAAM,QAAQ,GAAG,0BAA0B,CAAC,SAAS,CAAC,CAAC;IACvD,0EAA0E;IAC1E,OAAO,QAAQ,EAAE,aAAa,IAAI,IAAI,CAAC;AACzC,CAAC;AAED;;;;;;GAMG;AACH,MAAM,UAAU,oBAAoB,CAAC,KAAsB;IACzD,MAAM,OAAO,GAAa,EAAE,CAAC;IAE7B,KAAK,MAAM,SAAS,IAAI,MAAM,CAAC,IAAI,CAAC,gBAAgB,CAAC,EAAE,CAAC;QACtD,IAAI,yBAAyB,CAAC,SAAS,CAAC,KAAK,KAAK,EAAE,CAAC;YACnD,OAAO,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QAC1B,CAAC;IACH,CAAC;IAED,OAAO,OAAO,CAAC;AACjB,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,yBAAyB;IACvC,OAAO,oBAAoB,CAAC,eAAe,CAAC,YAAY,CAAC,CAAC;AAC5D,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,iCAAiC;IAC/C,OAAO;QACL,GAAG,oBAAoB,CAAC,eAAe,CAAC,eAAe,CAAC;QACxD,GAAG,oBAAoB,CAAC,eAAe,CAAC,kBAAkB,CAAC;KAC5D,CAAC;AACJ,CAAC","sourcesContent":["/**\n * Operation Policies\n *\n * Default permission levels are derived from the operation's endpoint routing:\n * - READ    → AUTO_APPROVE (read-only, no side effects)\n * - CREATE  → CONFIRM_SESSION (additive state changes, safe once approved)\n * - UPDATE  → CONFIRM_SINGLE_USE (modifying existing data, each instance reviewed)\n * - DELETE  → CONFIRM_SINGLE_USE (destructive, each instance reviewed)\n * - EXECUTE → CONFIRM_SINGLE_USE (unpredictable side effects)\n *\n * OPERATION_POLICY_OVERRIDES contains only operations that deviate from their\n * endpoint default — e.g., operations that need stricter policies, special\n * canBeElevated flags, or looser defaults than their endpoint implies.\n *\n * Resolution hierarchy:\n * 1. Element policy (Layer 2 — active element allow/confirm/deny lists)\n * 2. Explicit override in OPERATION_POLICY_OVERRIDES\n * 3. Endpoint default (derived from OperationRouter)\n * 4. Secure fallback: CONFIRM_SINGLE_USE for unknown operations\n *\n * Examples:\n *\n *   activate_element → READ endpoint → AUTO_APPROVE (no override needed)\n *     Routing to READ is sufficient. No entry in OPERATION_POLICY_OVERRIDES.\n *\n *   create_element → CREATE endpoint → CONFIRM_SESSION (no override needed)\n *     Endpoint default handles it. First create confirms, rest of session is smooth.\n *\n *   verify_challenge → CREATE endpoint → overridden to AUTO_APPROVE\n *     On CREATE (default CONFIRM_SESSION) but must be frictionless to avoid\n *     requiring confirmation to complete a verification flow.\n *\n *   delete_element → DELETE endpoint → CONFIRM_SINGLE_USE + canBeElevated: false\n *     Matches endpoint default level, but override locks canBeElevated so no\n *     element policy can silently auto-approve deletions.\n *\n *   record_execution_step → CREATE endpoint → CONFIRM_SESSION (no override needed)\n *     Moved from EXECUTE to CREATE so it inherits CONFIRM_SESSION, matching\n *     the polling loop's need for session-level frictionless approval.\n *\n *   A persona with gatekeeper: { confirm: ['create_element'] } → CONFIRM_SESSION\n *     Element policy (Layer 2) can tighten or loosen within what the operation allows.\n *     This is a runtime override via element activation, not a code change.\n */\n\nimport { PermissionLevel, type OperationPolicy } from '../GatekeeperTypes.js';\nimport { getRoute, OPERATION_ROUTES, type CRUDEndpoint } from '../OperationRouter.js';\n\n/**\n * Map CRUDE endpoints to their default permission levels.\n *\n * This is the single source of truth for what each endpoint implies.\n * If an operation is routed to READ, it gets AUTO_APPROVE unless overridden.\n */\nconst ENDPOINT_DEFAULT_LEVELS: Record<CRUDEndpoint, PermissionLevel> = {\n  READ: PermissionLevel.AUTO_APPROVE,\n  CREATE: PermissionLevel.CONFIRM_SESSION,\n  UPDATE: PermissionLevel.CONFIRM_SINGLE_USE,\n  DELETE: PermissionLevel.CONFIRM_SINGLE_USE,\n  EXECUTE: PermissionLevel.CONFIRM_SINGLE_USE,\n};\n\n/**\n * Get the default permission level for a CRUDE endpoint.\n *\n * @param endpoint - The CRUDE endpoint\n * @returns The default permission level for that endpoint\n */\nexport function getEndpointDefaultLevel(endpoint: CRUDEndpoint): PermissionLevel {\n  return ENDPOINT_DEFAULT_LEVELS[endpoint];\n}\n\n/**\n * Explicit policy overrides for operations that deviate from their endpoint default.\n *\n * Only add entries here when an operation needs:\n * - A different permission level than its endpoint implies\n * - A canBeElevated: false restriction\n * - A specific rationale that differs from the endpoint's general rationale\n *\n * Operations NOT listed here inherit their permission level from their endpoint.\n */\nexport const OPERATION_POLICY_OVERRIDES: Record<string, OperationPolicy> = {\n  // ===== CREATE endpoint overrides =====\n  // These are on CREATE (default CONFIRM_SESSION) but need AUTO_APPROVE\n  verify_challenge: {\n    defaultLevel: PermissionLevel.AUTO_APPROVE,\n    rationale: 'Verification flow — must be auto-approved to avoid requiring confirmation to verify',\n  },\n  release_deadlock: {\n    defaultLevel: PermissionLevel.AUTO_APPROVE,\n    rationale: 'Deadlock relief flow — requires out-of-band verification and must remain reachable even when confirmations are blocked',\n    canBeElevated: false,\n  },\n  beetlejuice_beetlejuice_beetlejuice: {\n    defaultLevel: PermissionLevel.AUTO_APPROVE,\n    rationale: 'Test fixture for danger zone verification — must be auto-approved to avoid requiring confirmation to test',\n  },\n\n  // ===== DELETE endpoint overrides =====\n  // These match the endpoint default (CONFIRM_SINGLE_USE) but need canBeElevated: false\n  delete_element: {\n    defaultLevel: PermissionLevel.CONFIRM_SINGLE_USE,\n    rationale: 'Destructive operation, permanently removes data',\n    canBeElevated: false,\n  },\n  clear: {\n    defaultLevel: PermissionLevel.CONFIRM_SINGLE_USE,\n    rationale: 'Destructive operation, clears all memory entries',\n    canBeElevated: false,\n  },\n  clear_github_auth: {\n    defaultLevel: PermissionLevel.CONFIRM_SINGLE_USE,\n    rationale: 'Destructive operation, removes authentication credentials',\n    canBeElevated: false,\n  },\n\n  // ===== EXECUTE endpoint overrides =====\n  // These are on EXECUTE (default CONFIRM_SINGLE_USE) but need different levels\n\n  // Must be auto-approved to avoid infinite confirmation loops\n  confirm_operation: {\n    defaultLevel: PermissionLevel.AUTO_APPROVE,\n    rationale: 'Gatekeeper confirmation flow — must be auto-approved to avoid requiring confirmation to confirm',\n  },\n  // Issue #625: CLI-level permission delegation\n  permission_prompt: {\n    defaultLevel: PermissionLevel.AUTO_APPROVE,\n    rationale: 'Permission evaluation flow — must be auto-approved to avoid infinite permission loops',\n    canBeElevated: false,\n  },\n  // Issue #625 Phase 3: CLI approval workflow\n  approve_cli_permission: {\n    defaultLevel: PermissionLevel.AUTO_APPROVE,\n    rationale: 'CLI approval flow — must be auto-approved to avoid requiring confirmation to approve',\n    canBeElevated: false,\n  },\n\n  // Note: get_execution_state and get_gathered_data moved to READ endpoint —\n  // they inherit AUTO_APPROVE from the endpoint default, no override needed.\n  // record_execution_step moved to CREATE endpoint —\n  // it inherits CONFIRM_SESSION from the endpoint default, no override needed.\n\n  // Session-confirmable lifecycle operations (less friction than per-use)\n  complete_execution: {\n    defaultLevel: PermissionLevel.CONFIRM_SESSION,\n    rationale: 'Signals execution completion, part of active execution',\n    canBeElevated: true,\n  },\n  continue_execution: {\n    defaultLevel: PermissionLevel.CONFIRM_SESSION,\n    rationale: 'Resumes execution from saved state',\n    canBeElevated: true,\n  },\n  prepare_handoff: {\n    defaultLevel: PermissionLevel.CONFIRM_SESSION,\n    rationale: 'Prepares session handoff snapshot, read-heavy with serialization',\n    canBeElevated: true,\n  },\n\n  // Non-elevatable execute operations (match endpoint default but lock elevation)\n  execute_agent: {\n    defaultLevel: PermissionLevel.CONFIRM_SINGLE_USE,\n    rationale: 'Executes agent with unpredictable side effects',\n    canBeElevated: false,\n  },\n  abort_execution: {\n    defaultLevel: PermissionLevel.CONFIRM_SINGLE_USE,\n    rationale: 'Terminates agent execution, rejecting further operations for the goalId',\n    canBeElevated: false,\n  },\n  resume_from_handoff: {\n    defaultLevel: PermissionLevel.CONFIRM_SINGLE_USE,\n    rationale: 'Resumes execution from external handoff data, requires validation',\n    canBeElevated: false,\n  },\n};\n\n/**\n * @deprecated Use OPERATION_POLICY_OVERRIDES instead.\n * Kept as an alias for backward compatibility with code that imports OPERATION_POLICIES.\n */\nexport const OPERATION_POLICIES = OPERATION_POLICY_OVERRIDES;\n\n/**\n * Get the explicit policy override for an operation, if one exists.\n *\n * @param operation - The operation name\n * @returns The override policy, or undefined if no override exists\n */\nexport function getOperationPolicy(operation: string): OperationPolicy | undefined {\n  return OPERATION_POLICY_OVERRIDES[operation];\n}\n\n/**\n * Get the default permission level for an operation.\n *\n * Resolution order:\n * 1. Explicit override in OPERATION_POLICY_OVERRIDES\n * 2. Endpoint default from OperationRouter\n * 3. CONFIRM_SINGLE_USE for unknown operations (secure fallback)\n *\n * @param operation - The operation name\n * @returns The effective default permission level\n */\nexport function getDefaultPermissionLevel(operation: string): PermissionLevel {\n  // 1. Check for explicit override\n  const override = OPERATION_POLICY_OVERRIDES[operation];\n  if (override) {\n    return override.defaultLevel;\n  }\n\n  // 2. Derive from endpoint routing\n  const route = getRoute(operation);\n  if (route) {\n    return ENDPOINT_DEFAULT_LEVELS[route.endpoint];\n  }\n\n  // 3. Secure fallback for unknown operations\n  return PermissionLevel.CONFIRM_SINGLE_USE;\n}\n\n/**\n * Check if an operation can have its permission level elevated.\n * Some destructive operations cannot be elevated to AUTO_APPROVE.\n *\n * @param operation - The operation name\n * @returns true if the operation can be elevated\n */\nexport function canOperationBeElevated(operation: string): boolean {\n  const override = OPERATION_POLICY_OVERRIDES[operation];\n  // Default to allowing elevation for operations without explicit overrides\n  return override?.canBeElevated ?? true;\n}\n\n/**\n * Get all operations at a specific effective permission level.\n * Considers both explicit overrides and endpoint-derived defaults.\n *\n * @param level - The permission level to filter by\n * @returns Array of operation names at that level\n */\nexport function getOperationsAtLevel(level: PermissionLevel): string[] {\n  const results: string[] = [];\n\n  for (const operation of Object.keys(OPERATION_ROUTES)) {\n    if (getDefaultPermissionLevel(operation) === level) {\n      results.push(operation);\n    }\n  }\n\n  return results;\n}\n\n/**\n * Get all auto-approved operations.\n * These are safe to execute without any confirmation.\n */\nexport function getAutoApprovedOperations(): string[] {\n  return getOperationsAtLevel(PermissionLevel.AUTO_APPROVE);\n}\n\n/**\n * Get all operations requiring confirmation.\n * These need user approval before execution.\n */\nexport function getConfirmationRequiredOperations(): string[] {\n  return [\n    ...getOperationsAtLevel(PermissionLevel.CONFIRM_SESSION),\n    ...getOperationsAtLevel(PermissionLevel.CONFIRM_SINGLE_USE),\n  ];\n}\n"]}