@dollhousemcp/mcp-server 2.0.31 → 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.
@@ -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":"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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{ 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"]}
@@ -2006,6 +2006,9 @@ globalThis.DollhouseConsoleUI.clearBanner = function(bannerId) {
2006
2006
 
2007
2007
  // ── Tab switching ─────────────────────────────────────────────────────────
2008
2008
  const consoleTabs = document.getElementById('console-tabs');
2009
+ const headerNavRow = document.querySelector('.header-nav-row');
2010
+ const consoleTabMenuToggle = document.getElementById('console-tab-menu-toggle');
2011
+ const consoleTabMenu = document.getElementById('console-tab-menu');
2009
2012
  const tabInits = { logs: false, metrics: false, permissions: false, security: false };
2010
2013
 
2011
2014
  const TAB_KEY = 'dollhousemcp-active-tab';
@@ -2090,8 +2093,157 @@ globalThis.DollhouseConsoleUI.clearBanner = function(bannerId) {
2090
2093
  p.hidden = p.id !== 'tab-' + tabName;
2091
2094
  p.classList.toggle('active', p.id === 'tab-' + tabName);
2092
2095
  });
2096
+ syncConsoleTabMenuSelection();
2097
+ closeConsoleTabMenu();
2098
+ scheduleConsoleTabOverflowCheck();
2093
2099
  };
2094
2100
 
