@dollhousemcp/mcp-server 2.0.30 → 2.0.32

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (40) hide show
  1. package/CHANGELOG.md +9 -0
  2. package/data/ensembles/welcome-to-the-dollhouse.md +94 -0
  3. package/data/personas/dollhouse-expert.md +242 -14
  4. package/data/skills/research-to-elements.md +136 -0
  5. package/dist/elements/agents/AgentManager.d.ts +8 -1
  6. package/dist/elements/agents/AgentManager.d.ts.map +1 -1
  7. package/dist/elements/agents/AgentManager.js +49 -9
  8. package/dist/elements/ensembles/EnsembleManager.d.ts +36 -0
  9. package/dist/elements/ensembles/EnsembleManager.d.ts.map +1 -1
  10. package/dist/elements/ensembles/EnsembleManager.js +94 -5
  11. package/dist/generated/version.d.ts +2 -2
  12. package/dist/generated/version.js +3 -3
  13. package/dist/handlers/mcp-aql/MCPAQLHandler.d.ts.map +1 -1
  14. package/dist/handlers/mcp-aql/MCPAQLHandler.js +31 -2
  15. package/dist/handlers/mcp-aql/OperationSchema.d.ts.map +1 -1
  16. package/dist/handlers/mcp-aql/OperationSchema.js +8 -7
  17. package/dist/handlers/mcp-aql/evaluatePermission.d.ts.map +1 -1
  18. package/dist/handlers/mcp-aql/evaluatePermission.js +6 -3
  19. package/dist/server/tools/MCPAQLTools.js +21 -5
  20. package/dist/services/BuildInfoService.d.ts +3 -1
  21. package/dist/services/BuildInfoService.d.ts.map +1 -1
  22. package/dist/services/BuildInfoService.js +27 -5
  23. package/dist/utils/permissionHookShared.d.ts +39 -0
  24. package/dist/utils/permissionHookShared.d.ts.map +1 -1
  25. package/dist/utils/permissionHookShared.js +38 -1
  26. package/dist/utils/permissionHookStatus.d.ts.map +1 -1
  27. package/dist/utils/permissionHookStatus.js +6 -2
  28. package/dist/utils/permissionHooks.d.ts +1 -1
  29. package/dist/utils/permissionHooks.d.ts.map +1 -1
  30. package/dist/utils/permissionHooks.js +2 -2
  31. package/dist/web/public/app.js +213 -12
  32. package/dist/web/public/index.html +16 -7
  33. package/dist/web/public/styles.css +229 -30
  34. package/dist/web/routes/healthRoutes.d.ts.map +1 -1
  35. package/dist/web/routes/healthRoutes.js +3 -1
  36. package/dist/web/routes/permissionRoutes.d.ts.map +1 -1
  37. package/dist/web/routes/permissionRoutes.js +10 -2
  38. package/package.json +1 -1
  39. package/scripts/pretooluse-dollhouse.sh +157 -39
  40. package/server.json +2 -2