2101
+ function renderConsoleTabMenu() {
2102
+ if (!consoleTabs || !consoleTabMenu) return;
2103
+ consoleTabMenu.innerHTML = '';
2104
+ consoleTabs.querySelectorAll('.console-tab').forEach((btn) => {
2105
+ const item = document.createElement('button');
2106
+ item.type = 'button';
2107
+ item.className = 'console-tab-menu-item';
2108
+ item.dataset.tab = btn.dataset.tab || '';
2109
+ item.setAttribute('role', 'menuitem');
2110
+ item.textContent = btn.textContent || '';
2111
+ if (btn.classList.contains('active')) item.classList.add('active');
2112
+ consoleTabMenu.appendChild(item);
2113
+ });
2114
+ }
2115
+
2116
+ function syncConsoleTabMenuSelection() {
2117
+ if (!consoleTabs || !consoleTabMenu) return;
2118
+ const activeTab = consoleTabs.querySelector('.console-tab.active')?.dataset.tab || '';
2119
+ consoleTabMenu.querySelectorAll('.console-tab-menu-item').forEach((item) => {
2120
+ item.classList.toggle('active', item.dataset.tab === activeTab);
2121
+ });
2122
+ }
2123
+
2124
+ function closeConsoleTabMenu() {
2125
+ if (!consoleTabMenu || !consoleTabMenuToggle) return;
2126
+ consoleTabMenu.hidden = true;
2127
+ consoleTabMenuToggle.setAttribute('aria-expanded', 'false');
2128
+ }
2129
+
2130
+ function getConsoleTabMenuItems() {
2131
+ if (!consoleTabMenu) return [];
2132
+ return Array.from(consoleTabMenu.querySelectorAll('.console-tab-menu-item'));
2133
+ }
2134
+
2135
+ function openConsoleTabMenu(focusTarget) {
2136
+ if (!consoleTabMenu || !consoleTabMenuToggle) return;
2137
+ consoleTabMenu.hidden = false;
2138
+ consoleTabMenuToggle.setAttribute('aria-expanded', 'true');
2139
+
2140
+ if (!focusTarget) return;
2141
+
2142
+ const items = getConsoleTabMenuItems();
2143
+ if (!items.length) return;
2144
+ const focusItem = (index) => {
2145
+ const boundedIndex = Math.max(0, Math.min(index, items.length - 1));
2146
+ items[boundedIndex].focus();
2147
+ };
2148
+
2149
+ if (focusTarget === 'first') {
2150
+ focusItem(0);
2151
+ return;
2152
+ }
2153
+
2154
+ if (focusTarget === 'last') {
2155
+ focusItem(items.length - 1);
2156
+ return;
2157
+ }
2158
+
2159
+ const activeIndex = Math.max(items.findIndex((item) => item.classList.contains('active')), 0);
2160
+ focusItem(activeIndex);
2161
+ }
2162
+
2163
+ function moveConsoleTabMenuFocus(step, origin) {
2164
+ const items = getConsoleTabMenuItems();
2165
+ if (!items.length) return;
2166
+
2167
+ const currentIndex = origin instanceof Element
2168
+ ? items.findIndex((item) => item === origin || item.contains(origin))
2169
+ : -1;
2170
+ const activeIndex = Math.max(items.findIndex((item) => item.classList.contains('active')), 0);
2171
+ const baseIndex = currentIndex >= 0 ? currentIndex : activeIndex;
2172
+ const nextIndex = (baseIndex + step + items.length) % items.length;
2173
+ items[nextIndex].focus();
2174
+ }
2175
+
2176
+ function handleConsoleTabMenuToggleKeydown(event) {
2177
+ if (event.key === 'ArrowDown') {
2178
+ event.preventDefault();
2179
+ openConsoleTabMenu('first');
2180
+ } else if (event.key === 'ArrowUp') {
2181
+ event.preventDefault();
2182
+ openConsoleTabMenu('last');
2183
+ }
2184
+ }
2185
+
2186
+ function handleConsoleTabMenuKeydown(event) {
2187
+ switch (event.key) {
2188
+ case 'ArrowDown':
2189
+ event.preventDefault();
2190
+ moveConsoleTabMenuFocus(1, event.target);
2191
+ break;
2192
+ case 'ArrowUp':
2193
+ event.preventDefault();
2194
+ moveConsoleTabMenuFocus(-1, event.target);
2195
+ break;
2196
+ case 'Home':
2197
+ event.preventDefault();
2198
+ getConsoleTabMenuItems()[0]?.focus();
2199
+ break;
2200
+ case 'End': {
2201
+ event.preventDefault();
2202
+ const items = getConsoleTabMenuItems();
2203
+ items.at(-1)?.focus();
2204
+ break;
2205
+ }
2206
+ case 'Escape':
2207
+ event.preventDefault();
2208
+ closeConsoleTabMenu();
2209
+ consoleTabMenuToggle?.focus();
2210
+ break;
2211
+ default:
2212
+ break;
2213
+ }
2214
+ }
2215
+
2216
+ function tabsNeedOverflowMenu() {
2217
+ if (!consoleTabs) return false;
2218
+ const buttons = Array.from(consoleTabs.querySelectorAll('.console-tab'));
2219
+ if (buttons.length < 2) return false;
2220
+ const firstTop = buttons[0].offsetTop;
2221
+ return buttons.some((btn) => btn.offsetTop !== firstTop);
2222
+ }
2223
+
2224
+ let consoleTabOverflowFrame = 0;
2225
+ function updateConsoleTabOverflow() {
2226
+ if (!consoleTabs || !headerNavRow || !consoleTabMenuToggle) return;
2227
+ headerNavRow.classList.remove('header-nav-row--collapsed');
2228
+ consoleTabMenuToggle.hidden = true;
2229
+ const shouldCollapse = tabsNeedOverflowMenu();
2230
+ if (shouldCollapse) {
2231
+ headerNavRow.classList.add('header-nav-row--collapsed');
2232
+ consoleTabMenuToggle.hidden = false;
2233
+ }
2234
+ if (!shouldCollapse) {
2235
+ closeConsoleTabMenu();
2236
+ }
2237
+ }
2238
+
2239
+ function scheduleConsoleTabOverflowCheck() {
2240
+ if (consoleTabOverflowFrame) cancelAnimationFrame(consoleTabOverflowFrame);
2241
+ consoleTabOverflowFrame = requestAnimationFrame(() => {
2242
+ consoleTabOverflowFrame = 0;
2243
+ updateConsoleTabOverflow();
2244
+ });
2245
+ }
2246
+
2095
2247
  /**
2096
2248
  * Parse the URL hash into tab name and query parameters.
2097
2249
  * Supports fragment-query pattern: #tab?key=value&key=value
@@ -2208,26 +2360,41 @@ globalThis.DollhouseConsoleUI.clearBanner = function(bannerId) {
2208
2360
  return false;
2209
2361
  }
2210
2362
 
2211
- if (!applyHashTab()) {
2363
+ function restoreInitialConsoleTab() {
2212
2364
  // Version check takes priority over saved tab — upgraders must see Setup
2213
2365
  // regardless of whether they have a saved tab from their previous session.
2214
2366
  if (localStorage.getItem(SETUP_SEEN_KEY) === currentServerVersion) {
2215
2367
  const savedTab = localStorage.getItem(TAB_KEY);
2216
- if (savedTab) {
2217
- switchToTab(savedTab);
2218
- lazyInitTab(savedTab, tabInits);
2219
- }
2220
- } else {
2221
- localStorage.setItem(SETUP_SEEN_KEY, currentServerVersion);
2222
- switchToTab('setup');
2368
+ if (!savedTab) return;
2369
+ switchToTab(savedTab);
2370
+ lazyInitTab(savedTab, tabInits);
2371
+ return;
2223
2372
  }
2373
+
2374
+ localStorage.setItem(SETUP_SEEN_KEY, currentServerVersion);
2375
+ switchToTab('setup');
2376
+ }
2377
+
2378
+ function initializeConsoleTabs() {
2379
+ if (applyHashTab()) return;
2380
+ restoreInitialConsoleTab();
2224
2381
  }
2225
2382
 
2226
- // Handle hash changes for deep-linking (e.g., open_logs operation)
2227
- globalThis.addEventListener('hashchange', () => applyHashTab());
2383
+ function registerConsoleTabObservers() {
2384
+ globalThis.addEventListener('hashchange', () => applyHashTab());
2385
+ renderConsoleTabMenu();
2386
+ syncConsoleTabMenuSelection();
2387
+ scheduleConsoleTabOverflowCheck();
2388
+ globalThis.addEventListener('resize', scheduleConsoleTabOverflowCheck);
2389
+
2390
+ if (typeof ResizeObserver !== 'function' || !headerNavRow) return;
2228
2391
 
2229
- if (consoleTabs) {
2230
- consoleTabs.addEventListener('click', (e) => {
2392
+ const tabOverflowObserver = new ResizeObserver(() => scheduleConsoleTabOverflowCheck());
2393
+ tabOverflowObserver.observe(headerNavRow);
2394
+ }
2395
+
2396
+ function registerConsoleTabInteractions() {
2397
+ consoleTabs?.addEventListener('click', (e) => {
2231
2398
  const btn = e.target.closest('.console-tab');
2232
2399
  if (!btn) return;
2233
2400
  const tab = btn.dataset.tab;
@@ -2238,8 +2405,42 @@ globalThis.DollhouseConsoleUI.clearBanner = function(bannerId) {
2238
2405
 
2239
2406
  lazyInitTab(tab, tabInits);
2240
2407
  });
2408
+
2409
+ consoleTabMenuToggle?.addEventListener('click', () => {
2410
+ if (!consoleTabMenu) return;
2411
+ if (consoleTabMenu.hidden) {
2412
+ openConsoleTabMenu();
2413
+ return;
2414
+ }
2415
+
2416
+ closeConsoleTabMenu();
2417
+ });
2418
+
2419
+ consoleTabMenuToggle?.addEventListener('keydown', handleConsoleTabMenuToggleKeydown);
2420
+ consoleTabMenu?.addEventListener('keydown', handleConsoleTabMenuKeydown);
2421
+ consoleTabMenu?.addEventListener('click', (e) => {
2422
+ const btn = e.target.closest('.console-tab-menu-item');
2423
+ const tab = btn?.dataset.tab;
2424
+ if (!tab) return;
2425
+ switchToTab(tab);
2426
+ localStorage.setItem(TAB_KEY, tab);
2427
+ lazyInitTab(tab, tabInits);
2428
+ });
2429
+
2430
+ globalThis.addEventListener('click', (e) => {
2431
+ if (!consoleTabMenu || !consoleTabMenuToggle) return;
2432
+ if (consoleTabMenu.hidden) return;
2433
+ const target = e.target;
2434
+ if (!(target instanceof Element)) return;
2435
+ if (consoleTabMenu.contains(target) || consoleTabMenuToggle.contains(target)) return;
2436
+ closeConsoleTabMenu();
2437
+ });
2241
2438
  }
2242
2439
 
2440
+ initializeConsoleTabs();
2441
+ registerConsoleTabObservers();
2442
+ registerConsoleTabInteractions();
2443
+
2243
2444
  function lazyInitTab(tab, tabInits, params) {
2244
2445
  const dc = globalThis.DollhouseConsole;
2245
2446
  const module = dc?.[tab];
@@ -43,7 +43,14 @@
43
43
  <p class="site-tagline">Management Console</p>
44
44
  </div>
45
45
  </div>
46
- <div class="header-right">
46
+ <div class="header-controls">
47
+ <div class="site-stats" id="stats" aria-live="polite"></div>
48
+ <button class="theme-toggle" id="theme-toggle" type="button" aria-label="Switch to dark mode" title="Switch to dark mode">
49
+ <span class="theme-toggle-icon" id="theme-toggle-icon" aria-hidden="true">&#9790;</span>
50
+ <span class="sr-only" id="theme-toggle-label">Switch to dark mode</span>
51
+ </button>
52
+ </div>
53
+ <div class="header-nav-row">
47
54
  <nav class="console-tabs" id="console-tabs" aria-label="Console tabs">
48
55
  <button class="console-tab" data-tab="setup">Setup</button>
49
56
  <button class="console-tab" data-tab="security">Auth</button>
@@ -52,12 +59,14 @@
52
59
  <button class="console-tab" data-tab="permissions">Permissions</button>
53
60
  <button class="console-tab active" data-tab="portfolio">Portfolio</button>
54
61
  </nav>
62
+ <div class="console-tab-menu-shell" id="console-tab-menu-shell">
63
+ <button class="console-tab-menu-toggle" id="console-tab-menu-toggle" type="button" hidden aria-haspopup="menu" aria-expanded="false" aria-controls="console-tab-menu">
64
+ <span class="console-tab-menu-icon" aria-hidden="true">&#9776;</span>
65
+ <span class="console-tab-menu-label">Menu</span>
66
+ </button>
67
+ <div class="console-tab-menu" id="console-tab-menu" role="menu" aria-orientation="vertical" hidden></div>
68
+ </div>
55
69
  <div class="session-indicator" id="session-indicator" title="Active sessions"></div>
56
- <div class="site-stats" id="stats" aria-live="polite"></div>
57
- <button class="theme-toggle" id="theme-toggle" type="button" aria-label="Switch to dark mode" title="Switch to dark mode">
58
- <span class="theme-toggle-icon" id="theme-toggle-icon" aria-hidden="true">&#9790;</span>
59
- <span class="sr-only" id="theme-toggle-label">Switch to dark mode</span>
60
- </button>
61
70
  </div>
62
71
  </header>
63
72
 
@@ -572,7 +581,7 @@ npm install @dollhousemcp/mcp-server</code></pre>
572
581
  <a href="https://github.com/DollhouseMCP/mcp-server" class="footer-link">GitHub Repository</a>
573
582
  <a href="./collection-index.json" class="footer-link">JSON API</a>
574
583
  <a href="https://dollhousemcp.com" class="footer-link">DollhouseMCP</a>
575
- <span class="footer-version" id="footer-version"></span>
584
+ <span class="footer-version" id="footer-version" aria-live="polite" aria-atomic="true"></span>
576
585
  <span class="footer-updated" id="footer-updated"></span>
577
586
  <span class="footer-copyright">&copy; 2026 DollhouseMCP</span>
578
587
  </div>