@@ -1 +1 @@
1
- {"version":3,"file":"evaluatePermission.d.ts","sourceRoot":"","sources":["../../../src/handlers/mcp-aql/evaluatePermission.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;GAcG;AAEH,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,4BAA4B,CAAC;AAC9D,OAAO,KAAK,EAAE,wBAAwB,EAAE,mBAAmB,EAAE,MAAM,kCAAkC,CAAC;AACtG,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,+BAA+B,CAAC;AAEnE,wEAAwE;AACxE,qBAAa,yBAA0B,SAAQ,KAAK;IAClD,SAAgB,KAAK,EAAE,YAAY,GAAG,gBAAgB,GAAG,QAAQ,GAAG,eAAe,CAAC;IACpF,SAAgB,QAAQ,EAAE,MAAM,CAAC;gBAG/B,OAAO,EAAE,MAAM,EACf,KAAK,EAAE,YAAY,GAAG,gBAAgB,GAAG,QAAQ,GAAG,eAAe,EACnE,QAAQ,EAAE,MAAM,EAChB,KAAK,CAAC,EAAE,OAAO;CAOlB;AAED,+CAA+C;AAC/C,MAAM,WAAW,sBAAsB;IACrC,uBAAuB,EAAE,WAAW,CAAC;IACrC,YAAY,EAAE,CAAC,QAAQ,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,KAAK,wBAAwB,CAAC;IACjG,qBAAqB,EAAE,CAAC,QAAQ,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAAE,QAAQ,EAAE,aAAa,EAAE,KAAK,mBAAmB,CAAC;IAChI,iBAAiB,EAAE,CAAC,SAAS,CAAC,EAAE,MAAM,KAAK,OAAO,CAAC,aAAa,EAAE,CAAC,CAAC;CACrE;AAqDD,iCAAiC;AACjC,eAAO,MAAM,mBAAmB,UAAkC,CAAC;AAiBnE;;;;;GAKG;AACH,wBAAgB,wBAAwB,CACtC,QAAQ,EAAE,OAAO,GAAG,MAAM,GAAG,KAAK,EAClC,QAAQ,EAAE,MAAM,EAChB,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAC/B,MAAM,CAAC,EAAE,MAAM,GACd,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAYzB;AAED;;;;;;GAMG;AACH,wBAAsB,kBAAkB,CACtC,MAAM,EAAE;IAAE,SAAS,CAAC,EAAE,OAAO,CAAC;IAAC,KAAK,CAAC,EAAE,OAAO,CAAC;IAAC,QAAQ,CAAC,EAAE,OAAO,CAAC;IAAC,UAAU,CAAC,EAAE,OAAO,CAAA;CAAE,EAC1F,IAAI,EAAE,sBAAsB,GAC3B,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC,CAiDlC"}
1
+ {"version":3,"file":"evaluatePermission.d.ts","sourceRoot":"","sources":["../../../src/handlers/mcp-aql/evaluatePermission.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;GAcG;AAEH,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,4BAA4B,CAAC;AAC9D,OAAO,KAAK,EAAE,wBAAwB,EAAE,mBAAmB,EAAE,MAAM,kCAAkC,CAAC;AACtG,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,+BAA+B,CAAC;AAEnE,wEAAwE;AACxE,qBAAa,yBAA0B,SAAQ,KAAK;IAClD,SAAgB,KAAK,EAAE,YAAY,GAAG,gBAAgB,GAAG,QAAQ,GAAG,eAAe,CAAC;IACpF,SAAgB,QAAQ,EAAE,MAAM,CAAC;gBAG/B,OAAO,EAAE,MAAM,EACf,KAAK,EAAE,YAAY,GAAG,gBAAgB,GAAG,QAAQ,GAAG,eAAe,EACnE,QAAQ,EAAE,MAAM,EAChB,KAAK,CAAC,EAAE,OAAO;CAOlB;AAED,+CAA+C;AAC/C,MAAM,WAAW,sBAAsB;IACrC,uBAAuB,EAAE,WAAW,CAAC;IACrC,YAAY,EAAE,CAAC,QAAQ,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,KAAK,wBAAwB,CAAC;IACjG,qBAAqB,EAAE,CAAC,QAAQ,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAAE,QAAQ,EAAE,aAAa,EAAE,KAAK,mBAAmB,CAAC;IAChI,iBAAiB,EAAE,CAAC,SAAS,CAAC,EAAE,MAAM,KAAK,OAAO,CAAC,aAAa,EAAE,CAAC,CAAC;CACrE;AAyDD,iCAAiC;AACjC,eAAO,MAAM,mBAAmB,UAAkC,CAAC;AAiBnE;;;;;GAKG;AACH,wBAAgB,wBAAwB,CACtC,QAAQ,EAAE,OAAO,GAAG,MAAM,GAAG,KAAK,EAClC,QAAQ,EAAE,MAAM,EAChB,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAC/B,MAAM,CAAC,EAAE,MAAM,GACd,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAYzB;AAED;;;;;;GAMG;AACH,wBAAsB,kBAAkB,CACtC,MAAM,EAAE;IAAE,SAAS,CAAC,EAAE,OAAO,CAAC;IAAC,KAAK,CAAC,EAAE,OAAO,CAAC;IAAC,QAAQ,CAAC,EAAE,OAAO,CAAC;IAAC,UAAU,CAAC,EAAE,OAAO,CAAA;CAAE,EAC1F,IAAI,EAAE,sBAAsB,GAC3B,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC,CAiDlC"}
@@ -40,13 +40,16 @@ function formatCursor(decision, reason) {
40
40
  function formatWindsurf(decision, reason) {
41
41
  return withReason({ allowed: decision === 'allow' }, reason);
42
42
  }
43
- /** Codex: explicit PreToolUse payload, maps 'ask' to 'deny' */
43
+ /** Codex PreToolUse currently supports block/deny only; allow is empty stdout. */
44
44
  function formatCodex(decision, reason) {
45
+ if (decision === 'allow') {
46
+ return {};
47
+ }
45
48
  return {
46
49
  hookSpecificOutput: {
47
50
  hookEventName: 'PreToolUse',
48
51
  permissionDecision: decision === 'ask' ? 'deny' : decision,
49
- permissionDecisionReason: reason ?? '',
52
+ ...(reason ? { permissionDecisionReason: reason } : {}),
50
53
  },
51
54
  };
52
55
  }
@@ -147,4 +150,4 @@ export async function evaluatePermission(params, deps) {
147
150
  // Default: allow
148
151
  return formatPermissionResponse('allow', platform, input);
149
152
  }
150
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"evaluatePermission.js","sourceRoot":"","sources":["../../../src/handlers/mcp-aql/evaluatePermission.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Permission evaluation for PreToolUse hooks across all AI platforms.\n *\n * Provides the `evaluate_permission` MCP-AQL READ operation, enabling\n * Claude Code, Cursor, Gemini CLI, Windsurf, and Codex to use\n * DollhouseMCP as their permission evaluation backend via hooks.\n *\n * Three-stage evaluation pipeline:\n * 1. Rate limiting — prevents abuse\n * 2. Static tool classification — built-in allow/deny rules\n * 3. Element policy evaluation — active element gatekeeper policies\n *\n * Returns platform-specific response formats so each platform's hook\n * script receives the JSON shape it expects.\n */\n\nimport type { RateLimiter } from '../../utils/RateLimiter.js';\nimport type { ToolClassificationResult, CliToolPolicyResult } from './policies/ToolClassification.js';\nimport type { ActiveElement } from './policies/ElementPolicies.js';\n\n/** Error thrown when permission evaluation fails at a specific stage */\nexport class PermissionEvaluationError extends Error {\n  public readonly stage: 'rate_limit' | 'classification' | 'policy' | 'element_fetch';\n  public readonly toolName: string;\n\n  constructor(\n    message: string,\n    stage: 'rate_limit' | 'classification' | 'policy' | 'element_fetch',\n    toolName: string,\n    cause?: unknown,\n  ) {\n    super(message, cause ? { cause } : undefined);\n    this.name = 'PermissionEvaluationError';\n    this.stage = stage;\n    this.toolName = toolName;\n  }\n}\n\n/** Dependencies injected from MCPAQLHandler */\nexport interface EvaluatePermissionDeps {\n  permissionPromptLimiter: RateLimiter;\n  classifyTool: (toolName: string, toolInput: Record<string, unknown>) => ToolClassificationResult;\n  evaluateCliToolPolicy: (toolName: string, toolInput: Record<string, unknown>, elements: ActiveElement[]) => CliToolPolicyResult;\n  getActiveElements: (sessionId?: string) => Promise<ActiveElement[]>;\n}\n\n/** Optional reason field, only included when reason is provided */\nfunction withReason(obj: Record<string, unknown>, reason?: string, key = 'reason'): Record<string, unknown> {\n  return reason ? { ...obj, [key]: reason } : obj;\n}\n\n/** Gemini: maps 'ask' to 'deny' (no interactive support) */\nfunction formatGemini(decision: string, reason?: string): Record<string, unknown> {\n  return withReason({ decision: decision === 'ask' ? 'deny' : decision }, reason);\n}\n\n/** Cursor: uses 'permission' field instead of 'decision' */\nfunction formatCursor(decision: string, reason?: string): Record<string, unknown> {\n  return withReason({ permission: decision }, reason);\n}\n\n/** Windsurf: uses boolean 'allowed' field */\nfunction formatWindsurf(decision: string, reason?: string): Record<string, unknown> {\n  return withReason({ allowed: decision === 'allow' }, reason);\n}\n\n/** Codex: explicit PreToolUse payload, maps 'ask' to 'deny' */\nfunction formatCodex(decision: string, reason?: string): Record<string, unknown> {\n  return {\n    hookSpecificOutput: {\n      hookEventName: 'PreToolUse',\n      permissionDecision: decision === 'ask' ? 'deny' : decision,\n      permissionDecisionReason: reason ?? '',\n    },\n  };\n}\n\n/** Claude Code (default): uses hookSpecificOutput.permissionDecision for PreToolUse */\nfunction formatClaudeCode(decision: string, reason?: string): Record<string, unknown> {\n  return {\n    hookSpecificOutput: withReason({\n      hookEventName: 'PreToolUse',\n      permissionDecision: decision,\n    }, reason, 'permissionDecisionReason'),\n  };\n}\n\n/** Platform formatter lookup */\nconst platformFormatters: Record<string, (decision: string, reason?: string) => Record<string, unknown>> = {\n  gemini: formatGemini,\n  cursor: formatCursor,\n  windsurf: formatWindsurf,\n  codex: formatCodex,\n  vscode: formatClaudeCode,\n  claude_code: formatClaudeCode,\n};\n\n/** Known platform identifiers */\nexport const SUPPORTED_PLATFORMS = Object.keys(platformFormatters);\n\nfunction warnOnUnknownPlatform(platform: string): void {\n  void import('../../utils/logger.js')\n    .then(({ logger }) => {\n      logger.warn(\n        `[evaluatePermission] Unknown platform \"${platform}\", defaulting to claude_code format. Supported: ${SUPPORTED_PLATFORMS.join(', ')}`,\n      );\n    })\n    .catch((error) => {\n      console.warn(\n        `[evaluatePermission] Failed to load logger while handling unknown platform \"${platform}\".`,\n        error,\n      );\n    });\n}\n\n/**\n * Format permission evaluation response for platform-specific hook scripts.\n * Each platform expects a different JSON shape from its hook response.\n *\n * Unknown platforms default to claude_code format with a warning log.\n */\nexport function formatPermissionResponse(\n  decision: 'allow' | 'deny' | 'ask',\n  platform: string,\n  _input: Record<string, unknown>,\n  reason?: string,\n): Record<string, unknown> {\n  switch (platform) {\n    case 'gemini': return formatGemini(decision, reason);\n    case 'cursor': return formatCursor(decision, reason);\n    case 'windsurf': return formatWindsurf(decision, reason);\n    case 'codex': return formatCodex(decision, reason);\n    case 'claude_code': return formatClaudeCode(decision, reason);\n    default:\n      // Import lazily to avoid circular dependency at module load time\n      warnOnUnknownPlatform(platform);\n      return formatClaudeCode(decision, reason);\n  }\n}\n\n/**\n * Evaluate a CLI permission request for PreToolUse hooks.\n *\n * @param params - Tool name, input, and target platform\n * @param deps - Injected dependencies from MCPAQLHandler\n * @returns Platform-formatted permission decision\n */\nexport async function evaluatePermission(\n  params: { tool_name?: unknown; input?: unknown; platform?: unknown; session_id?: unknown },\n  deps: EvaluatePermissionDeps,\n): Promise<Record<string, unknown>> {\n  const toolName = typeof params.tool_name === 'string' ? params.tool_name : '';\n  const inputRaw = params.input;\n  const input = (inputRaw && typeof inputRaw === 'object')\n    ? inputRaw as Record<string, unknown>\n    : {};\n  const platform = typeof params.platform === 'string' ? params.platform : 'claude_code';\n  const sessionId = typeof params.session_id === 'string' && params.session_id.trim() !== ''\n    ? params.session_id\n    : undefined;\n\n  // Rate limit\n  const rateStatus = deps.permissionPromptLimiter.checkLimit();\n  if (!rateStatus.allowed) {\n    return formatPermissionResponse('deny', platform, input, 'Rate limit exceeded');\n  }\n  deps.permissionPromptLimiter.consumeToken();\n\n  // Stage 1: Static classification\n  const classification = deps.classifyTool(toolName, input);\n  if (classification.behavior === 'allow') {\n    return formatPermissionResponse('allow', platform, input);\n  }\n  if (classification.behavior === 'deny') {\n    return formatPermissionResponse('deny', platform, input, classification.reason);\n  }\n\n  // Stage 2: Element policy evaluation\n  let elements: ActiveElement[];\n  try {\n    elements = await deps.getActiveElements(sessionId);\n  } catch (err) {\n    throw new PermissionEvaluationError(\n      `Failed to fetch active elements for policy evaluation: ${err instanceof Error ? err.message : String(err)}`,\n      'element_fetch', toolName, err,\n    );\n  }\n  const decision = deps.evaluateCliToolPolicy(toolName, input, elements);\n\n  if (decision.behavior === 'deny') {\n    return formatPermissionResponse('deny', platform, input, decision.message);\n  }\n  if (decision.behavior === 'confirm') {\n    return formatPermissionResponse('ask', platform, input,\n      decision.message || 'Requires confirmation per element policy');\n  }\n\n  // Default: allow\n  return formatPermissionResponse('allow', platform, input);\n}\n"]}
153
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"evaluatePermission.js","sourceRoot":"","sources":["../../../src/handlers/mcp-aql/evaluatePermission.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Permission evaluation for PreToolUse hooks across all AI platforms.\n *\n * Provides the `evaluate_permission` MCP-AQL READ operation, enabling\n * Claude Code, Cursor, Gemini CLI, Windsurf, and Codex to use\n * DollhouseMCP as their permission evaluation backend via hooks.\n *\n * Three-stage evaluation pipeline:\n * 1. Rate limiting — prevents abuse\n * 2. Static tool classification — built-in allow/deny rules\n * 3. Element policy evaluation — active element gatekeeper policies\n *\n * Returns platform-specific response formats so each platform's hook\n * script receives the JSON shape it expects.\n */\n\nimport type { RateLimiter } from '../../utils/RateLimiter.js';\nimport type { ToolClassificationResult, CliToolPolicyResult } from './policies/ToolClassification.js';\nimport type { ActiveElement } from './policies/ElementPolicies.js';\n\n/** Error thrown when permission evaluation fails at a specific stage */\nexport class PermissionEvaluationError extends Error {\n  public readonly stage: 'rate_limit' | 'classification' | 'policy' | 'element_fetch';\n  public readonly toolName: string;\n\n  constructor(\n    message: string,\n    stage: 'rate_limit' | 'classification' | 'policy' | 'element_fetch',\n    toolName: string,\n    cause?: unknown,\n  ) {\n    super(message, cause ? { cause } : undefined);\n    this.name = 'PermissionEvaluationError';\n    this.stage = stage;\n    this.toolName = toolName;\n  }\n}\n\n/** Dependencies injected from MCPAQLHandler */\nexport interface EvaluatePermissionDeps {\n  permissionPromptLimiter: RateLimiter;\n  classifyTool: (toolName: string, toolInput: Record<string, unknown>) => ToolClassificationResult;\n  evaluateCliToolPolicy: (toolName: string, toolInput: Record<string, unknown>, elements: ActiveElement[]) => CliToolPolicyResult;\n  getActiveElements: (sessionId?: string) => Promise<ActiveElement[]>;\n}\n\n/** Optional reason field, only included when reason is provided */\nfunction withReason(obj: Record<string, unknown>, reason?: string, key = 'reason'): Record<string, unknown> {\n  return reason ? { ...obj, [key]: reason } : obj;\n}\n\n/** Gemini: maps 'ask' to 'deny' (no interactive support) */\nfunction formatGemini(decision: string, reason?: string): Record<string, unknown> {\n  return withReason({ decision: decision === 'ask' ? 'deny' : decision }, reason);\n}\n\n/** Cursor: uses 'permission' field instead of 'decision' */\nfunction formatCursor(decision: string, reason?: string): Record<string, unknown> {\n  return withReason({ permission: decision }, reason);\n}\n\n/** Windsurf: uses boolean 'allowed' field */\nfunction formatWindsurf(decision: string, reason?: string): Record<string, unknown> {\n  return withReason({ allowed: decision === 'allow' }, reason);\n}\n\n/** Codex PreToolUse currently supports block/deny only; allow is empty stdout. */\nfunction formatCodex(decision: string, reason?: string): Record<string, unknown> {\n  if (decision === 'allow') {\n    return {};\n  }\n\n  return {\n    hookSpecificOutput: {\n      hookEventName: 'PreToolUse',\n      permissionDecision: decision === 'ask' ? 'deny' : decision,\n      ...(reason ? { permissionDecisionReason: reason } : {}),\n    },\n  };\n}\n\n/** Claude Code (default): uses hookSpecificOutput.permissionDecision for PreToolUse */\nfunction formatClaudeCode(decision: string, reason?: string): Record<string, unknown> {\n  return {\n    hookSpecificOutput: withReason({\n      hookEventName: 'PreToolUse',\n      permissionDecision: decision,\n    }, reason, 'permissionDecisionReason'),\n  };\n}\n\n/** Platform formatter lookup */\nconst platformFormatters: Record<string, (decision: string, reason?: string) => Record<string, unknown>> = {\n  gemini: formatGemini,\n  cursor: formatCursor,\n  windsurf: formatWindsurf,\n  codex: formatCodex,\n  vscode: formatClaudeCode,\n  claude_code: formatClaudeCode,\n};\n\n/** Known platform identifiers */\nexport const SUPPORTED_PLATFORMS = Object.keys(platformFormatters);\n\nfunction warnOnUnknownPlatform(platform: string): void {\n  void import('../../utils/logger.js')\n    .then(({ logger }) => {\n      logger.warn(\n        `[evaluatePermission] Unknown platform \"${platform}\", defaulting to claude_code format. Supported: ${SUPPORTED_PLATFORMS.join(', ')}`,\n      );\n    })\n    .catch((error) => {\n      console.warn(\n        `[evaluatePermission] Failed to load logger while handling unknown platform \"${platform}\".`,\n        error,\n      );\n    });\n}\n\n/**\n * Format permission evaluation response for platform-specific hook scripts.\n * Each platform expects a different JSON shape from its hook response.\n *\n * Unknown platforms default to claude_code format with a warning log.\n */\nexport function formatPermissionResponse(\n  decision: 'allow' | 'deny' | 'ask',\n  platform: string,\n  _input: Record<string, unknown>,\n  reason?: string,\n): Record<string, unknown> {\n  switch (platform) {\n    case 'gemini': return formatGemini(decision, reason);\n    case 'cursor': return formatCursor(decision, reason);\n    case 'windsurf': return formatWindsurf(decision, reason);\n    case 'codex': return formatCodex(decision, reason);\n    case 'claude_code': return formatClaudeCode(decision, reason);\n    default:\n      // Import lazily to avoid circular dependency at module load time\n      warnOnUnknownPlatform(platform);\n      return formatClaudeCode(decision, reason);\n  }\n}\n\n/**\n * Evaluate a CLI permission request for PreToolUse hooks.\n *\n * @param params - Tool name, input, and target platform\n * @param deps - Injected dependencies from MCPAQLHandler\n * @returns Platform-formatted permission decision\n */\nexport async function evaluatePermission(\n  params: { tool_name?: unknown; input?: unknown; platform?: unknown; session_id?: unknown },\n  deps: EvaluatePermissionDeps,\n): Promise<Record<string, unknown>> {\n  const toolName = typeof params.tool_name === 'string' ? params.tool_name : '';\n  const inputRaw = params.input;\n  const input = (inputRaw && typeof inputRaw === 'object')\n    ? inputRaw as Record<string, unknown>\n    : {};\n  const platform = typeof params.platform === 'string' ? params.platform : 'claude_code';\n  const sessionId = typeof params.session_id === 'string' && params.session_id.trim() !== ''\n    ? params.session_id\n    : undefined;\n\n  // Rate limit\n  const rateStatus = deps.permissionPromptLimiter.checkLimit();\n  if (!rateStatus.allowed) {\n    return formatPermissionResponse('deny', platform, input, 'Rate limit exceeded');\n  }\n  deps.permissionPromptLimiter.consumeToken();\n\n  // Stage 1: Static classification\n  const classification = deps.classifyTool(toolName, input);\n  if (classification.behavior === 'allow') {\n    return formatPermissionResponse('allow', platform, input);\n  }\n  if (classification.behavior === 'deny') {\n    return formatPermissionResponse('deny', platform, input, classification.reason);\n  }\n\n  // Stage 2: Element policy evaluation\n  let elements: ActiveElement[];\n  try {\n    elements = await deps.getActiveElements(sessionId);\n  } catch (err) {\n    throw new PermissionEvaluationError(\n      `Failed to fetch active elements for policy evaluation: ${err instanceof Error ? err.message : String(err)}`,\n      'element_fetch', toolName, err,\n    );\n  }\n  const decision = deps.evaluateCliToolPolicy(toolName, input, elements);\n\n  if (decision.behavior === 'deny') {\n    return formatPermissionResponse('deny', platform, input, decision.message);\n  }\n  if (decision.behavior === 'confirm') {\n    return formatPermissionResponse('ask', platform, input,\n      decision.message || 'Requires confirmation per element policy');\n  }\n\n  // Default: allow\n  return formatPermissionResponse('allow', platform, input);\n}\n"]}
@@ -183,6 +183,13 @@ Quick start examples:
183
183
  { operation: "create_element", element_type: "memory", params: { element_name: "session-notes", description: "Session context and notes" } }
184
184
  { operation: "addEntry", params: { element_name: "session-notes", content: "Remember this fact", tags: ["important"] } }
185
185
  { operation: "execute_agent", params: { element_name: "MyAgent", parameters: { objective: "Review code" } } }
186
+ { operation: "record_execution_step", params: { element_name: "MyAgent", stepDescription: "Reviewed auth module", outcome: "success", findings: "Found 2 issues" } }
187
+ { operation: "complete_execution", params: { element_name: "MyAgent", outcome: "success", summary: "Review complete" } }
188
+
189
+ Execution loop: call execute_agent once to start, then record_execution_step after
190
+ each work chunk, then complete_execution when done. Use continue_execution only to
191
+ resume a paused execution, and pass the same goal parameters you used for
192
+ execute_agent when resuming.
186
193
 
187
194
  Gatekeeper: Some operations may return a confirmation prompt instead of executing immediately. Use confirm_operation to approve, then retry.
188
195
 
@@ -249,6 +256,7 @@ Note: addEntry content supports markdown (headers, lists, bold, tables, code blo
249
256
 
250
257
  Execution lifecycle — record agent progress (appends step records, like addEntry):
251
258
  { operation: "record_execution_step", params: { element_name: "code-reviewer", stepDescription: "Analyzed files", outcome: "success", findings: "Found 3 issues" } }
259
+ This is the normal next lifecycle call after mcp_aql_execute { operation: "execute_agent", ... }.
252
260
  Response flow: record_execution_step returns { autonomy: { continue, factors, notifications? } }. Check autonomy.continue to decide whether to proceed. Check autonomy.notifications for permission_pending (gatekeeper blocks), autonomy_pause, or danger_zone alerts to relay to human operators.
253
261
 
254
262
  Import & portfolio:
@@ -330,6 +338,7 @@ Memory-specific search (filter by tags):
330
338
  Execution lifecycle — read-only queries:
331
339
  { operation: "get_execution_state", params: { element_name: "code-reviewer" } }
332
340
  { operation: "get_gathered_data", params: { element_name: "code-reviewer", goalId: "goal-id" } }
341
+ For execution-state reads, reuse the same element_name you passed to execute_agent. If element_name is missing, retry with the same agent name rather than inventing a new one.
333
342
 
334
343
  Collection:
335
344
  { operation: "browse_collection", params: { section: "personas" } }
@@ -484,8 +493,8 @@ Supported operations: ${getOperationsString('EXECUTE')}
484
493
 
485
494
  These operations manage runtime execution state. Unlike CRUD operations (which manage definitions), Execute operations handle the execution lifecycle:
486
495
  - execute_agent: Start a new execution (returns goalId and stateVersion for tracking)
487
- - complete_execution: Signal successful completion
488
- - continue_execution: Resume from saved state
496
+ - complete_execution: Signal successful completion once the goal is done
497
+ - continue_execution: Resume a previously paused execution with the same goal parameters
489
498
  - abort_execution: Abort a running execution, rejecting further operations
490
499
  - confirm_operation: Confirm a pending operation that requires user approval (Gatekeeper flow)
491
500
  - approve_cli_permission: Approve a pending CLI tool permission request
@@ -496,13 +505,20 @@ IMPORTANT: Execute operations are potentially destructive (agents can perform an
496
505
 
497
506
  ⚠️ SECURITY: Do not auto-allow this endpoint in your host settings (e.g., Claude Code settings.json). Each execution should require explicit human approval. Auto-allowing bypasses the per-operation confirmation gate. While DangerZone verification and element deny policies still provide protection, the primary human review checkpoint is lost.
498
507
 
499
- Response flow: execute_agent returns { goalId, stateVersion, activeElements, safetyTier, ... }. Use goalId with record_execution_step and complete_execution. stateVersion enables optimistic locking. record_execution_step returns { autonomy: { continue, notifications? } } — check notifications for gatekeeper blocks and danger zone alerts.
508
+ Canonical loop:
509
+ 1. Call execute_agent once to start the goal and receive { goalId, stateVersion, activeElements, safetyTier, ... }.
510
+ 2. After each chunk of work, use mcp_aql_create: { operation: "record_execution_step", ... }.
511
+ 3. Read record_execution_step.autonomy.continue and any autonomy.notifications to decide whether to continue, pause for a human, or handle a gatekeeper block.
512
+ 4. When the goal is finished, call complete_execution.
513
+ Use continue_execution only when an already-started goal was paused and you are resuming it with the same goal parameters. It is not the normal next call after execute_agent.
500
514
 
501
515
  Quick start examples:
502
516
  { operation: "execute_agent", params: { element_name: "code-reviewer", parameters: { objective: "Review code" } } }
517
+ Next lifecycle step — use mcp_aql_create:
518
+ { operation: "record_execution_step", params: { element_name: "code-reviewer", stepDescription: "Reviewed auth module", outcome: "success", findings: "Found 2 security issues" } }
503
519
  { operation: "complete_execution", params: { element_name: "code-reviewer", outcome: "success", summary: "Completed review" } }
504
520
  { operation: "abort_execution", params: { element_name: "data-collector", reason: "User requested cancellation" } }
505
- { operation: "continue_execution", params: { element_name: "code-reviewer" } }
521
+ { operation: "continue_execution", params: { element_name: "rubric-qa-agent", previousStepResult: "Verified citation set", parameters: { run_dir: "/app/run", deliverable_path: "/app/run/output.docx" } } }
506
522
  { operation: "confirm_operation", params: { operation: "execute_agent" } }
507
523
  { operation: "approve_cli_permission", params: { request_id: "req-123", decision: "allow" } }
508
524
  { operation: "prepare_handoff", params: { element_name: "code-reviewer" } }
@@ -530,4 +546,4 @@ Discover required parameters — use mcp_aql_read:
530
546
  }
531
547
  ];
532
548
  }
533
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"MCPAQLTools.js","sourceRoot":"","sources":["../../../src/server/tools/MCPAQLTools.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * MCP-AQL Tools - Unified query interface for AI agents\n *\n * Provides two endpoint modes configurable via MCP_AQL_ENDPOINT_MODE environment variable:\n *\n * ## Mode 1: CRUDE Endpoints (Default) - MCP_AQL_ENDPOINT_MODE=crude\n * 5 tools: mcp_aql_create, mcp_aql_read, mcp_aql_update, mcp_aql_delete, mcp_aql_execute (~4,300 tokens)\n *\n * ## Mode 2: Single Endpoint (Minimal) - MCP_AQL_ENDPOINT_MODE=single\n * 1 tool: mcp_aql (~350 tokens)\n * Ideal for multi-server deployments where token budget is constrained.\n *\n * Note: These tools are only registered when MCP_INTERFACE_MODE=mcpaql (default).\n * When MCP_INTERFACE_MODE=discrete, discrete tools are registered instead.\n *\n * ## Why 5 CRUDE Endpoints? (Default Mode)\n *\n * The 5-endpoint CRUDE pattern (Create, Read, Update, Delete, Execute) was chosen for:\n *\n * 1. **User Comprehension**: CRUDE extends CRUD with Execute for non-idempotent\n *    operations, making it easy for users to reason about what each endpoint can do.\n *\n * 2. **Platform Annotations**: MCP platforms like ChatGPT's apps require\n *    tool annotations that describe safety and destructiveness. The 5-endpoint\n *    split maps directly to distinct permission levels:\n *    - CREATE: additive, non-destructive (readOnlyHint: false, destructiveHint: false)\n *    - READ: safe, read-only (readOnlyHint: true, destructiveHint: false)\n *    - UPDATE: modifying, potentially destructive (readOnlyHint: false, destructiveHint: true)\n *    - DELETE: destructive (readOnlyHint: false, destructiveHint: true)\n *    - EXECUTE: non-idempotent, potentially destructive (readOnlyHint: false, destructiveHint: true)\n *\n * 3. **Granular Permission Control**: Users can grant READ endpoint full access\n *    while locking down CREATE, UPDATE, DELETE, and EXECUTE. This enables safe\n *    read-only integrations without exposing mutation capabilities.\n *\n * ## Why Single Endpoint? (Minimal Mode)\n *\n * The single-endpoint mode was added for:\n *\n * 1. **Token Efficiency**: ~350 tokens vs ~4,300 tokens (92% reduction)\n * 2. **Multi-Server Deployments**: When running multiple MCP servers, token\n *    budgets can be constrained. Single endpoint reduces overhead.\n * 3. **Simplified Integration**: Some clients prefer a single entry point.\n *\n * Security is maintained through server-side Gatekeeper enforcement - the\n * server determines which operation type is being executed and applies\n * appropriate permission checks.\n *\n * ## GraphQL-Style Introspection\n *\n * MCP-AQL follows GraphQL patterns for self-documentation. Tool descriptions\n * are generated dynamically from OPERATION_ROUTES, ensuring they always reflect\n * the current available operations.\n *\n * ### Introspection Examples\n *\n * List all available operations:\n * ```json\n * { \"operation\": \"introspect\", \"params\": { \"query\": \"operations\" } }\n * ```\n *\n * Get details for a specific operation (parameters, examples, return types):\n * ```json\n * { \"operation\": \"introspect\", \"params\": { \"query\": \"operations\", \"name\": \"create_element\" } }\n * ```\n *\n * Discover available types (e.g., ElementType enum values):\n * ```json\n * { \"operation\": \"introspect\", \"params\": { \"query\": \"types\", \"name\": \"ElementType\" } }\n * ```\n *\n * ### Design Philosophy\n *\n * - **Single source of truth**: OPERATION_ROUTES defines all operations\n * - **Dynamic descriptions**: Adding an operation to OPERATION_ROUTES automatically\n *   updates tool descriptions - no manual synchronization needed\n * - **Introspection-first**: LLMs discover parameters via introspect, not static docs\n * - **GraphQL heritage**: Patterns familiar to any LLM trained on GraphQL\n */\n\nimport type { MCPAQLHandler } from '../../handlers/mcp-aql/MCPAQLHandler.js';\nimport { UnifiedEndpoint } from '../../handlers/mcp-aql/UnifiedEndpoint.js';\nimport { getOperationsForEndpoint, type CRUDEndpoint } from '../../handlers/mcp-aql/OperationRouter.js';\nimport { ElementType } from '../../handlers/mcp-aql/types.js';\nimport type { ToolDefinition, ToolHandler } from '../../handlers/types/ToolTypes.js';\nimport { env } from '../../config/env.js';\nimport { ELEMENT_ROLES } from '../../elements/ensembles/constants.js';\n\n// ============================================================================\n// Dynamic Description Generation\n// ============================================================================\n\n/**\n * Get element types as a comma-separated string.\n * Derived from the ElementType enum to ensure consistency.\n */\nfunction getElementTypesString(): string {\n  return Object.values(ElementType).join(', ');\n}\n\n/**\n * Get operations for an endpoint as a comma-separated string.\n * Derived from OPERATION_ROUTES to ensure consistency.\n */\nfunction getOperationsString(endpoint: CRUDEndpoint): string {\n  return getOperationsForEndpoint(endpoint).join(', ');\n}\n\n// ============================================================================\n// Tool Schema\n// ============================================================================\n\n/**\n * Shared input schema for CRUD operations (create, read, update, delete)\n * All operations use the OperationInput structure from the GraphQL schema\n */\nconst operationInputSchema = {\n  type: \"object\" as const,\n  properties: {\n    operation: {\n      type: \"string\",\n      description: \"Operation name to execute\"\n    },\n    element_type: {\n      type: \"string\",\n      description: \"Target element type (optional)\"\n    },\n    params: {\n      type: \"object\",\n      description: \"Operation parameters\"\n    },\n    operations: {\n      type: \"array\",\n      description: \"Array of operations for batch execution\",\n      items: {\n        type: \"object\",\n        properties: {\n          operation: { type: \"string\" },\n          element_type: { type: \"string\" },\n          params: { type: \"object\" }\n        },\n        required: [\"operation\"]\n      }\n    }\n  },\n  required: [\"operation\" as const]\n};\n\n// ============================================================================\n// Tool Registration\n// ============================================================================\n\n/**\n * Get MCP-AQL tools for registration in the ToolRegistry.\n *\n * Returns different tools based on MCP_AQL_ENDPOINT_MODE environment variable:\n * - 'crude' (default): 5 CRUDE endpoints (Create, Read, Update, Delete, Execute) (~4,300 tokens)\n * - 'single': 1 unified endpoint (~350 tokens)\n *\n * Note: MCP_AQL_MODE is supported as a deprecated alias for backward compatibility.\n */\nexport function getMCPAQLTools(handler: MCPAQLHandler): Array<{ tool: ToolDefinition; handler: ToolHandler }> {\n  // Use MCP_AQL_ENDPOINT_MODE, falling back to deprecated MCP_AQL_MODE for backward compatibility\n  const mode = env.MCP_AQL_ENDPOINT_MODE ?? env.MCP_AQL_MODE ?? 'crude';\n\n  if (mode === 'single') {\n    return getUnifiedTools(handler);\n  }\n\n  // Default: CRUDE mode (5 endpoints)\n  return getCRUDETools(handler);\n}\n\n/**\n * Get the unified single endpoint tool (MCP_AQL_MODE=single)\n * Token footprint: ~300-400 tokens\n */\nfunction getUnifiedTools(handler: MCPAQLHandler): Array<{ tool: ToolDefinition; handler: ToolHandler }> {\n  const unifiedEndpoint = new UnifiedEndpoint(handler);\n\n  // Build dynamic description from OPERATION_ROUTES\n  const description = `DollhouseMCP unified API - GraphQL-style query interface for AI element management.\n\nCRUDE Operations:\n- CREATE: ${getOperationsString('CREATE')}\n- READ: ${getOperationsString('READ')}\n- UPDATE: ${getOperationsString('UPDATE')}\n- DELETE: ${getOperationsString('DELETE')}\n- EXECUTE: ${getOperationsString('EXECUTE')}\n\nElement types: ${getElementTypesString()}\n\nQuick start examples:\n{ operation: \"list_elements\", element_type: \"persona\" }\n{ operation: \"create_element\", element_type: \"persona\", params: { element_name: \"MyPersona\", description: \"A helpful assistant\", instructions: \"You ARE a helpful assistant. ALWAYS provide clear, accurate responses.\" } }\n{ operation: \"create_element\", element_type: \"agent\", params: { element_name: \"MyAgent\", description: \"Task executor\", instructions: \"Execute goals methodically. Report progress at each step.\", goal: { template: \"Complete: {objective}\", parameters: [{ name: \"objective\", type: \"string\", required: true }] } } }\n{ operation: \"create_element\", element_type: \"memory\", params: { element_name: \"session-notes\", description: \"Session context and notes\" } }\n{ operation: \"addEntry\", params: { element_name: \"session-notes\", content: \"Remember this fact\", tags: [\"important\"] } }\n{ operation: \"execute_agent\", params: { element_name: \"MyAgent\", parameters: { objective: \"Review code\" } } }\n\nGatekeeper: Some operations may return a confirmation prompt instead of executing immediately. Use confirm_operation to approve, then retry.\n\nDiscover all operations:\n{ operation: \"introspect\", params: { query: \"operations\" } }`;\n\n  return [\n    {\n      tool: {\n        name: \"mcp_aql\",\n        description,\n        inputSchema: operationInputSchema,\n        annotations: {\n          // Unified endpoint can perform any operation, so we use conservative hints\n          // The actual operation's safety is enforced server-side by Gatekeeper\n          readOnlyHint: false,\n          destructiveHint: true // Conservative: some operations are destructive\n        }\n      },\n      handler: async (args: any) => {\n        const result = await unifiedEndpoint.handle(args);\n        return {\n          content: [\n            {\n              type: \"text\",\n              text: JSON.stringify(result, null, 2)\n            }\n          ]\n        };\n      }\n    }\n  ];\n}\n\n/**\n * Get the 5 CRUDE endpoint tools (MCP_AQL_ENDPOINT_MODE=crude, default)\n * CRUDE = Create, Read, Update, Delete, Execute\n * Token footprint: ~4,300 tokens (measured via Claude Code /context)\n *\n * Descriptions are generated dynamically from OPERATION_ROUTES to ensure\n * they always reflect the current available operations.\n */\nfunction getCRUDETools(handler: MCPAQLHandler): Array<{ tool: ToolDefinition; handler: ToolHandler }> {\n  const elementTypes = getElementTypesString();\n\n  return [\n    // mcp_aql_create - Additive, non-destructive operations\n    {\n      tool: {\n        name: \"mcp_aql_create\",\n        description: `Additive, non-destructive operations.\n\nSupported operations: ${getOperationsString('CREATE')}\n\nElement types: ${elementTypes}\n\nThese operations add new data without removing or overwriting existing content.\n\nQuick start examples:\n{ operation: \"create_element\", element_type: \"persona\", params: { element_name: \"MyPersona\", description: \"A helpful assistant\", instructions: \"You ARE a helpful assistant. ALWAYS provide clear, accurate responses.\" } }\n{ operation: \"create_element\", element_type: \"agent\", params: { element_name: \"MyAgent\", description: \"Task executor\", instructions: \"Execute goals methodically. Report progress at each step.\", goal: { template: \"Complete: {objective}\", parameters: [{ name: \"objective\", type: \"string\", required: true }] } } }\n{ operation: \"create_element\", element_type: \"memory\", params: { element_name: \"session-notes\", description: \"Session context and notes\" } }\n{ operation: \"create_element\", element_type: \"ensemble\", params: { element_name: \"my-ensemble\", description: \"Combined element set\", metadata: { elements: [{ element_name: \"expert\", element_type: \"persona\", role: \"primary\" }, { element_name: \"analysis\", element_type: \"skill\", role: \"support\" }] } } }\nValid ensemble roles: ${ELEMENT_ROLES.join(', ')}\n{ operation: \"addEntry\", params: { element_name: \"session-notes\", content: \"Remember this fact\", tags: [\"important\"] } }\nNote: addEntry content supports markdown (headers, lists, bold, tables, code blocks). Ensure markdown content is properly JSON-escaped — use ${String.raw`\\n`} for newlines, ${String.raw`\\\"`} for quotes, and ${String.raw`\\\\`} for backslashes within the JSON string value.\n\nExecution lifecycle — record agent progress (appends step records, like addEntry):\n{ operation: \"record_execution_step\", params: { element_name: \"code-reviewer\", stepDescription: \"Analyzed files\", outcome: \"success\", findings: \"Found 3 issues\" } }\nResponse flow: record_execution_step returns { autonomy: { continue, factors, notifications? } }. Check autonomy.continue to decide whether to proceed. Check autonomy.notifications for permission_pending (gatekeeper blocks), autonomy_pause, or danger_zone alerts to relay to human operators.\n\nImport & portfolio:\n{ operation: \"import_element\", element_type: \"skill\", params: { element_name: \"code-formatter\", data: \"...\" } }\n{ operation: \"import_persona\", params: { source: \"/path/to/persona.md\" } }\n{ operation: \"install_collection_content\", params: { element_type: \"persona\", element_name: \"Creative-Writer\" } }\n{ operation: \"submit_collection_content\", params: { element_type: \"skill\", element_name: \"code-formatter\" } }\n{ operation: \"init_portfolio\" }\n{ operation: \"sync_portfolio\" }\n{ operation: \"portfolio_element_manager\", params: { action: \"push\", element_type: \"persona\", element_name: \"Tech-Writer\" } }\n\nAuth & verification:\n{ operation: \"setup_github_auth\" }\n{ operation: \"configure_oauth\", params: { client_id: \"your-client-id\" } }\n{ operation: \"verify_challenge\", params: { code: \"ABC123\" } }\n{ operation: \"release_deadlock\" }\n{ operation: \"beetlejuice_beetlejuice_beetlejuice\" }\n\nBatch operations: Use the operations array to execute multiple operations sequentially in a single request.\n{ operations: [{ operation: \"addEntry\", params: { element_name: \"log\", content: \"Step 1\" } }, { operation: \"addEntry\", params: { element_name: \"log\", content: \"Step 2\" } }] }\n\nDiscover required parameters — use mcp_aql_read:\n{ operation: \"introspect\", params: { query: \"operations\", name: \"create_element\" } }\nDiscover element format specs (required fields, syntax, examples) — use mcp_aql_read:\n{ operation: \"introspect\", params: { query: \"format\", name: \"template\" } }`,\n        inputSchema: operationInputSchema,\n        annotations: {\n          readOnlyHint: false,\n          destructiveHint: false\n        }\n      },\n      handler: async (args: any) => {\n        const result = await handler.handleCreate(args);\n        return {\n          content: [\n            {\n              type: \"text\",\n              text: JSON.stringify(result, null, 2)\n            }\n          ]\n        };\n      }\n    },\n\n    // mcp_aql_read - Safe, read-only operations\n    {\n      tool: {\n        name: \"mcp_aql_read\",\n        description: `Safe, read-only operations.\n\nSupported operations: ${getOperationsString('READ')}\n\nElement types: ${elementTypes}\n\nThese queries only read data and never modify server state.\n\nQuick start examples:\n{ operation: \"list_elements\", element_type: \"persona\" }\n{ operation: \"get_active_elements\", element_type: \"persona\" }\n{ operation: \"search_elements\", params: { query: \"creative\" } }\n{ operation: \"get_element\", element_type: \"memory\", params: { element_name: \"session-notes\" } }\n\nElement operations:\n{ operation: \"activate_element\", element_type: \"persona\", params: { element_name: \"Default\" } }\n{ operation: \"deactivate_element\", element_type: \"persona\", params: { element_name: \"Default\" } }\n{ operation: \"get_element_details\", element_type: \"skill\", params: { element_name: \"code-review\" } }\n{ operation: \"query_elements\", element_type: \"persona\", params: { filters: { category: \"creative\" } } }\n{ operation: \"validate_element\", element_type: \"agent\", params: { element_name: \"task-planner\" } }\n{ operation: \"render\", params: { element_name: \"meeting-notes\", variables: { date: \"2026-03-03\" } } }\n{ operation: \"export_element\", element_type: \"persona\", params: { element_name: \"Tech-Writer\" } }\n{ operation: \"open_portfolio_browser\" }\n{ operation: \"open_logs\" }\n{ operation: \"open_metrics\" }\n{ operation: \"open_permissions\" }\n{ operation: \"open_setup\" }\n\nMemory-specific search (filter by tags):\n{ operation: \"search\", params: { query: \"*\", type: \"memory\", filters: { tags: [\"important\"] } } }\n\nExecution lifecycle — read-only queries:\n{ operation: \"get_execution_state\", params: { element_name: \"code-reviewer\" } }\n{ operation: \"get_gathered_data\", params: { element_name: \"code-reviewer\", goalId: \"goal-id\" } }\n\nCollection:\n{ operation: \"browse_collection\", params: { section: \"personas\" } }\n{ operation: \"search_collection\", params: { query: \"creative\" } }\n{ operation: \"search_collection_enhanced\", params: { query: \"creative\", page: 1 } }\n{ operation: \"get_collection_content\", params: { element_type: \"persona\", element_name: \"Creative-Writer\" } }\n{ operation: \"get_collection_cache_health\" }\n\nPortfolio:\n{ operation: \"portfolio_status\" }\n{ operation: \"portfolio_config\" }\n{ operation: \"search_portfolio\", params: { query: \"creative\" } }\n{ operation: \"search_all\", params: { query: \"creative\" } }\n\nSystem:\n{ operation: \"dollhouse_config\" }\n{ operation: \"get_build_info\" }\n{ operation: \"get_cache_budget_report\" }\n{ operation: \"query_logs\", params: { level: \"error\", limit: 10 } }\n{ operation: \"query_metrics\" }\n{ operation: \"query_metrics\", params: { names: [\"system.memory.*\"], type: \"gauge\" } }\n{ operation: \"convert_skill_format\", params: { direction: \"agent_to_dollhouse\", agent_skill: { \"SKILL.md\": \"---\\\\nname: my-skill\\\\ndescription: test\\\\n---\\\\n\\\\nUse this skill.\" } } }\n{ operation: \"convert_skill_format\", params: { direction: \"agent_to_dollhouse\", security_mode: \"warn\", path_mode: \"lossless\", agent_skill: { \"SKILL.md\": \"---\\\\nname: my-skill\\\\ndescription: test\\\\n---\\\\n\\\\nUse this skill.\" } } }\n{ operation: \"convert_skill_format\", params: { direction: \"dollhouse_to_agent\", path_mode: \"lossless\", dollhouse_markdown: \"---\\\\nname: my-skill\\\\ndescription: test\\\\ninstructions: Use this skill.\\\\n---\\\\n\\\\n### binaries/logo.png\\\\n(binary link: ./skills/binaries/logo.png)\" } }\n\nAuth:\n{ operation: \"check_github_auth\" }\n{ operation: \"oauth_helper_status\" }\n\nGatekeeper & CLI policies:\n{ operation: \"permission_prompt\", params: { tool: \"Bash\", prompt: \"run npm test\" } }\n{ operation: \"evaluate_permission\", params: { tool_name: \"Bash\", input: { command: \"git status\" }, platform: \"claude_code\" } }\n{ operation: \"get_effective_cli_policies\" }\n{ operation: \"get_pending_cli_approvals\" }\n{ operation: \"get_permission_authority\" }\n{ operation: \"get_permission_authority\", params: { host: \"claude-code\" } }\n\nEnhanced index:\n{ operation: \"find_similar_elements\", params: { element_type: \"persona\", element_name: \"Creative-Writer\" } }\n{ operation: \"get_element_relationships\", params: { element_type: \"skill\", element_name: \"code-review\" } }\n{ operation: \"search_by_verb\", params: { verb: \"review\" } }\n{ operation: \"get_relationship_stats\" }\n\nDiscover all operations and parameters:\n{ operation: \"get_capabilities\" }\n{ operation: \"get_capabilities\", params: { category: \"Element Lifecycle\" } }\n{ operation: \"introspect\", params: { query: \"operations\" } }`,\n        inputSchema: operationInputSchema,\n        annotations: {\n          readOnlyHint: true,\n          destructiveHint: false\n        }\n      },\n      handler: async (args: any) => {\n        const result = await handler.handleRead(args);\n        return {\n          content: [\n            {\n              type: \"text\",\n              text: JSON.stringify(result, null, 2)\n            }\n          ]\n        };\n      }\n    },\n\n    // mcp_aql_update - Modifying operations that overwrite data\n    {\n      tool: {\n        name: \"mcp_aql_update\",\n        description: `Modifying operations that overwrite data.\n\nSupported operations: ${getOperationsString('UPDATE')}\n\nElement types: ${elementTypes}\n\nThese operations modify existing data, potentially overwriting previous values.\n\nNote: Memories are append-only and do not support edit_element. Use addEntry (CREATE) to add new entries.\n\nQuick start example:\n{ operation: \"edit_element\", element_type: \"persona\", params: { element_name: \"MyPersona\", input: { description: \"Updated description\" } } }\n{ operation: \"edit_element\", element_type: \"persona\", params: { element_name: \"Friendly-Teacher\", input: { instructions: \"Updated behavioral directives.\" } } }\n{ operation: \"edit_element\", element_type: \"agent\", params: { element_name: \"code-reviewer\", input: { instructions: \"Updated agent behavioral profile.\", goal: { template: \"Complete: {task}\" } } } }\n{ operation: \"upgrade_element\", element_type: \"agent\", params: { element_name: \"task-planner\" } }\n\nDiscover required parameters — use mcp_aql_read:\n{ operation: \"introspect\", params: { query: \"operations\", name: \"edit_element\" } }`,\n        inputSchema: operationInputSchema,\n        annotations: {\n          readOnlyHint: false,\n          destructiveHint: true\n        }\n      },\n      handler: async (args: any) => {\n        const result = await handler.handleUpdate(args);\n        return {\n          content: [\n            {\n              type: \"text\",\n              text: JSON.stringify(result, null, 2)\n            }\n          ]\n        };\n      }\n    },\n\n    // mcp_aql_delete - Destructive operations that remove data\n    {\n      tool: {\n        name: \"mcp_aql_delete\",\n        description: `Destructive operations that remove data.\n\nSupported operations: ${getOperationsString('DELETE')}\n\nElement types: ${elementTypes}\n\nThese operations remove data. Use with caution.\n\n⚠️ SECURITY: Do not auto-allow this endpoint in your host settings (e.g., Claude Code settings.json). Each delete operation should require explicit human approval. Auto-allowing bypasses the per-operation confirmation gate, leaving only element deny policies as protection against unintended data loss.\n\nQuick start examples:\n{ operation: \"delete_element\", element_type: \"persona\", params: { element_name: \"Old-Persona\" } }\n{ operation: \"clear\", params: { element_name: \"temp-notes\" } }\n{ operation: \"clear_github_auth\" }\n\nDiscover required parameters — use mcp_aql_read:\n{ operation: \"introspect\", params: { query: \"operations\", name: \"delete_element\" } }`,\n        inputSchema: operationInputSchema,\n        annotations: {\n          readOnlyHint: false,\n          destructiveHint: true\n        }\n      },\n      handler: async (args: any) => {\n        const result = await handler.handleDelete(args);\n        return {\n          content: [\n            {\n              type: \"text\",\n              text: JSON.stringify(result, null, 2)\n            }\n          ]\n        };\n      }\n    },\n\n    // mcp_aql_execute - Execution lifecycle operations (CRUDE's 'E')\n    {\n      tool: {\n        name: \"mcp_aql_execute\",\n        description: `Execution lifecycle operations for executable elements (agents, workflows, pipelines).\n\nSupported operations: ${getOperationsString('EXECUTE')}\n\nThese operations manage runtime execution state. Unlike CRUD operations (which manage definitions), Execute operations handle the execution lifecycle:\n- execute_agent: Start a new execution (returns goalId and stateVersion for tracking)\n- complete_execution: Signal successful completion\n- continue_execution: Resume from saved state\n- abort_execution: Abort a running execution, rejecting further operations\n- confirm_operation: Confirm a pending operation that requires user approval (Gatekeeper flow)\n- approve_cli_permission: Approve a pending CLI tool permission request\n- prepare_handoff: Serialize goal progress into a portable handoff block for session transfer\n- resume_from_handoff: Resume agent execution from a handoff block with integrity validation\n\nIMPORTANT: Execute operations are potentially destructive (agents can perform any action) and non-idempotent (calling execute_agent twice creates two separate executions).\n\n⚠️ SECURITY: Do not auto-allow this endpoint in your host settings (e.g., Claude Code settings.json). Each execution should require explicit human approval. Auto-allowing bypasses the per-operation confirmation gate. While DangerZone verification and element deny policies still provide protection, the primary human review checkpoint is lost.\n\nResponse flow: execute_agent returns { goalId, stateVersion, activeElements, safetyTier, ... }. Use goalId with record_execution_step and complete_execution. stateVersion enables optimistic locking. record_execution_step returns { autonomy: { continue, notifications? } } — check notifications for gatekeeper blocks and danger zone alerts.\n\nQuick start examples:\n{ operation: \"execute_agent\", params: { element_name: \"code-reviewer\", parameters: { objective: \"Review code\" } } }\n{ operation: \"complete_execution\", params: { element_name: \"code-reviewer\", outcome: \"success\", summary: \"Completed review\" } }\n{ operation: \"abort_execution\", params: { element_name: \"data-collector\", reason: \"User requested cancellation\" } }\n{ operation: \"continue_execution\", params: { element_name: \"code-reviewer\" } }\n{ operation: \"confirm_operation\", params: { operation: \"execute_agent\" } }\n{ operation: \"approve_cli_permission\", params: { request_id: \"req-123\", decision: \"allow\" } }\n{ operation: \"prepare_handoff\", params: { element_name: \"code-reviewer\" } }\n{ operation: \"resume_from_handoff\", params: { element_name: \"code-reviewer\", handoff_block: \"...\" } }\n\nDiscover required parameters — use mcp_aql_read:\n{ operation: \"introspect\", params: { query: \"operations\", name: \"execute_agent\" } }`,\n        inputSchema: operationInputSchema,\n        annotations: {\n          readOnlyHint: false,\n          destructiveHint: true  // Potentially destructive - agents can perform any action\n        }\n      },\n      handler: async (args: any) => {\n        const result = await handler.handleExecute(args);\n        return {\n          content: [\n            {\n              type: \"text\",\n              text: JSON.stringify(result, null, 2)\n            }\n          ]\n        };\n      }\n    }\n  ];\n}\n"]}
549
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"MCPAQLTools.js","sourceRoot":"","sources":["../../../src/server/tools/MCPAQLTools.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA8EG;AAGH,OAAO,EAAE,eAAe,EAAE,MAAM,2CAA2C,CAAC;AAC5E,OAAO,EAAE,wBAAwB,EAAqB,MAAM,2CAA2C,CAAC;AACxG,OAAO,EAAE,WAAW,EAAE,MAAM,iCAAiC,CAAC;AAE9D,OAAO,EAAE,GAAG,EAAE,MAAM,qBAAqB,CAAC;AAC1C,OAAO,EAAE,aAAa,EAAE,MAAM,uCAAuC,CAAC;AAEtE,+EAA+E;AAC/E,iCAAiC;AACjC,+EAA+E;AAE/E;;;GAGG;AACH,SAAS,qBAAqB;IAC5B,OAAO,MAAM,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;AAC/C,CAAC;AAED;;;GAGG;AACH,SAAS,mBAAmB,CAAC,QAAsB;IACjD,OAAO,wBAAwB,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;AACvD,CAAC;AAED,+EAA+E;AAC/E,cAAc;AACd,+EAA+E;AAE/E;;;GAGG;AACH,MAAM,oBAAoB,GAAG;IAC3B,IAAI,EAAE,QAAiB;IACvB,UAAU,EAAE;QACV,SAAS,EAAE;YACT,IAAI,EAAE,QAAQ;YACd,WAAW,EAAE,2BAA2B;SACzC;QACD,YAAY,EAAE;YACZ,IAAI,EAAE,QAAQ;YACd,WAAW,EAAE,gCAAgC;SAC9C;QACD,MAAM,EAAE;YACN,IAAI,EAAE,QAAQ;YACd,WAAW,EAAE,sBAAsB;SACpC;QACD,UAAU,EAAE;YACV,IAAI,EAAE,OAAO;YACb,WAAW,EAAE,yCAAyC;YACtD,KAAK,EAAE;gBACL,IAAI,EAAE,QAAQ;gBACd,UAAU,EAAE;oBACV,SAAS,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE;oBAC7B,YAAY,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE;oBAChC,MAAM,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE;iBAC3B;gBACD,QAAQ,EAAE,CAAC,WAAW,CAAC;aACxB;SACF;KACF;IACD,QAAQ,EAAE,CAAC,WAAoB,CAAC;CACjC,CAAC;AAEF,+EAA+E;AAC/E,oBAAoB;AACpB,+EAA+E;AAE/E;;;;;;;;GAQG;AACH,MAAM,UAAU,cAAc,CAAC,OAAsB;IACnD,gGAAgG;IAChG,MAAM,IAAI,GAAG,GAAG,CAAC,qBAAqB,IAAI,GAAG,CAAC,YAAY,IAAI,OAAO,CAAC;IAEtE,IAAI,IAAI,KAAK,QAAQ,EAAE,CAAC;QACtB,OAAO,eAAe,CAAC,OAAO,CAAC,CAAC;IAClC,CAAC;IAED,oCAAoC;IACpC,OAAO,aAAa,CAAC,OAAO,CAAC,CAAC;AAChC,CAAC;AAED;;;GAGG;AACH,SAAS,eAAe,CAAC,OAAsB;IAC7C,MAAM,eAAe,GAAG,IAAI,eAAe,CAAC,OAAO,CAAC,CAAC;IAErD,kDAAkD;IAClD,MAAM,WAAW,GAAG;;;YAGV,mBAAmB,CAAC,QAAQ,CAAC;UAC/B,mBAAmB,CAAC,MAAM,CAAC;YACzB,mBAAmB,CAAC,QAAQ,CAAC;YAC7B,mBAAmB,CAAC,QAAQ,CAAC;aAC5B,mBAAmB,CAAC,SAAS,CAAC;;iBAE1B,qBAAqB,EAAE;;;;;;;;;;;;;;;;;;;;6DAoBqB,CAAC;IAE5D,OAAO;QACL;YACE,IAAI,EAAE;gBACJ,IAAI,EAAE,SAAS;gBACf,WAAW;gBACX,WAAW,EAAE,oBAAoB;gBACjC,WAAW,EAAE;oBACX,2EAA2E;oBAC3E,sEAAsE;oBACtE,YAAY,EAAE,KAAK;oBACnB,eAAe,EAAE,IAAI,CAAC,gDAAgD;iBACvE;aACF;YACD,OAAO,EAAE,KAAK,EAAE,IAAS,EAAE,EAAE;gBAC3B,MAAM,MAAM,GAAG,MAAM,eAAe,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAClD,OAAO;oBACL,OAAO,EAAE;wBACP;4BACE,IAAI,EAAE,MAAM;4BACZ,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC;yBACtC;qBACF;iBACF,CAAC;YACJ,CAAC;SACF;KACF,CAAC;AACJ,CAAC;AAED;;;;;;;GAOG;AACH,SAAS,aAAa,CAAC,OAAsB;IAC3C,MAAM,YAAY,GAAG,qBAAqB,EAAE,CAAC;IAE7C,OAAO;QACL,wDAAwD;QACxD;YACE,IAAI,EAAE;gBACJ,IAAI,EAAE,gBAAgB;gBACtB,WAAW,EAAE;;wBAEG,mBAAmB,CAAC,QAAQ,CAAC;;iBAEpC,YAAY;;;;;;;;;wBASL,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC;;+IAE+F,MAAM,CAAC,GAAG,CAAA,IAAI,kBAAkB,MAAM,CAAC,GAAG,CAAA,IAAI,oBAAoB,MAAM,CAAC,GAAG,CAAA,IAAI;;;;;;;;;;;;;;;;;;;;;;;;;;;;;2EA6BpJ;gBACnE,WAAW,EAAE,oBAAoB;gBACjC,WAAW,EAAE;oBACX,YAAY,EAAE,KAAK;oBACnB,eAAe,EAAE,KAAK;iBACvB;aACF;YACD,OAAO,EAAE,KAAK,EAAE,IAAS,EAAE,EAAE;gBAC3B,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC,YAAY,CAAC,IAAI,CAAC,CAAC;gBAChD,OAAO;oBACL,OAAO,EAAE;wBACP;4BACE,IAAI,EAAE,MAAM;4BACZ,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC;yBACtC;qBACF;iBACF,CAAC;YACJ,CAAC;SACF;QAED,4CAA4C;QAC5C;YACE,IAAI,EAAE;gBACJ,IAAI,EAAE,cAAc;gBACpB,WAAW,EAAE;;wBAEG,mBAAmB,CAAC,MAAM,CAAC;;iBAElC,YAAY;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;6DA6EgC;gBACrD,WAAW,EAAE,oBAAoB;gBACjC,WAAW,EAAE;oBACX,YAAY,EAAE,IAAI;oBAClB,eAAe,EAAE,KAAK;iBACvB;aACF;YACD,OAAO,EAAE,KAAK,EAAE,IAAS,EAAE,EAAE;gBAC3B,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;gBAC9C,OAAO;oBACL,OAAO,EAAE;wBACP;4BACE,IAAI,EAAE,MAAM;4BACZ,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC;yBACtC;qBACF;iBACF,CAAC;YACJ,CAAC;SACF;QAED,4DAA4D;QAC5D;YACE,IAAI,EAAE;gBACJ,IAAI,EAAE,gBAAgB;gBACtB,WAAW,EAAE;;wBAEG,mBAAmB,CAAC,QAAQ,CAAC;;iBAEpC,YAAY;;;;;;;;;;;;;mFAasD;gBAC3E,WAAW,EAAE,oBAAoB;gBACjC,WAAW,EAAE;oBACX,YAAY,EAAE,KAAK;oBACnB,eAAe,EAAE,IAAI;iBACtB;aACF;YACD,OAAO,EAAE,KAAK,EAAE,IAAS,EAAE,EAAE;gBAC3B,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC,YAAY,CAAC,IAAI,CAAC,CAAC;gBAChD,OAAO;oBACL,OAAO,EAAE;wBACP;4BACE,IAAI,EAAE,MAAM;4BACZ,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC;yBACtC;qBACF;iBACF,CAAC;YACJ,CAAC;SACF;QAED,2DAA2D;QAC3D;YACE,IAAI,EAAE;gBACJ,IAAI,EAAE,gBAAgB;gBACtB,WAAW,EAAE;;wBAEG,mBAAmB,CAAC,QAAQ,CAAC;;iBAEpC,YAAY;;;;;;;;;;;;qFAYwD;gBAC7E,WAAW,EAAE,oBAAoB;gBACjC,WAAW,EAAE;oBACX,YAAY,EAAE,KAAK;oBACnB,eAAe,EAAE,IAAI;iBACtB;aACF;YACD,OAAO,EAAE,KAAK,EAAE,IAAS,EAAE,EAAE;gBAC3B,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC,YAAY,CAAC,IAAI,CAAC,CAAC;gBAChD,OAAO;oBACL,OAAO,EAAE;wBACP;4BACE,IAAI,EAAE,MAAM;4BACZ,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC;yBACtC;qBACF;iBACF,CAAC;YACJ,CAAC;SACF;QAED,iEAAiE;QACjE;YACE,IAAI,EAAE;gBACJ,IAAI,EAAE,iBAAiB;gBACvB,WAAW,EAAE;;wBAEG,mBAAmB,CAAC,SAAS,CAAC;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;oFAoC8B;gBAC5E,WAAW,EAAE,oBAAoB;gBACjC,WAAW,EAAE;oBACX,YAAY,EAAE,KAAK;oBACnB,eAAe,EAAE,IAAI,CAAE,0DAA0D;iBAClF;aACF;YACD,OAAO,EAAE,KAAK,EAAE,IAAS,EAAE,EAAE;gBAC3B,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC;gBACjD,OAAO;oBACL,OAAO,EAAE;wBACP;4BACE,IAAI,EAAE,MAAM;4BACZ,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC;yBACtC;qBACF;iBACF,CAAC;YACJ,CAAC;SACF;KACF,CAAC;AACJ,CAAC","sourcesContent":["/**\n * MCP-AQL Tools - Unified query interface for AI agents\n *\n * Provides two endpoint modes configurable via MCP_AQL_ENDPOINT_MODE environment variable:\n *\n * ## Mode 1: CRUDE Endpoints (Default) - MCP_AQL_ENDPOINT_MODE=crude\n * 5 tools: mcp_aql_create, mcp_aql_read, mcp_aql_update, mcp_aql_delete, mcp_aql_execute (~4,300 tokens)\n *\n * ## Mode 2: Single Endpoint (Minimal) - MCP_AQL_ENDPOINT_MODE=single\n * 1 tool: mcp_aql (~350 tokens)\n * Ideal for multi-server deployments where token budget is constrained.\n *\n * Note: These tools are only registered when MCP_INTERFACE_MODE=mcpaql (default).\n * When MCP_INTERFACE_MODE=discrete, discrete tools are registered instead.\n *\n * ## Why 5 CRUDE Endpoints? (Default Mode)\n *\n * The 5-endpoint CRUDE pattern (Create, Read, Update, Delete, Execute) was chosen for:\n *\n * 1. **User Comprehension**: CRUDE extends CRUD with Execute for non-idempotent\n *    operations, making it easy for users to reason about what each endpoint can do.\n *\n * 2. **Platform Annotations**: MCP platforms like ChatGPT's apps require\n *    tool annotations that describe safety and destructiveness. The 5-endpoint\n *    split maps directly to distinct permission levels:\n *    - CREATE: additive, non-destructive (readOnlyHint: false, destructiveHint: false)\n *    - READ: safe, read-only (readOnlyHint: true, destructiveHint: false)\n *    - UPDATE: modifying, potentially destructive (readOnlyHint: false, destructiveHint: true)\n *    - DELETE: destructive (readOnlyHint: false, destructiveHint: true)\n *    - EXECUTE: non-idempotent, potentially destructive (readOnlyHint: false, destructiveHint: true)\n *\n * 3. **Granular Permission Control**: Users can grant READ endpoint full access\n *    while locking down CREATE, UPDATE, DELETE, and EXECUTE. This enables safe\n *    read-only integrations without exposing mutation capabilities.\n *\n * ## Why Single Endpoint? (Minimal Mode)\n *\n * The single-endpoint mode was added for:\n *\n * 1. **Token Efficiency**: ~350 tokens vs ~4,300 tokens (92% reduction)\n * 2. **Multi-Server Deployments**: When running multiple MCP servers, token\n *    budgets can be constrained. Single endpoint reduces overhead.\n * 3. **Simplified Integration**: Some clients prefer a single entry point.\n *\n * Security is maintained through server-side Gatekeeper enforcement - the\n * server determines which operation type is being executed and applies\n * appropriate permission checks.\n *\n * ## GraphQL-Style Introspection\n *\n * MCP-AQL follows GraphQL patterns for self-documentation. Tool descriptions\n * are generated dynamically from OPERATION_ROUTES, ensuring they always reflect\n * the current available operations.\n *\n * ### Introspection Examples\n *\n * List all available operations:\n * ```json\n * { \"operation\": \"introspect\", \"params\": { \"query\": \"operations\" } }\n * ```\n *\n * Get details for a specific operation (parameters, examples, return types):\n * ```json\n * { \"operation\": \"introspect\", \"params\": { \"query\": \"operations\", \"name\": \"create_element\" } }\n * ```\n *\n * Discover available types (e.g., ElementType enum values):\n * ```json\n * { \"operation\": \"introspect\", \"params\": { \"query\": \"types\", \"name\": \"ElementType\" } }\n * ```\n *\n * ### Design Philosophy\n *\n * - **Single source of truth**: OPERATION_ROUTES defines all operations\n * - **Dynamic descriptions**: Adding an operation to OPERATION_ROUTES automatically\n *   updates tool descriptions - no manual synchronization needed\n * - **Introspection-first**: LLMs discover parameters via introspect, not static docs\n * - **GraphQL heritage**: Patterns familiar to any LLM trained on GraphQL\n */\n\nimport type { MCPAQLHandler } from '../../handlers/mcp-aql/MCPAQLHandler.js';\nimport { UnifiedEndpoint } from '../../handlers/mcp-aql/UnifiedEndpoint.js';\nimport { getOperationsForEndpoint, type CRUDEndpoint } from '../../handlers/mcp-aql/OperationRouter.js';\nimport { ElementType } from '../../handlers/mcp-aql/types.js';\nimport type { ToolDefinition, ToolHandler } from '../../handlers/types/ToolTypes.js';\nimport { env } from '../../config/env.js';\nimport { ELEMENT_ROLES } from '../../elements/ensembles/constants.js';\n\n// ============================================================================\n// Dynamic Description Generation\n// ============================================================================\n\n/**\n * Get element types as a comma-separated string.\n * Derived from the ElementType enum to ensure consistency.\n */\nfunction getElementTypesString(): string {\n  return Object.values(ElementType).join(', ');\n}\n\n/**\n * Get operations for an endpoint as a comma-separated string.\n * Derived from OPERATION_ROUTES to ensure consistency.\n */\nfunction getOperationsString(endpoint: CRUDEndpoint): string {\n  return getOperationsForEndpoint(endpoint).join(', ');\n}\n\n// ============================================================================\n// Tool Schema\n// ============================================================================\n\n/**\n * Shared input schema for CRUD operations (create, read, update, delete)\n * All operations use the OperationInput structure from the GraphQL schema\n */\nconst operationInputSchema = {\n  type: \"object\" as const,\n  properties: {\n    operation: {\n      type: \"string\",\n      description: \"Operation name to execute\"\n    },\n    element_type: {\n      type: \"string\",\n      description: \"Target element type (optional)\"\n    },\n    params: {\n      type: \"object\",\n      description: \"Operation parameters\"\n    },\n    operations: {\n      type: \"array\",\n      description: \"Array of operations for batch execution\",\n      items: {\n        type: \"object\",\n        properties: {\n          operation: { type: \"string\" },\n          element_type: { type: \"string\" },\n          params: { type: \"object\" }\n        },\n        required: [\"operation\"]\n      }\n    }\n  },\n  required: [\"operation\" as const]\n};\n\n// ============================================================================\n// Tool Registration\n// ============================================================================\n\n/**\n * Get MCP-AQL tools for registration in the ToolRegistry.\n *\n * Returns different tools based on MCP_AQL_ENDPOINT_MODE environment variable:\n * - 'crude' (default): 5 CRUDE endpoints (Create, Read, Update, Delete, Execute) (~4,300 tokens)\n * - 'single': 1 unified endpoint (~350 tokens)\n *\n * Note: MCP_AQL_MODE is supported as a deprecated alias for backward compatibility.\n */\nexport function getMCPAQLTools(handler: MCPAQLHandler): Array<{ tool: ToolDefinition; handler: ToolHandler }> {\n  // Use MCP_AQL_ENDPOINT_MODE, falling back to deprecated MCP_AQL_MODE for backward compatibility\n  const mode = env.MCP_AQL_ENDPOINT_MODE ?? env.MCP_AQL_MODE ?? 'crude';\n\n  if (mode === 'single') {\n    return getUnifiedTools(handler);\n  }\n\n  // Default: CRUDE mode (5 endpoints)\n  return getCRUDETools(handler);\n}\n\n/**\n * Get the unified single endpoint tool (MCP_AQL_MODE=single)\n * Token footprint: ~300-400 tokens\n */\nfunction getUnifiedTools(handler: MCPAQLHandler): Array<{ tool: ToolDefinition; handler: ToolHandler }> {\n  const unifiedEndpoint = new UnifiedEndpoint(handler);\n\n  // Build dynamic description from OPERATION_ROUTES\n  const description = `DollhouseMCP unified API - GraphQL-style query interface for AI element management.\n\nCRUDE Operations:\n- CREATE: ${getOperationsString('CREATE')}\n- READ: ${getOperationsString('READ')}\n- UPDATE: ${getOperationsString('UPDATE')}\n- DELETE: ${getOperationsString('DELETE')}\n- EXECUTE: ${getOperationsString('EXECUTE')}\n\nElement types: ${getElementTypesString()}\n\nQuick start examples:\n{ operation: \"list_elements\", element_type: \"persona\" }\n{ operation: \"create_element\", element_type: \"persona\", params: { element_name: \"MyPersona\", description: \"A helpful assistant\", instructions: \"You ARE a helpful assistant. ALWAYS provide clear, accurate responses.\" } }\n{ operation: \"create_element\", element_type: \"agent\", params: { element_name: \"MyAgent\", description: \"Task executor\", instructions: \"Execute goals methodically. Report progress at each step.\", goal: { template: \"Complete: {objective}\", parameters: [{ name: \"objective\", type: \"string\", required: true }] } } }\n{ operation: \"create_element\", element_type: \"memory\", params: { element_name: \"session-notes\", description: \"Session context and notes\" } }\n{ operation: \"addEntry\", params: { element_name: \"session-notes\", content: \"Remember this fact\", tags: [\"important\"] } }\n{ operation: \"execute_agent\", params: { element_name: \"MyAgent\", parameters: { objective: \"Review code\" } } }\n{ operation: \"record_execution_step\", params: { element_name: \"MyAgent\", stepDescription: \"Reviewed auth module\", outcome: \"success\", findings: \"Found 2 issues\" } }\n{ operation: \"complete_execution\", params: { element_name: \"MyAgent\", outcome: \"success\", summary: \"Review complete\" } }\n\nExecution loop: call execute_agent once to start, then record_execution_step after\neach work chunk, then complete_execution when done. Use continue_execution only to\nresume a paused execution, and pass the same goal parameters you used for\nexecute_agent when resuming.\n\nGatekeeper: Some operations may return a confirmation prompt instead of executing immediately. Use confirm_operation to approve, then retry.\n\nDiscover all operations:\n{ operation: \"introspect\", params: { query: \"operations\" } }`;\n\n  return [\n    {\n      tool: {\n        name: \"mcp_aql\",\n        description,\n        inputSchema: operationInputSchema,\n        annotations: {\n          // Unified endpoint can perform any operation, so we use conservative hints\n          // The actual operation's safety is enforced server-side by Gatekeeper\n          readOnlyHint: false,\n          destructiveHint: true // Conservative: some operations are destructive\n        }\n      },\n      handler: async (args: any) => {\n        const result = await unifiedEndpoint.handle(args);\n        return {\n          content: [\n            {\n              type: \"text\",\n              text: JSON.stringify(result, null, 2)\n            }\n          ]\n        };\n      }\n    }\n  ];\n}\n\n/**\n * Get the 5 CRUDE endpoint tools (MCP_AQL_ENDPOINT_MODE=crude, default)\n * CRUDE = Create, Read, Update, Delete, Execute\n * Token footprint: ~4,300 tokens (measured via Claude Code /context)\n *\n * Descriptions are generated dynamically from OPERATION_ROUTES to ensure\n * they always reflect the current available operations.\n */\nfunction getCRUDETools(handler: MCPAQLHandler): Array<{ tool: ToolDefinition; handler: ToolHandler }> {\n  const elementTypes = getElementTypesString();\n\n  return [\n    // mcp_aql_create - Additive, non-destructive operations\n    {\n      tool: {\n        name: \"mcp_aql_create\",\n        description: `Additive, non-destructive operations.\n\nSupported operations: ${getOperationsString('CREATE')}\n\nElement types: ${elementTypes}\n\nThese operations add new data without removing or overwriting existing content.\n\nQuick start examples:\n{ operation: \"create_element\", element_type: \"persona\", params: { element_name: \"MyPersona\", description: \"A helpful assistant\", instructions: \"You ARE a helpful assistant. ALWAYS provide clear, accurate responses.\" } }\n{ operation: \"create_element\", element_type: \"agent\", params: { element_name: \"MyAgent\", description: \"Task executor\", instructions: \"Execute goals methodically. Report progress at each step.\", goal: { template: \"Complete: {objective}\", parameters: [{ name: \"objective\", type: \"string\", required: true }] } } }\n{ operation: \"create_element\", element_type: \"memory\", params: { element_name: \"session-notes\", description: \"Session context and notes\" } }\n{ operation: \"create_element\", element_type: \"ensemble\", params: { element_name: \"my-ensemble\", description: \"Combined element set\", metadata: { elements: [{ element_name: \"expert\", element_type: \"persona\", role: \"primary\" }, { element_name: \"analysis\", element_type: \"skill\", role: \"support\" }] } } }\nValid ensemble roles: ${ELEMENT_ROLES.join(', ')}\n{ operation: \"addEntry\", params: { element_name: \"session-notes\", content: \"Remember this fact\", tags: [\"important\"] } }\nNote: addEntry content supports markdown (headers, lists, bold, tables, code blocks). Ensure markdown content is properly JSON-escaped — use ${String.raw`\\n`} for newlines, ${String.raw`\\\"`} for quotes, and ${String.raw`\\\\`} for backslashes within the JSON string value.\n\nExecution lifecycle — record agent progress (appends step records, like addEntry):\n{ operation: \"record_execution_step\", params: { element_name: \"code-reviewer\", stepDescription: \"Analyzed files\", outcome: \"success\", findings: \"Found 3 issues\" } }\nThis is the normal next lifecycle call after mcp_aql_execute { operation: \"execute_agent\", ... }.\nResponse flow: record_execution_step returns { autonomy: { continue, factors, notifications? } }. Check autonomy.continue to decide whether to proceed. Check autonomy.notifications for permission_pending (gatekeeper blocks), autonomy_pause, or danger_zone alerts to relay to human operators.\n\nImport & portfolio:\n{ operation: \"import_element\", element_type: \"skill\", params: { element_name: \"code-formatter\", data: \"...\" } }\n{ operation: \"import_persona\", params: { source: \"/path/to/persona.md\" } }\n{ operation: \"install_collection_content\", params: { element_type: \"persona\", element_name: \"Creative-Writer\" } }\n{ operation: \"submit_collection_content\", params: { element_type: \"skill\", element_name: \"code-formatter\" } }\n{ operation: \"init_portfolio\" }\n{ operation: \"sync_portfolio\" }\n{ operation: \"portfolio_element_manager\", params: { action: \"push\", element_type: \"persona\", element_name: \"Tech-Writer\" } }\n\nAuth & verification:\n{ operation: \"setup_github_auth\" }\n{ operation: \"configure_oauth\", params: { client_id: \"your-client-id\" } }\n{ operation: \"verify_challenge\", params: { code: \"ABC123\" } }\n{ operation: \"release_deadlock\" }\n{ operation: \"beetlejuice_beetlejuice_beetlejuice\" }\n\nBatch operations: Use the operations array to execute multiple operations sequentially in a single request.\n{ operations: [{ operation: \"addEntry\", params: { element_name: \"log\", content: \"Step 1\" } }, { operation: \"addEntry\", params: { element_name: \"log\", content: \"Step 2\" } }] }\n\nDiscover required parameters — use mcp_aql_read:\n{ operation: \"introspect\", params: { query: \"operations\", name: \"create_element\" } }\nDiscover element format specs (required fields, syntax, examples) — use mcp_aql_read:\n{ operation: \"introspect\", params: { query: \"format\", name: \"template\" } }`,\n        inputSchema: operationInputSchema,\n        annotations: {\n          readOnlyHint: false,\n          destructiveHint: false\n        }\n      },\n      handler: async (args: any) => {\n        const result = await handler.handleCreate(args);\n        return {\n          content: [\n            {\n              type: \"text\",\n              text: JSON.stringify(result, null, 2)\n            }\n          ]\n        };\n      }\n    },\n\n    // mcp_aql_read - Safe, read-only operations\n    {\n      tool: {\n        name: \"mcp_aql_read\",\n        description: `Safe, read-only operations.\n\nSupported operations: ${getOperationsString('READ')}\n\nElement types: ${elementTypes}\n\nThese queries only read data and never modify server state.\n\nQuick start examples:\n{ operation: \"list_elements\", element_type: \"persona\" }\n{ operation: \"get_active_elements\", element_type: \"persona\" }\n{ operation: \"search_elements\", params: { query: \"creative\" } }\n{ operation: \"get_element\", element_type: \"memory\", params: { element_name: \"session-notes\" } }\n\nElement operations:\n{ operation: \"activate_element\", element_type: \"persona\", params: { element_name: \"Default\" } }\n{ operation: \"deactivate_element\", element_type: \"persona\", params: { element_name: \"Default\" } }\n{ operation: \"get_element_details\", element_type: \"skill\", params: { element_name: \"code-review\" } }\n{ operation: \"query_elements\", element_type: \"persona\", params: { filters: { category: \"creative\" } } }\n{ operation: \"validate_element\", element_type: \"agent\", params: { element_name: \"task-planner\" } }\n{ operation: \"render\", params: { element_name: \"meeting-notes\", variables: { date: \"2026-03-03\" } } }\n{ operation: \"export_element\", element_type: \"persona\", params: { element_name: \"Tech-Writer\" } }\n{ operation: \"open_portfolio_browser\" }\n{ operation: \"open_logs\" }\n{ operation: \"open_metrics\" }\n{ operation: \"open_permissions\" }\n{ operation: \"open_setup\" }\n\nMemory-specific search (filter by tags):\n{ operation: \"search\", params: { query: \"*\", type: \"memory\", filters: { tags: [\"important\"] } } }\n\nExecution lifecycle — read-only queries:\n{ operation: \"get_execution_state\", params: { element_name: \"code-reviewer\" } }\n{ operation: \"get_gathered_data\", params: { element_name: \"code-reviewer\", goalId: \"goal-id\" } }\nFor execution-state reads, reuse the same element_name you passed to execute_agent. If element_name is missing, retry with the same agent name rather than inventing a new one.\n\nCollection:\n{ operation: \"browse_collection\", params: { section: \"personas\" } }\n{ operation: \"search_collection\", params: { query: \"creative\" } }\n{ operation: \"search_collection_enhanced\", params: { query: \"creative\", page: 1 } }\n{ operation: \"get_collection_content\", params: { element_type: \"persona\", element_name: \"Creative-Writer\" } }\n{ operation: \"get_collection_cache_health\" }\n\nPortfolio:\n{ operation: \"portfolio_status\" }\n{ operation: \"portfolio_config\" }\n{ operation: \"search_portfolio\", params: { query: \"creative\" } }\n{ operation: \"search_all\", params: { query: \"creative\" } }\n\nSystem:\n{ operation: \"dollhouse_config\" }\n{ operation: \"get_build_info\" }\n{ operation: \"get_cache_budget_report\" }\n{ operation: \"query_logs\", params: { level: \"error\", limit: 10 } }\n{ operation: \"query_metrics\" }\n{ operation: \"query_metrics\", params: { names: [\"system.memory.*\"], type: \"gauge\" } }\n{ operation: \"convert_skill_format\", params: { direction: \"agent_to_dollhouse\", agent_skill: { \"SKILL.md\": \"---\\\\nname: my-skill\\\\ndescription: test\\\\n---\\\\n\\\\nUse this skill.\" } } }\n{ operation: \"convert_skill_format\", params: { direction: \"agent_to_dollhouse\", security_mode: \"warn\", path_mode: \"lossless\", agent_skill: { \"SKILL.md\": \"---\\\\nname: my-skill\\\\ndescription: test\\\\n---\\\\n\\\\nUse this skill.\" } } }\n{ operation: \"convert_skill_format\", params: { direction: \"dollhouse_to_agent\", path_mode: \"lossless\", dollhouse_markdown: \"---\\\\nname: my-skill\\\\ndescription: test\\\\ninstructions: Use this skill.\\\\n---\\\\n\\\\n### binaries/logo.png\\\\n(binary link: ./skills/binaries/logo.png)\" } }\n\nAuth:\n{ operation: \"check_github_auth\" }\n{ operation: \"oauth_helper_status\" }\n\nGatekeeper & CLI policies:\n{ operation: \"permission_prompt\", params: { tool: \"Bash\", prompt: \"run npm test\" } }\n{ operation: \"evaluate_permission\", params: { tool_name: \"Bash\", input: { command: \"git status\" }, platform: \"claude_code\" } }\n{ operation: \"get_effective_cli_policies\" }\n{ operation: \"get_pending_cli_approvals\" }\n{ operation: \"get_permission_authority\" }\n{ operation: \"get_permission_authority\", params: { host: \"claude-code\" } }\n\nEnhanced index:\n{ operation: \"find_similar_elements\", params: { element_type: \"persona\", element_name: \"Creative-Writer\" } }\n{ operation: \"get_element_relationships\", params: { element_type: \"skill\", element_name: \"code-review\" } }\n{ operation: \"search_by_verb\", params: { verb: \"review\" } }\n{ operation: \"get_relationship_stats\" }\n\nDiscover all operations and parameters:\n{ operation: \"get_capabilities\" }\n{ operation: \"get_capabilities\", params: { category: \"Element Lifecycle\" } }\n{ operation: \"introspect\", params: { query: \"operations\" } }`,\n        inputSchema: operationInputSchema,\n        annotations: {\n          readOnlyHint: true,\n          destructiveHint: false\n        }\n      },\n      handler: async (args: any) => {\n        const result = await handler.handleRead(args);\n        return {\n          content: [\n            {\n              type: \"text\",\n              text: JSON.stringify(result, null, 2)\n            }\n          ]\n        };\n      }\n    },\n\n    // mcp_aql_update - Modifying operations that overwrite data\n    {\n      tool: {\n        name: \"mcp_aql_update\",\n        description: `Modifying operations that overwrite data.\n\nSupported operations: ${getOperationsString('UPDATE')}\n\nElement types: ${elementTypes}\n\nThese operations modify existing data, potentially overwriting previous values.\n\nNote: Memories are append-only and do not support edit_element. Use addEntry (CREATE) to add new entries.\n\nQuick start example:\n{ operation: \"edit_element\", element_type: \"persona\", params: { element_name: \"MyPersona\", input: { description: \"Updated description\" } } }\n{ operation: \"edit_element\", element_type: \"persona\", params: { element_name: \"Friendly-Teacher\", input: { instructions: \"Updated behavioral directives.\" } } }\n{ operation: \"edit_element\", element_type: \"agent\", params: { element_name: \"code-reviewer\", input: { instructions: \"Updated agent behavioral profile.\", goal: { template: \"Complete: {task}\" } } } }\n{ operation: \"upgrade_element\", element_type: \"agent\", params: { element_name: \"task-planner\" } }\n\nDiscover required parameters — use mcp_aql_read:\n{ operation: \"introspect\", params: { query: \"operations\", name: \"edit_element\" } }`,\n        inputSchema: operationInputSchema,\n        annotations: {\n          readOnlyHint: false,\n          destructiveHint: true\n        }\n      },\n      handler: async (args: any) => {\n        const result = await handler.handleUpdate(args);\n        return {\n          content: [\n            {\n              type: \"text\",\n              text: JSON.stringify(result, null, 2)\n            }\n          ]\n        };\n      }\n    },\n\n    // mcp_aql_delete - Destructive operations that remove data\n    {\n      tool: {\n        name: \"mcp_aql_delete\",\n        description: `Destructive operations that remove data.\n\nSupported operations: ${getOperationsString('DELETE')}\n\nElement types: ${elementTypes}\n\nThese operations remove data. Use with caution.\n\n⚠️ SECURITY: Do not auto-allow this endpoint in your host settings (e.g., Claude Code settings.json). Each delete operation should require explicit human approval. Auto-allowing bypasses the per-operation confirmation gate, leaving only element deny policies as protection against unintended data loss.\n\nQuick start examples:\n{ operation: \"delete_element\", element_type: \"persona\", params: { element_name: \"Old-Persona\" } }\n{ operation: \"clear\", params: { element_name: \"temp-notes\" } }\n{ operation: \"clear_github_auth\" }\n\nDiscover required parameters — use mcp_aql_read:\n{ operation: \"introspect\", params: { query: \"operations\", name: \"delete_element\" } }`,\n        inputSchema: operationInputSchema,\n        annotations: {\n          readOnlyHint: false,\n          destructiveHint: true\n        }\n      },\n      handler: async (args: any) => {\n        const result = await handler.handleDelete(args);\n        return {\n          content: [\n            {\n              type: \"text\",\n              text: JSON.stringify(result, null, 2)\n            }\n          ]\n        };\n      }\n    },\n\n    // mcp_aql_execute - Execution lifecycle operations (CRUDE's 'E')\n    {\n      tool: {\n        name: \"mcp_aql_execute\",\n        description: `Execution lifecycle operations for executable elements (agents, workflows, pipelines).\n\nSupported operations: ${getOperationsString('EXECUTE')}\n\nThese operations manage runtime execution state. Unlike CRUD operations (which manage definitions), Execute operations handle the execution lifecycle:\n- execute_agent: Start a new execution (returns goalId and stateVersion for tracking)\n- complete_execution: Signal successful completion once the goal is done\n- continue_execution: Resume a previously paused execution with the same goal parameters\n- abort_execution: Abort a running execution, rejecting further operations\n- confirm_operation: Confirm a pending operation that requires user approval (Gatekeeper flow)\n- approve_cli_permission: Approve a pending CLI tool permission request\n- prepare_handoff: Serialize goal progress into a portable handoff block for session transfer\n- resume_from_handoff: Resume agent execution from a handoff block with integrity validation\n\nIMPORTANT: Execute operations are potentially destructive (agents can perform any action) and non-idempotent (calling execute_agent twice creates two separate executions).\n\n⚠️ SECURITY: Do not auto-allow this endpoint in your host settings (e.g., Claude Code settings.json). Each execution should require explicit human approval. Auto-allowing bypasses the per-operation confirmation gate. While DangerZone verification and element deny policies still provide protection, the primary human review checkpoint is lost.\n\nCanonical loop:\n1. Call execute_agent once to start the goal and receive { goalId, stateVersion, activeElements, safetyTier, ... }.\n2. After each chunk of work, use mcp_aql_create: { operation: \"record_execution_step\", ... }.\n3. Read record_execution_step.autonomy.continue and any autonomy.notifications to decide whether to continue, pause for a human, or handle a gatekeeper block.\n4. When the goal is finished, call complete_execution.\nUse continue_execution only when an already-started goal was paused and you are resuming it with the same goal parameters. It is not the normal next call after execute_agent.\n\nQuick start examples:\n{ operation: \"execute_agent\", params: { element_name: \"code-reviewer\", parameters: { objective: \"Review code\" } } }\nNext lifecycle step — use mcp_aql_create:\n{ operation: \"record_execution_step\", params: { element_name: \"code-reviewer\", stepDescription: \"Reviewed auth module\", outcome: \"success\", findings: \"Found 2 security issues\" } }\n{ operation: \"complete_execution\", params: { element_name: \"code-reviewer\", outcome: \"success\", summary: \"Completed review\" } }\n{ operation: \"abort_execution\", params: { element_name: \"data-collector\", reason: \"User requested cancellation\" } }\n{ operation: \"continue_execution\", params: { element_name: \"rubric-qa-agent\", previousStepResult: \"Verified citation set\", parameters: { run_dir: \"/app/run\", deliverable_path: \"/app/run/output.docx\" } } }\n{ operation: \"confirm_operation\", params: { operation: \"execute_agent\" } }\n{ operation: \"approve_cli_permission\", params: { request_id: \"req-123\", decision: \"allow\" } }\n{ operation: \"prepare_handoff\", params: { element_name: \"code-reviewer\" } }\n{ operation: \"resume_from_handoff\", params: { element_name: \"code-reviewer\", handoff_block: \"...\" } }\n\nDiscover required parameters — use mcp_aql_read:\n{ operation: \"introspect\", params: { query: \"operations\", name: \"execute_agent\" } }`,\n        inputSchema: operationInputSchema,\n        annotations: {\n          readOnlyHint: false,\n          destructiveHint: true  // Potentially destructive - agents can perform any action\n        }\n      },\n      handler: async (args: any) => {\n        const result = await handler.handleExecute(args);\n        return {\n          content: [\n            {\n              type: \"text\",\n              text: JSON.stringify(result, null, 2)\n            }\n          ]\n        };\n      }\n    }\n  ];\n}\n"]}
@@ -10,7 +10,7 @@
10
10
  */
11
11
  import { IFileOperationsService } from './FileOperationsService.js';
12
12
  import type { StartupTimer, StartupReport } from '../telemetry/StartupTimer.js';
13
- import { type PermissionHookHealthSummary, type PermissionHookStartupRepairSummary } from '../utils/permissionHooks.js';
13
+ import { type PermissionHookDiagnosticRecord, type PermissionHookHealthSummary, type PermissionHookStartupRepairSummary } from '../utils/permissionHooks.js';
14
14
  export interface BuildInfo {
15
15
  sessionId: string;
16
16
  runtimeSessionId: string;
@@ -51,6 +51,8 @@ export interface BuildInfo {
51
51
  currentHosts: string[];
52
52
  repairedHosts: string[];
53
53
  needsRepairHosts: string[];
54
+ diagnosticsPath: string;
55
+ lastDiagnostic: PermissionHookDiagnosticRecord | null;
54
56
  lastStartupRepair: PermissionHookStartupRepairSummary | null;
55
57
  };
56
58
  /** Issue #706: Startup timing and readiness status. */
@@ -1 +1 @@
1
- {"version":3,"file":"BuildInfoService.d.ts","sourceRoot":"","sources":["../../src/services/BuildInfoService.ts"],"names":[],"mappings":"AAAA;;;;;;;;;GASG;AAIH,OAAO,EAAE,sBAAsB,EAAE,MAAM,4BAA4B,CAAC;AAEpE,OAAO,KAAK,EAAE,YAAY,EAAE,aAAa,EAAE,MAAM,8BAA8B,CAAC;AAEhF,OAAO,EAGL,KAAK,2BAA2B,EAChC,KAAK,kCAAkC,EACxC,MAAM,6BAA6B,CAAC;AAErC,MAAM,WAAW,SAAS;IACxB,SAAS,EAAE,MAAM,CAAC;IAClB,gBAAgB,EAAE,MAAM,CAAC;IACzB,aAAa,EAAE,KAAK,GAAG,SAAS,CAAC;IACjC,OAAO,EAAE;QACP,IAAI,EAAE,MAAM,CAAC;QACb,OAAO,EAAE,MAAM,CAAC;KACjB,CAAC;IACF,KAAK,EAAE;QACL,SAAS,CAAC,EAAE,MAAM,CAAC;QACnB,IAAI,EAAE,KAAK,GAAG,KAAK,GAAG,SAAS,CAAC;QAChC,SAAS,CAAC,EAAE,MAAM,CAAC;QACnB,SAAS,CAAC,EAAE,MAAM,CAAC;KACpB,CAAC;IACF,OAAO,EAAE;QACP,WAAW,EAAE,MAAM,CAAC;QACpB,QAAQ,EAAE,MAAM,CAAC;QACjB,IAAI,EAAE,MAAM,CAAC;QACb,MAAM,EAAE,MAAM,CAAC;QACf,WAAW,EAAE,MAAM,CAAC,WAAW,CAAC;KACjC,CAAC;IACF,WAAW,EAAE;QACX,OAAO,CAAC,EAAE,MAAM,CAAC;QACjB,YAAY,EAAE,OAAO,CAAC;QACtB,aAAa,EAAE,OAAO,CAAC;QACvB,OAAO,EAAE,OAAO,CAAC;QACjB,QAAQ,EAAE,OAAO,CAAC;QAClB,UAAU,CAAC,EAAE,MAAM,CAAC;KACrB,CAAC;IACF,MAAM,EAAE;QACN,SAAS,EAAE,IAAI,CAAC;QAChB,MAAM,EAAE,MAAM,CAAC;QACf,aAAa,EAAE,OAAO,CAAC;KACxB,CAAC;IACF,eAAe,CAAC,EAAE;QAChB,MAAM,EAAE,2BAA2B,CAAC;QACpC,cAAc,EAAE,MAAM,EAAE,CAAC;QACzB,YAAY,EAAE,MAAM,EAAE,CAAC;QACvB,aAAa,EAAE,MAAM,EAAE,CAAC;QACxB,gBAAgB,EAAE,MAAM,EAAE,CAAC;QAC3B,iBAAiB,EAAE,kCAAkC,GAAG,IAAI,CAAC;KAC9D,CAAC;IACF,uDAAuD;IACvD,OAAO,CAAC,EAAE;QACR,MAAM,EAAE,OAAO,GAAG,cAAc,CAAC;QACjC,qBAAqB,EAAE,OAAO,CAAC;QAC/B,QAAQ,EAAE,MAAM,CAAC;QACjB,eAAe,CAAC,EAAE,aAAa,CAAC;KACjC,CAAC;CACH;AAED,qBAAa,gBAAgB;IAC3B,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAO;IACjC,OAAO,CAAC,QAAQ,CAAC,cAAc,CAAyB;IACxD,qEAAqE;IACrE,OAAO,CAAC,YAAY,CAA6B;IACjD,2DAA2D;IAC3D,OAAO,CAAC,oBAAoB,CAAgC;gBAEhD,cAAc,EAAE,sBAAsB;IAKlD;;;OAGG;IACH,eAAe,CAAC,KAAK,EAAE,YAAY,GAAG,IAAI;IAI1C;;OAEG;IACH,uBAAuB,CAAC,OAAO,EAAE,MAAM,OAAO,GAAG,IAAI;IAIrD;;;;;;;OAOG;IACU,YAAY,IAAI,OAAO,CAAC,SAAS,CAAC;IA2F/C;;;;OAIG;IACI,eAAe,CAAC,IAAI,EAAE,SAAS,GAAG,MAAM;IA+H/C;;;OAGG;YACW,UAAU;IAcxB;;;OAGG;YACW,aAAa;IA+B3B,OAAO,CAAC,YAAY;IAepB,OAAO,CAAC,YAAY;CAIrB"}
1
+ {"version":3,"file":"BuildInfoService.d.ts","sourceRoot":"","sources":["../../src/services/BuildInfoService.ts"],"names":[],"mappings":"AAAA;;;;;;;;;GASG;AAIH,OAAO,EAAE,sBAAsB,EAAE,MAAM,4BAA4B,CAAC;AAEpE,OAAO,KAAK,EAAE,YAAY,EAAE,aAAa,EAAE,MAAM,8BAA8B,CAAC;AAEhF,OAAO,EAEL,KAAK,8BAA8B,EAEnC,KAAK,2BAA2B,EAChC,KAAK,kCAAkC,EACxC,MAAM,6BAA6B,CAAC;AAErC,MAAM,WAAW,SAAS;IACxB,SAAS,EAAE,MAAM,CAAC;IAClB,gBAAgB,EAAE,MAAM,CAAC;IACzB,aAAa,EAAE,KAAK,GAAG,SAAS,CAAC;IACjC,OAAO,EAAE;QACP,IAAI,EAAE,MAAM,CAAC;QACb,OAAO,EAAE,MAAM,CAAC;KACjB,CAAC;IACF,KAAK,EAAE;QACL,SAAS,CAAC,EAAE,MAAM,CAAC;QACnB,IAAI,EAAE,KAAK,GAAG,KAAK,GAAG,SAAS,CAAC;QAChC,SAAS,CAAC,EAAE,MAAM,CAAC;QACnB,SAAS,CAAC,EAAE,MAAM,CAAC;KACpB,CAAC;IACF,OAAO,EAAE;QACP,WAAW,EAAE,MAAM,CAAC;QACpB,QAAQ,EAAE,MAAM,CAAC;QACjB,IAAI,EAAE,MAAM,CAAC;QACb,MAAM,EAAE,MAAM,CAAC;QACf,WAAW,EAAE,MAAM,CAAC,WAAW,CAAC;KACjC,CAAC;IACF,WAAW,EAAE;QACX,OAAO,CAAC,EAAE,MAAM,CAAC;QACjB,YAAY,EAAE,OAAO,CAAC;QACtB,aAAa,EAAE,OAAO,CAAC;QACvB,OAAO,EAAE,OAAO,CAAC;QACjB,QAAQ,EAAE,OAAO,CAAC;QAClB,UAAU,CAAC,EAAE,MAAM,CAAC;KACrB,CAAC;IACF,MAAM,EAAE;QACN,SAAS,EAAE,IAAI,CAAC;QAChB,MAAM,EAAE,MAAM,CAAC;QACf,aAAa,EAAE,OAAO,CAAC;KACxB,CAAC;IACF,eAAe,CAAC,EAAE;QAChB,MAAM,EAAE,2BAA2B,CAAC;QACpC,cAAc,EAAE,MAAM,EAAE,CAAC;QACzB,YAAY,EAAE,MAAM,EAAE,CAAC;QACvB,aAAa,EAAE,MAAM,EAAE,CAAC;QACxB,gBAAgB,EAAE,MAAM,EAAE,CAAC;QAC3B,eAAe,EAAE,MAAM,CAAC;QACxB,cAAc,EAAE,8BAA8B,GAAG,IAAI,CAAC;QACtD,iBAAiB,EAAE,kCAAkC,GAAG,IAAI,CAAC;KAC9D,CAAC;IACF,uDAAuD;IACvD,OAAO,CAAC,EAAE;QACR,MAAM,EAAE,OAAO,GAAG,cAAc,CAAC;QACjC,qBAAqB,EAAE,OAAO,CAAC;QAC/B,QAAQ,EAAE,MAAM,CAAC;QACjB,eAAe,CAAC,EAAE,aAAa,CAAC;KACjC,CAAC;CACH;AAED,qBAAa,gBAAgB;IAC3B,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAO;IACjC,OAAO,CAAC,QAAQ,CAAC,cAAc,CAAyB;IACxD,qEAAqE;IACrE,OAAO,CAAC,YAAY,CAA6B;IACjD,2DAA2D;IAC3D,OAAO,CAAC,oBAAoB,CAAgC;gBAEhD,cAAc,EAAE,sBAAsB;IAKlD;;;OAGG;IACH,eAAe,CAAC,KAAK,EAAE,YAAY,GAAG,IAAI;IAI1C;;OAEG;IACH,uBAAuB,CAAC,OAAO,EAAE,MAAM,OAAO,GAAG,IAAI;IAIrD;;;;;;;OAOG;IACU,YAAY,IAAI,OAAO,CAAC,SAAS,CAAC;IAuG/C;;;;OAIG;IACI,eAAe,CAAC,IAAI,EAAE,SAAS,GAAG,MAAM;IAoJ/C;;;OAGG;YACW,UAAU;IAcxB;;;OAGG;YACW,aAAa;IA+B3B,OAAO,CAAC,YAAY;IAepB,OAAO,CAAC,YAAY;CAIrB"}