UNPKG

@dollhousemcp/mcp-server

Version:

DollhouseMCP - A Model Context Protocol (MCP) server that enables dynamic AI persona management from markdown files, allowing Claude and other compatible AI assistants to activate and switch between different behavioral personas.

549 lines (501 loc) 72.9 kB
/** * MCP-AQL Tools - Unified query interface for AI agents * * Provides two endpoint modes configurable via MCP_AQL_ENDPOINT_MODE environment variable: * * ## Mode 1: CRUDE Endpoints (Default) - MCP_AQL_ENDPOINT_MODE=crude * 5 tools: mcp_aql_create, mcp_aql_read, mcp_aql_update, mcp_aql_delete, mcp_aql_execute (~4,300 tokens) * * ## Mode 2: Single Endpoint (Minimal) - MCP_AQL_ENDPOINT_MODE=single * 1 tool: mcp_aql (~350 tokens) * Ideal for multi-server deployments where token budget is constrained. * * Note: These tools are only registered when MCP_INTERFACE_MODE=mcpaql (default). * When MCP_INTERFACE_MODE=discrete, discrete tools are registered instead. * * ## Why 5 CRUDE Endpoints? (Default Mode) * * The 5-endpoint CRUDE pattern (Create, Read, Update, Delete, Execute) was chosen for: * * 1. **User Comprehension**: CRUDE extends CRUD with Execute for non-idempotent * operations, making it easy for users to reason about what each endpoint can do. * * 2. **Platform Annotations**: MCP platforms like ChatGPT's apps require * tool annotations that describe safety and destructiveness. The 5-endpoint * split maps directly to distinct permission levels: * - CREATE: additive, non-destructive (readOnlyHint: false, destructiveHint: false) * - READ: safe, read-only (readOnlyHint: true, destructiveHint: false) * - UPDATE: modifying, potentially destructive (readOnlyHint: false, destructiveHint: true) * - DELETE: destructive (readOnlyHint: false, destructiveHint: true) * - EXECUTE: non-idempotent, potentially destructive (readOnlyHint: false, destructiveHint: true) * * 3. **Granular Permission Control**: Users can grant READ endpoint full access * while locking down CREATE, UPDATE, DELETE, and EXECUTE. This enables safe * read-only integrations without exposing mutation capabilities. * * ## Why Single Endpoint? (Minimal Mode) * * The single-endpoint mode was added for: * * 1. **Token Efficiency**: ~350 tokens vs ~4,300 tokens (92% reduction) * 2. **Multi-Server Deployments**: When running multiple MCP servers, token * budgets can be constrained. Single endpoint reduces overhead. * 3. **Simplified Integration**: Some clients prefer a single entry point. * * Security is maintained through server-side Gatekeeper enforcement - the * server determines which operation type is being executed and applies * appropriate permission checks. * * ## GraphQL-Style Introspection * * MCP-AQL follows GraphQL patterns for self-documentation. Tool descriptions * are generated dynamically from OPERATION_ROUTES, ensuring they always reflect * the current available operations. * * ### Introspection Examples * * List all available operations: * ```json * { "operation": "introspect", "params": { "query": "operations" } } * ``` * * Get details for a specific operation (parameters, examples, return types): * ```json * { "operation": "introspect", "params": { "query": "operations", "name": "create_element" } } * ``` * * Discover available types (e.g., ElementType enum values): * ```json * { "operation": "introspect", "params": { "query": "types", "name": "ElementType" } } * ``` * * ### Design Philosophy * * - **Single source of truth**: OPERATION_ROUTES defines all operations * - **Dynamic descriptions**: Adding an operation to OPERATION_ROUTES automatically * updates tool descriptions - no manual synchronization needed * - **Introspection-first**: LLMs discover parameters via introspect, not static docs * - **GraphQL heritage**: Patterns familiar to any LLM trained on GraphQL */ import { UnifiedEndpoint } from '../../handlers/mcp-aql/UnifiedEndpoint.js'; import { getOperationsForEndpoint } from '../../handlers/mcp-aql/OperationRouter.js'; import { ElementType } from '../../handlers/mcp-aql/types.js'; import { env } from '../../config/env.js'; import { ELEMENT_ROLES } from '../../elements/ensembles/constants.js'; // ============================================================================ // Dynamic Description Generation // ============================================================================ /** * Get element types as a comma-separated string. * Derived from the ElementType enum to ensure consistency. */ function getElementTypesString() { return Object.values(ElementType).join(', '); } /** * Get operations for an endpoint as a comma-separated string. * Derived from OPERATION_ROUTES to ensure consistency. */ function getOperationsString(endpoint) { return getOperationsForEndpoint(endpoint).join(', '); } // ============================================================================ // Tool Schema // ============================================================================ /** * Shared input schema for CRUD operations (create, read, update, delete) * All operations use the OperationInput structure from the GraphQL schema */ const operationInputSchema = { type: "object", properties: { operation: { type: "string", description: "Operation name to execute" }, element_type: { type: "string", description: "Target element type (optional)" }, params: { type: "object", description: "Operation parameters" }, operations: { type: "array", description: "Array of operations for batch execution", items: { type: "object", properties: { operation: { type: "string" }, element_type: { type: "string" }, params: { type: "object" } }, required: ["operation"] } } }, required: ["operation"] }; // ============================================================================ // Tool Registration // ============================================================================ /** * Get MCP-AQL tools for registration in the ToolRegistry. * * Returns different tools based on MCP_AQL_ENDPOINT_MODE environment variable: * - 'crude' (default): 5 CRUDE endpoints (Create, Read, Update, Delete, Execute) (~4,300 tokens) * - 'single': 1 unified endpoint (~350 tokens) * * Note: MCP_AQL_MODE is supported as a deprecated alias for backward compatibility. */ export function getMCPAQLTools(handler) { // Use MCP_AQL_ENDPOINT_MODE, falling back to deprecated MCP_AQL_MODE for backward compatibility const mode = env.MCP_AQL_ENDPOINT_MODE ?? env.MCP_AQL_MODE ?? 'crude'; if (mode === 'single') { return getUnifiedTools(handler); } // Default: CRUDE mode (5 endpoints) return getCRUDETools(handler); } /** * Get the unified single endpoint tool (MCP_AQL_MODE=single) * Token footprint: ~300-400 tokens */ function getUnifiedTools(handler) { const unifiedEndpoint = new UnifiedEndpoint(handler); // Build dynamic description from OPERATION_ROUTES const description = `DollhouseMCP unified API - GraphQL-style query interface for AI element management. CRUDE Operations: - CREATE: ${getOperationsString('CREATE')} - READ: ${getOperationsString('READ')} - UPDATE: ${getOperationsString('UPDATE')} - DELETE: ${getOperationsString('DELETE')} - EXECUTE: ${getOperationsString('EXECUTE')} Element types: ${getElementTypesString()} Quick start examples: { operation: "list_elements", element_type: "persona" } { 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." } } { 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 }] } } } { operation: "create_element", element_type: "memory", params: { element_name: "session-notes", description: "Session context and notes" } } { operation: "addEntry", params: { element_name: "session-notes", content: "Remember this fact", tags: ["important"] } } { operation: "execute_agent", params: { element_name: "MyAgent", parameters: { objective: "Review code" } } } { operation: "record_execution_step", params: { element_name: "MyAgent", stepDescription: "Reviewed auth module", outcome: "success", findings: "Found 2 issues" } } { operation: "complete_execution", params: { element_name: "MyAgent", outcome: "success", summary: "Review complete" } } Execution loop: call execute_agent once to start, then record_execution_step after each work chunk, then complete_execution when done. Use continue_execution only to resume a paused execution, and pass the same goal parameters you used for execute_agent when resuming. Gatekeeper: Some operations may return a confirmation prompt instead of executing immediately. Use confirm_operation to approve, then retry. Discover all operations: { operation: "introspect", params: { query: "operations" } }`; return [ { tool: { name: "mcp_aql", description, inputSchema: operationInputSchema, annotations: { // Unified endpoint can perform any operation, so we use conservative hints // The actual operation's safety is enforced server-side by Gatekeeper readOnlyHint: false, destructiveHint: true // Conservative: some operations are destructive } }, handler: async (args) => { const result = await unifiedEndpoint.handle(args); return { content: [ { type: "text", text: JSON.stringify(result, null, 2) } ] }; } } ]; } /** * Get the 5 CRUDE endpoint tools (MCP_AQL_ENDPOINT_MODE=crude, default) * CRUDE = Create, Read, Update, Delete, Execute * Token footprint: ~4,300 tokens (measured via Claude Code /context) * * Descriptions are generated dynamically from OPERATION_ROUTES to ensure * they always reflect the current available operations. */ function getCRUDETools(handler) { const elementTypes = getElementTypesString(); return [ // mcp_aql_create - Additive, non-destructive operations { tool: { name: "mcp_aql_create", description: `Additive, non-destructive operations. Supported operations: ${getOperationsString('CREATE')} Element types: ${elementTypes} These operations add new data without removing or overwriting existing content. Quick start examples: { 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." } } { 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 }] } } } { operation: "create_element", element_type: "memory", params: { element_name: "session-notes", description: "Session context and notes" } } { 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" }] } } } Valid ensemble roles: ${ELEMENT_ROLES.join(', ')} { operation: "addEntry", params: { element_name: "session-notes", content: "Remember this fact", tags: ["important"] } } Note: 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. Execution lifecycle — record agent progress (appends step records, like addEntry): { operation: "record_execution_step", params: { element_name: "code-reviewer", stepDescription: "Analyzed files", outcome: "success", findings: "Found 3 issues" } } This is the normal next lifecycle call after mcp_aql_execute { operation: "execute_agent", ... }. 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. Import & portfolio: { operation: "import_element", element_type: "skill", params: { element_name: "code-formatter", data: "..." } } { operation: "import_persona", params: { source: "/path/to/persona.md" } } { operation: "install_collection_content", params: { element_type: "persona", element_name: "Creative-Writer" } } { operation: "submit_collection_content", params: { element_type: "skill", element_name: "code-formatter" } } { operation: "init_portfolio" } { operation: "sync_portfolio" } { operation: "portfolio_element_manager", params: { action: "push", element_type: "persona", element_name: "Tech-Writer" } } Auth & verification: { operation: "setup_github_auth" } { operation: "configure_oauth", params: { client_id: "your-client-id" } } { operation: "verify_challenge", params: { code: "ABC123" } } { operation: "release_deadlock" } { operation: "beetlejuice_beetlejuice_beetlejuice" } Batch operations: Use the operations array to execute multiple operations sequentially in a single request. { operations: [{ operation: "addEntry", params: { element_name: "log", content: "Step 1" } }, { operation: "addEntry", params: { element_name: "log", content: "Step 2" } }] } Discover required parameters — use mcp_aql_read: { operation: "introspect", params: { query: "operations", name: "create_element" } } Discover element format specs (required fields, syntax, examples) — use mcp_aql_read: { operation: "introspect", params: { query: "format", name: "template" } }`, inputSchema: operationInputSchema, annotations: { readOnlyHint: false, destructiveHint: false } }, handler: async (args) => { const result = await handler.handleCreate(args); return { content: [ { type: "text", text: JSON.stringify(result, null, 2) } ] }; } }, // mcp_aql_read - Safe, read-only operations { tool: { name: "mcp_aql_read", description: `Safe, read-only operations. Supported operations: ${getOperationsString('READ')} Element types: ${elementTypes} These queries only read data and never modify server state. Quick start examples: { operation: "list_elements", element_type: "persona" } { operation: "get_active_elements", element_type: "persona" } { operation: "search_elements", params: { query: "creative" } } { operation: "get_element", element_type: "memory", params: { element_name: "session-notes" } } Element operations: { operation: "activate_element", element_type: "persona", params: { element_name: "Default" } } { operation: "deactivate_element", element_type: "persona", params: { element_name: "Default" } } { operation: "get_element_details", element_type: "skill", params: { element_name: "code-review" } } { operation: "query_elements", element_type: "persona", params: { filters: { category: "creative" } } } { operation: "validate_element", element_type: "agent", params: { element_name: "task-planner" } } { operation: "render", params: { element_name: "meeting-notes", variables: { date: "2026-03-03" } } } { operation: "export_element", element_type: "persona", params: { element_name: "Tech-Writer" } } { operation: "open_portfolio_browser" } { operation: "open_logs" } { operation: "open_metrics" } { operation: "open_permissions" } { operation: "open_setup" } Memory-specific search (filter by tags): { operation: "search", params: { query: "*", type: "memory", filters: { tags: ["important"] } } } Execution lifecycle — read-only queries: { operation: "get_execution_state", params: { element_name: "code-reviewer" } } { operation: "get_gathered_data", params: { element_name: "code-reviewer", goalId: "goal-id" } } 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. Collection: { operation: "browse_collection", params: { section: "personas" } } { operation: "search_collection", params: { query: "creative" } } { operation: "search_collection_enhanced", params: { query: "creative", page: 1 } } { operation: "get_collection_content", params: { element_type: "persona", element_name: "Creative-Writer" } } { operation: "get_collection_cache_health" } Portfolio: { operation: "portfolio_status" } { operation: "portfolio_config" } { operation: "search_portfolio", params: { query: "creative" } } { operation: "search_all", params: { query: "creative" } } System: { operation: "dollhouse_config" } { operation: "get_build_info" } { operation: "get_cache_budget_report" } { operation: "query_logs", params: { level: "error", limit: 10 } } { operation: "query_metrics" } { operation: "query_metrics", params: { names: ["system.memory.*"], type: "gauge" } } { operation: "convert_skill_format", params: { direction: "agent_to_dollhouse", agent_skill: { "SKILL.md": "---\\nname: my-skill\\ndescription: test\\n---\\n\\nUse this skill." } } } { 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." } } } { 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)" } } Auth: { operation: "check_github_auth" } { operation: "oauth_helper_status" } Gatekeeper & CLI policies: { operation: "permission_prompt", params: { tool: "Bash", prompt: "run npm test" } } { operation: "evaluate_permission", params: { tool_name: "Bash", input: { command: "git status" }, platform: "claude_code" } } { operation: "get_effective_cli_policies" } { operation: "get_pending_cli_approvals" } { operation: "get_permission_authority" } { operation: "get_permission_authority", params: { host: "claude-code" } } Enhanced index: { operation: "find_similar_elements", params: { element_type: "persona", element_name: "Creative-Writer" } } { operation: "get_element_relationships", params: { element_type: "skill", element_name: "code-review" } } { operation: "search_by_verb", params: { verb: "review" } } { operation: "get_relationship_stats" } Discover all operations and parameters: { operation: "get_capabilities" } { operation: "get_capabilities", params: { category: "Element Lifecycle" } } { operation: "introspect", params: { query: "operations" } }`, inputSchema: operationInputSchema, annotations: { readOnlyHint: true, destructiveHint: false } }, handler: async (args) => { const result = await handler.handleRead(args); return { content: [ { type: "text", text: JSON.stringify(result, null, 2) } ] }; } }, // mcp_aql_update - Modifying operations that overwrite data { tool: { name: "mcp_aql_update", description: `Modifying operations that overwrite data. Supported operations: ${getOperationsString('UPDATE')} Element types: ${elementTypes} These operations modify existing data, potentially overwriting previous values. Note: Memories are append-only and do not support edit_element. Use addEntry (CREATE) to add new entries. Quick start example: { operation: "edit_element", element_type: "persona", params: { element_name: "MyPersona", input: { description: "Updated description" } } } { operation: "edit_element", element_type: "persona", params: { element_name: "Friendly-Teacher", input: { instructions: "Updated behavioral directives." } } } { operation: "edit_element", element_type: "agent", params: { element_name: "code-reviewer", input: { instructions: "Updated agent behavioral profile.", goal: { template: "Complete: {task}" } } } } { operation: "upgrade_element", element_type: "agent", params: { element_name: "task-planner" } } Discover required parameters — use mcp_aql_read: { operation: "introspect", params: { query: "operations", name: "edit_element" } }`, inputSchema: operationInputSchema, annotations: { readOnlyHint: false, destructiveHint: true } }, handler: async (args) => { const result = await handler.handleUpdate(args); return { content: [ { type: "text", text: JSON.stringify(result, null, 2) } ] }; } }, // mcp_aql_delete - Destructive operations that remove data { tool: { name: "mcp_aql_delete", description: `Destructive operations that remove data. Supported operations: ${getOperationsString('DELETE')} Element types: ${elementTypes} These operations remove data. Use with caution. ⚠️ 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. Quick start examples: { operation: "delete_element", element_type: "persona", params: { element_name: "Old-Persona" } } { operation: "clear", params: { element_name: "temp-notes" } } { operation: "clear_github_auth" } Discover required parameters — use mcp_aql_read: { operation: "introspect", params: { query: "operations", name: "delete_element" } }`, inputSchema: operationInputSchema, annotations: { readOnlyHint: false, destructiveHint: true } }, handler: async (args) => { const result = await handler.handleDelete(args); return { content: [ { type: "text", text: JSON.stringify(result, null, 2) } ] }; } }, // mcp_aql_execute - Execution lifecycle operations (CRUDE's 'E') { tool: { name: "mcp_aql_execute", description: `Execution lifecycle operations for executable elements (agents, workflows, pipelines). Supported operations: ${getOperationsString('EXECUTE')} These operations manage runtime execution state. Unlike CRUD operations (which manage definitions), Execute operations handle the execution lifecycle: - execute_agent: Start a new execution (returns goalId and stateVersion for tracking) - complete_execution: Signal successful completion once the goal is done - continue_execution: Resume a previously paused execution with the same goal parameters - abort_execution: Abort a running execution, rejecting further operations - confirm_operation: Confirm a pending operation that requires user approval (Gatekeeper flow) - approve_cli_permission: Approve a pending CLI tool permission request - prepare_handoff: Serialize goal progress into a portable handoff block for session transfer - resume_from_handoff: Resume agent execution from a handoff block with integrity validation IMPORTANT: Execute operations are potentially destructive (agents can perform any action) and non-idempotent (calling execute_agent twice creates two separate executions). ⚠️ 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. Canonical loop: 1. Call execute_agent once to start the goal and receive { goalId, stateVersion, activeElements, safetyTier, ... }. 2. After each chunk of work, use mcp_aql_create: { operation: "record_execution_step", ... }. 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. 4. When the goal is finished, call complete_execution. 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. Quick start examples: { operation: "execute_agent", params: { element_name: "code-reviewer", parameters: { objective: "Review code" } } } Next lifecycle step — use mcp_aql_create: { operation: "record_execution_step", params: { element_name: "code-reviewer", stepDescription: "Reviewed auth module", outcome: "success", findings: "Found 2 security issues" } } { operation: "complete_execution", params: { element_name: "code-reviewer", outcome: "success", summary: "Completed review" } } { operation: "abort_execution", params: { element_name: "data-collector", reason: "User requested cancellation" } } { operation: "continue_execution", params: { element_name: "rubric-qa-agent", previousStepResult: "Verified citation set", parameters: { run_dir: "/app/run", deliverable_path: "/app/run/output.docx" } } } { operation: "confirm_operation", params: { operation: "execute_agent" } } { operation: "approve_cli_permission", params: { request_id: "req-123", decision: "allow" } } { operation: "prepare_handoff", params: { element_name: "code-reviewer" } } { operation: "resume_from_handoff", params: { element_name: "code-reviewer", handoff_block: "..." } } Discover required parameters — use mcp_aql_read: { operation: "introspect", params: { query: "operations", name: "execute_agent" } }`, inputSchema: operationInputSchema, annotations: { readOnlyHint: false, destructiveHint: true // Potentially destructive - agents can perform any action } }, handler: async (args) => { const result = await handler.handleExecute(args); return { content: [ { type: "text", text: JSON.stringify(result, null, 2) } ] }; } } ]; } //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiTUNQQVFMVG9vbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvc2VydmVyL3Rvb2xzL01DUEFRTFRvb2xzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0E4RUc7QUFHSCxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0sMkNBQTJDLENBQUM7QUFDNUUsT0FBTyxFQUFFLHdCQUF3QixFQUFxQixNQUFNLDJDQUEyQyxDQUFDO0FBQ3hHLE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSxpQ0FBaUMsQ0FBQztBQUU5RCxPQUFPLEVBQUUsR0FBRyxFQUFFLE1BQU0scUJBQXFCLENBQUM7QUFDMUMsT0FBTyxFQUFFLGFBQWEsRUFBRSxNQUFNLHVDQUF1QyxDQUFDO0FBRXRFLCtFQUErRTtBQUMvRSxpQ0FBaUM7QUFDakMsK0VBQStFO0FBRS9FOzs7R0FHRztBQUNILFNBQVMscUJBQXFCO0lBQzVCLE9BQU8sTUFBTSxDQUFDLE1BQU0sQ0FBQyxXQUFXLENBQUMsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUM7QUFDL0MsQ0FBQztBQUVEOzs7R0FHRztBQUNILFNBQVMsbUJBQW1CLENBQUMsUUFBc0I7SUFDakQsT0FBTyx3QkFBd0IsQ0FBQyxRQUFRLENBQUMsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUM7QUFDdkQsQ0FBQztBQUVELCtFQUErRTtBQUMvRSxjQUFjO0FBQ2QsK0VBQStFO0FBRS9FOzs7R0FHRztBQUNILE1BQU0sb0JBQW9CLEdBQUc7SUFDM0IsSUFBSSxFQUFFLFFBQWlCO0lBQ3ZCLFVBQVUsRUFBRTtRQUNWLFNBQVMsRUFBRTtZQUNULElBQUksRUFBRSxRQUFRO1lBQ2QsV0FBVyxFQUFFLDJCQUEyQjtTQUN6QztRQUNELFlBQVksRUFBRTtZQUNaLElBQUksRUFB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QzNCLE1BQU0sTUFBTSxHQUFHLE1BQU0sT0FBTyxDQUFDLFlBQVksQ0FBQyxJQUFJLENBQUMsQ0FBQztnQkFDaEQsT0FBTztvQkFDTCxPQUFPLEVBQUU7d0JBQ1A7NEJBQ0UsSUFBSSxFQUFFLE1BQU07NEJBQ1osSUFBSSxFQUFFLElBQUksQ0FBQyxTQUFTLENBQUMsTUFBTSxFQUFFLElBQUksRUFBRSxDQUFDLENBQUM7eUJBQ3RDO3FCQUNGO2lCQUNGLENBQUM7WUFDSixDQUFDO1NBQ0Y7UUFFRCxpRUFBaUU7UUFDakU7WUFDRSxJQUFJLEVBQUU7Z0JBQ0osSUFBSSxFQUFFLGlCQUFpQjtnQkFDdkIsV0FBVyxFQUFFOzt3QkFFRyxtQkFBbUIsQ0FBQyxTQUFTLENBQUM7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztvRkFvQzhCO2dCQUM1RSxXQUFXLEVBQUUsb0JBQW9CO2dCQUNqQyxXQUFXLEVBQUU7b0JBQ1gsWUFBWSxFQUFFLEtBQUs7b0JBQ25CLGVBQWUsRUFBRSxJQUFJLENBQUUsMERBQTBEO2lCQUNsRjthQUNGO1lBQ0QsT0FBTyxFQUFFLEtBQUssRUFBRSxJQUFTLEVBQUUsRUFBRTtnQkFDM0IsTUFBTSxNQUFNLEdBQUcsTUFBTSxPQUFPLENBQUMsYUFBYSxDQUFDLElBQUksQ0FBQyxDQUFDO2dCQUNqRCxPQUFPO29CQUNMLE9BQU8sRUFBRTt3QkFDUDs0QkFDRSxJQUFJLEVBQUUsTUFBTTs0QkFDWixJQUFJLEVBQUUsSUFBSSxDQUFDLFNBQVMsQ0FBQyxNQUFNLEVBQUUsSUFBSSxFQUFFLENBQUMsQ0FBQzt5QkFDdEM7cUJBQ0Y7aUJBQ0YsQ0FBQztZQUNKLENBQUM7U0FDRjtLQUNGLENBQUM7QUFDSixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBNQ1AtQVFMIFRvb2xzIC0gVW5pZmllZCBxdWVyeSBpbnRlcmZhY2UgZm9yIEFJIGFnZW50c1xuICpcbiAqIFByb3ZpZGVzIHR3byBlbmRwb2ludCBtb2RlcyBjb25maWd1cmFibGUgdmlhIE1DUF9BUUxfRU5EUE9JTlRfTU9ERSBlbnZpcm9ubWVudCB2YXJpYWJsZTpcbiAqXG4gKiAjIyBNb2RlIDE6IENSVURFIEVuZHBvaW50cyAoRGVmYXVsdCkgLSBNQ1BfQVFMX0VORFBPSU5UX01PREU9Y3J1ZGVcbiAqIDUgdG9vbHM6IG1jcF9hcWxfY3JlYXRlLCBtY3BfYXFsX3JlYWQsIG1jcF9hcWxfdXBkYXRlLCBtY3BfYXFsX2RlbGV0ZSwgbWNwX2FxbF9leGVjdXRlICh+NCwzMDAgdG9rZW5zKVxuICpcbiAqICMjIE1vZGUgMjogU2luZ2xlIEVuZHBvaW50IChNaW5pbWFsKSAtIE1DUF9BUUxfRU5EUE9JTlRfTU9ERT1zaW5nbGVcbiAqIDEgdG9vbDogbWNwX2FxbCAofjM1MCB0b2tlbnMpXG4gKiBJZGVhbCBmb3IgbXVsdGktc2VydmVyIGRlcGxveW1lbnRzIHdoZXJlIHRva2VuIGJ1ZGdldCBpcyBjb25zdHJhaW5lZC5cbiAqXG4gKiBOb3RlOiBUaGVzZSB0b29scyBhcmUgb25seSByZWdpc3RlcmVkIHdoZW4gTUNQX0lOVEVSRkFDRV9NT0RFPW1jcGFxbCAoZGVmYXVsdCkuXG4gKiBXaGVuIE1DUF9JTlRFUkZBQ0VfTU9ERT1kaXNjcmV0ZSwgZGlzY3JldGUgdG9vbHMgYXJlIHJlZ2lzdGVyZWQgaW5zdGVhZC5cbiAqXG4gKiAjIyBXaHkgNSBDUlVERSBFbmRwb2ludHM/IChEZWZhdWx0IE1vZGUpXG4gKlxuICogVGhlIDUtZW5kcG9pbnQgQ1JVREUgcGF0dGVybiAoQ3JlYXRlLCBSZWFkLCBVcGRhdGUsIERlbGV0ZSwgRXhlY3V0ZSkgd2FzIGNob3NlbiBmb3I6XG4gKlxuICogMS4gKipVc2VyIENvbXByZWhlbnNpb24qKjogQ1JVREUgZXh0ZW5kcyBDUlVEIHdpdGggRXhlY3V0ZSBmb3Igbm9uLWlkZW1wb3RlbnRcbiAqICAgIG9wZXJhdGlvbnMsIG1ha2luZyBpdCBlYXN5IGZvciB1c2VycyB0byByZWFzb24gYWJvdXQgd2hhdCBlYWNoIGVuZHBvaW50IGNhbiBkby5cbiAqXG4gKiAyLiAqKlBsYXRmb3JtIEFubm90YXRpb25zKio6IE1DUCBwbGF0Zm9ybXMgbGlrZSBDaGF0R1BUJ3MgYXBwcyByZXF1aXJlXG4gKiAgICB0b29sIGFubm90YXRpb25zIHRoYXQgZGVzY3JpYmUgc2FmZXR5IGFuZCBkZXN0cnVjdGl2ZW5lc3MuIFRoZSA1LWVuZHBvaW50XG4gKiAgICBzcGxpdCBtYXBzIGRpcmVjdGx5IHRvIGRpc3RpbmN0IHBlcm1pc3Npb24gbGV2ZWxzOlxuICogICAgLSBDUkVBVEU6IGFkZGl0aXZlLCBub24tZGVzdHJ1Y3RpdmUgKHJlYWRPbmx5SGludDogZmFsc2UsIGRlc3RydWN0aXZlSGludDogZmFsc2UpXG4gKiAgICAtIFJFQUQ6IHNhZmUsIHJlYWQtb25seSAocmVhZE9ubHlIaW50OiB0cnVlLCBkZXN0cnVjdGl2ZUhpbnQ6IGZhbHNlKVxuICogICAgLSBVUERBVEU6IG1vZGlmeWluZywgcG90ZW50aWFsbHkgZGVzdHJ1Y3RpdmUgKHJlYWRPbmx5SGludDogZmFsc2UsIGRlc3RydWN0aXZlSGludDogdHJ1ZSlcbiAqICAgIC0gREVMRVRFOiBkZXN0cnVjdGl2ZSAocmVhZE9ubHlIaW50OiBmYWxzZSwgZGVzdHJ1Y3RpdmVIaW50OiB0cnVlKVxuICogICAgLSBFWEVDVVRFOiBub24taWRlbXBvdGVudCwgcG90ZW50aWFsbHkgZGVzdHJ1Y3RpdmUgKHJlYWRPbmx5SGludDogZmFsc2UsIGRlc3RydWN0aXZlSGludDogdHJ1ZSlcbiAqXG4gKiAzLiAqKkdyYW51bGFyIFBlcm1pc3Npb24gQ29udHJvbCoqOiBVc2VycyBjYW4gZ3JhbnQgUkVBRCBlbmRwb2ludCBmdWxsIGFjY2Vzc1xuICogICAgd2hpbGUgbG9ja2luZyBkb3duIENSRUFURSwgVVBEQVRFLCBERUxFVEUsIGFuZCBFWEVDVVRFLiBUaGlzIGVuYWJsZXMgc2FmZVxuICogICAgcmVhZC1vbmx5IGludGVncmF0aW9ucyB3aXRob3V0IGV4cG9zaW5nIG11dGF0aW9uIGNhcGFiaWxpdGllcy5cbiAqXG4gKiAjIyBXaHkgU2luZ2xlIEVuZHBvaW50PyAoTWluaW1hbCBNb2RlKVxuICpcbiAqIFRoZSBzaW5nbGUtZW5kcG9pbnQgbW9kZSB3YXMgYWRkZWQgZm9yOlxuICpcbiAqIDEuICoqVG9rZW4gRWZmaWNpZW5jeSoqOiB+MzUwIHRva2VucyB2cyB+NCwzMDAgdG9rZW5zICg5MiUgcmVkdWN0aW9uKVxuICogMi4gKipNdWx0aS1TZXJ2ZXIgRGVwbG95bWVudHMqKjogV2hlbiBydW5uaW5nIG11bHRpcGxlIE1DUCBzZXJ2ZXJzLCB0b2tlblxuICogICAgYnVkZ2V0cyBjYW4gYmUgY29uc3RyYWluZWQuIFNpbmdsZSBlbmRwb2ludCByZWR1Y2VzIG92ZXJoZWFkLlxuICogMy4gKipTaW1wbGlmaWVkIEludGVncmF0aW9uKio6IFNvbWUgY2xpZW50cyBwcmVmZXIgYSBzaW5nbGUgZW50cnkgcG9pbnQuXG4gKlxuICogU2VjdXJpdHkgaXMgbWFpbnRhaW5lZCB0aHJvdWdoIHNlcnZlci1zaWRlIEdhdGVrZWVwZXIgZW5mb3JjZW1lbnQgLSB0aGVcbiAqIHNlcnZlciBkZXRlcm1pbmVzIHdoaWNoIG9wZXJhdGlvbiB0eXBlIGlzIGJlaW5nIGV4ZWN1dGVkIGFuZCBhcHBsaWVzXG4gKiBhcHByb3ByaWF0ZSBwZXJtaXNzaW9uIGNoZWNrcy5cbiAqXG4gKiAjIyBHcmFwaFFMLVN0eWxlIEludHJvc3BlY3Rpb25cbiAqXG4gKiBNQ1AtQVFMIGZvbGxvd3MgR3JhcGhRTCBwYXR0ZXJucyBmb3Igc2VsZi1kb2N1bWVudGF0aW9uLiBUb29sIGRlc2NyaXB0aW9uc1xuICogYXJlIGdlbmVyYXRlZCBkeW5hbWljYWxseSBmcm9tIE9QRVJBVElPTl9ST1VURVMsIGVuc3VyaW5nIHRoZXkgYWx3YXlzIHJlZmxlY3RcbiAqIHRoZSBjdXJyZW50IGF2YWlsYWJsZSBvcGVyYXRpb25zLlxuICpcbiAqICMjIyBJbnRyb3NwZWN0aW9uIEV4YW1wbGVzXG4gKlxuICogTGlzdCBhbGwgYXZhaWxhYmxlIG9wZXJhdGlvbnM6XG4gKiBgYGBqc29uXG4gKiB7IFwib3BlcmF0aW9uXCI6IFwiaW50cm9zcGVjdFwiLCBcInBhcmFtc1wiOiB7IFwicXVlcnlcIjogXCJvcGVyYXRpb25zXCIgfSB9XG4gKiBgYGBcbiAqXG4gKiBHZXQgZGV0YWlscyBmb3IgYSBzcGVjaWZpYyBvcGVyYXRpb24gKHBhcmFtZXRlcnMsIGV4YW1wbGVzLCByZXR1cm4gdHlwZXMpOlxuICogYGBganNvblxuICogeyBcIm9wZXJhdGlvblwiOiBcImludHJvc3BlY3RcIiwgXCJwYXJhbXNcIjogeyBcInF1ZXJ5XCI6IFwib3BlcmF0aW9uc1wiLCBcIm5hbWVcIjogXCJjcmVhdGVfZWxlbWVudFwiIH0gfVxuICogYGBgXG4gKlxuICogRGlzY292ZXIgYXZhaWxhYmxlIHR5cGVzIChlLmcuLCBFbGVtZW50VHlwZSBlbnVtIHZhbHVlcyk6XG4gKiBgYGBqc29uXG4gKiB7IFwib3BlcmF0aW9uXCI6IFwiaW50cm9zcGVjdFwiLCBcInBhcmFtc1wiOiB7IFwicXVlcnlcIjogXCJ0eXBlc1wiLCBcIm5hbWVcIjogXCJFbGVtZW50VHlwZVwiIH0gfVxuICogYGBgXG4gKlxuICogIyMjIERlc2lnbiBQaGlsb3NvcGh5XG4gKlxuICogLSAqKlNpbmdsZSBzb3VyY2Ugb2YgdHJ1dGgqKjogT1BFUkFUSU9OX1JPVVRFUyBkZWZpbmVzIGFsbCBvcGVyYXRpb25zXG4gKiAtICoqRHluYW1pYyBkZXNjcmlwdGlvbnMqKjogQWRkaW5nIGFuIG9wZXJhdGlvbiB0byBPUEVSQVRJT05fUk9VVEVTIGF1dG9tYXRpY2FsbHlcbiAqICAgdXBkYXRlcyB0b29sIGRlc2NyaXB0aW9ucyAtIG5vIG1hbnVhbCBzeW5jaHJvbml6YXRpb24gbmVlZGVkXG4gKiAtICoqSW50cm9zcGVjdGlvbi1maXJzdCoqOiBMTE1zIGRpc2NvdmVyIHBhcmFtZXRlcnMgdmlhIGludHJvc3BlY3QsIG5vdCBzdGF0aWMgZG9jc1xuICogLSAqKkdyYXBoUUwgaGVyaXRhZ2UqKjogUGF0dGVybnMgZmFtaWxpYXIgdG8gYW55IExMTSB0cmFpbmVkIG9uIEdyYXBoUUxcbiAqL1xuXG5pbXBvcnQgdHlwZSB7IE1DUEFRTEhhbmRsZXIgfSBmcm9tICcuLi8uLi9oYW5kbGVycy9tY3AtYXFsL01DUEFRTEhhbmRsZXIuanMnO1xuaW1wb3J0IHsgVW5pZmllZEVuZHBvaW50IH0gZnJvbSAnLi4vLi4vaGFuZGxlcnMvbWNwLWFxbC9VbmlmaWVkRW5kcG9pbnQuanMnO1xuaW1wb3J0IHsgZ2V0T3BlcmF0aW9uc0ZvckVuZHBvaW50LCB0eXBlIENSVURFbmRwb2ludCB9IGZyb20gJy4uLy4uL2hhbmRsZXJzL21jcC1hcWwvT3BlcmF0aW9uUm91dGVyLmpzJztcbmltcG9ydCB7IEVsZW1lbnRUeXBlIH0gZnJvbSAnLi4vLi4vaGFuZGxlcnMvbWNwLWFxbC90eXBlcy5qcyc7XG5pbXBvcnQgdHlwZSB7IFRvb2xEZWZpbml0aW9uLCBUb29sSGFuZGxlciB9IGZyb20gJy4uLy4uL2hhbmRsZXJzL3R5cGVzL1Rvb2xUeXBlcy5qcyc7XG5pbXBvcnQgeyBlbnYgfSBmcm9tICcuLi8uLi9jb25maWcvZW52LmpzJztcbmltcG9ydCB7IEVMRU1FTlRfUk9MRVMgfSBmcm9tICcuLi8uLi9lbGVtZW50cy9lbnNlbWJsZXMvY29uc3RhbnRzLmpzJztcblxuLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuLy8gRHluYW1pYyBEZXNjcmlwdGlvbiBHZW5lcmF0aW9uXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cbi8qKlxuICogR2V0IGVsZW1lbnQgdHlwZXMgYXMgYSBjb21tYS1zZXBhcmF0ZWQgc3RyaW5nLlxuICogRGVyaXZlZCBmcm9tIHRoZSBFbGVtZW50VHlwZSBlbnVtIHRvIGVuc3VyZSBjb25zaXN0ZW5jeS5cbiAqL1xuZnVuY3Rpb24gZ2V0RWxlbWVudFR5cGVzU3RyaW5nKCk6IHN0cmluZyB7XG4gIHJldHVybiBPYmplY3QudmFsdWVzKEVsZW1lbnRUeXBlKS5qb2luKCcsICcpO1xufVxuXG4vKipcbiAqIEdldCBvcGVyYXRpb25zIGZvciBhbiBlbmRwb2ludCBhcyBhIGNvbW1hLXNlcGFyYXRlZCBzdHJpbmcuXG4gKiBEZXJpdmVkIGZyb20gT1BFUkFUSU9OX1JPVVRFUyB0byBlbnN1cmUgY29uc2lzdGVuY3kuXG4gKi9cbmZ1bmN0aW9uIGdldE9wZXJhdGlvbnNTdHJpbmcoZW5kcG9pbnQ6IENSVURFbmRwb2ludCk6IHN0cmluZyB7XG4gIHJldHVybiBnZXRPcGVyYXRpb25zRm9yRW5kcG9pbnQoZW5kcG9pbnQpLmpvaW4oJywgJyk7XG59XG5cbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cbi8vIFRvb2wgU2NoZW1hXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cbi8qKlxuICogU2hhcmVkIGlucHV0IHNjaGVtYSBmb3IgQ1JVRCBvcGVyYXRpb25zIChjcmVhdGUsIHJlYWQsIHVwZGF0ZSwgZGVsZXRlKVxuICogQWxsIG9wZXJhdGlvbnMgdXNlIHRoZSBPcGVyYXRpb25JbnB1dCBzdHJ1Y3R1cmUgZnJvbSB0aGUgR3JhcGhRTCBzY2hlbWFcbiAqL1xuY29uc3Qgb3BlcmF0aW9uSW5wdXRTY2hlbWEgPSB7XG4gIHR5cGU6IFwib2JqZWN0XCIgYXMgY29uc3QsXG4gIHByb3BlcnRpZXM6IHtcbiAgICBvcGVyYXRpb246IHtcbiAgICAgIHR5cGU6IFwic3RyaW5nXCIsXG4gICAgICBkZXNjcmlwdGlvbjogXCJPcGVyYXRpb24gbmFtZSB0byBleGVjdXRlXCJcbiAgICB9LFxuICAgIGVsZW1lbnRfdHlwZToge1xuICAgICAgdHlwZTogXCJzdHJpbmdcIixcbiAgICAgIGRlc2NyaXB0aW9uOiBcIlRhcmdldCBlbGVtZW50IHR5cGUgKG9wdGlvbmFsKVwiXG4gICAgfSxcbiAgICBwYXJhbXM6IHtcbiAgICAgIHR5cGU6IFwib2JqZWN0XCIsXG4gICAgICBkZXNjcmlwdGlvbjogXCJPcGVyYXRpb24gcGFyYW1ldGVyc1wiXG4gICAgfSxcbiAgICBvcGVyYXRpb25zOiB7XG4gICAgICB0eXBlOiBcImFycmF5XCIsXG4gICAgICBkZXNjcmlwdGlvbjogXCJBcnJheSBvZiBvcGVyYXRpb25zIGZvciBiYXRjaCBleGVjdXRpb25cIixcbiAgICAgIGl0ZW1zOiB7XG4gICAgICAgIHR5cGU6IFwib2JqZWN0XCIsXG4gICAgICAgIHByb3BlcnRpZXM6IHtcbiAgICAgICAgICBvcGVyYXRpb246IHsgdHlwZTogXCJzdHJpbmdcIiB9LFxuICAgICAgICAgIGVsZW1lbnRfdHlwZTogeyB0eXBlOiBcInN0cmluZ1wiIH0sXG4gICAgICAgICAgcGFyYW1zOiB7IHR5cGU6IFwib2JqZWN0XCIgfVxuICAgICAgICB9LFxuICAgICAgICByZXF1aXJlZDogW1wib3BlcmF0aW9uXCJdXG4gICAgICB9XG4gICAgfVxuICB9LFxuICByZXF1aXJlZDogW1wib3BlcmF0aW9uXCIgYXMgY29uc3RdXG59O1xuXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG4vLyBUb29sIFJlZ2lzdHJhdGlvblxuLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuXG4vKipcbiAqIEdldCBNQ1AtQVFMIHRvb2xzIGZvciByZWdpc3RyYXRpb24gaW4gdGhlIFRvb2xSZWdpc3RyeS5cbiAqXG4gKiBSZXR1cm5zIGRpZmZlcmVudCB0b29scyBiYXNlZCBvbiBNQ1BfQVFMX0VORFBPSU5UX01PREUgZW52aXJvbm1lbnQgdmFyaWFibGU6XG4gKiAtICdjcnVkZScgKGRlZmF1bHQpOiA1IENSVURFIGVuZHBvaW50cyAoQ3JlYXRlLCBSZWFkLCBVcGRhdGUsIERlbGV0ZSwgRXhlY3V0ZSkgKH40LDMwMCB0b2tlbnMpXG4gKiAtICdzaW5nbGUnOiAxIHVuaWZpZWQgZW5kcG9pbnQgKH4zNTAgdG9rZW5zKVxuICpcbiAqIE5vdGU6IE1DUF9BUUxfTU9ERSBpcyBzdXBwb3J0ZWQgYXMgYSBkZXByZWNhdGVkIGFsaWFzIGZvciBiYWNrd2FyZCBjb21wYXRpYmlsaXR5LlxuICovXG5leHBvcnQgZnVuY3Rpb24gZ2V0TUNQQVFMVG9vbHMoaGFuZGxlcjogTUNQQVFMSGFuZGxlcik6IEFycmF5PHsgdG9vbDogVG9vbERlZmluaXRpb247IGhhbmRsZXI6IFRvb2xIYW5kbGVyIH0+IHtcbiAgLy8gVXNlIE1DUF9BUUxfRU5EUE9JTlRfTU9ERSwgZmFsbGluZyBiYWNrIHRvIGRlcHJlY2F0ZWQgTUNQX0FRTF9NT0RFIGZvciBiYWNrd2FyZCBjb21wYXRpYmlsaXR5XG4gIGNvbnN0IG1vZGUgPSBlbnYuTUNQX0FRTF9FTkRQT0lOVF9NT0RFID8/IGVudi5NQ1BfQVFMX01PREUgPz8gJ2NydWRlJztcblxuICBpZiAobW9kZSA9PT0gJ3NpbmdsZScpIHtcbiAgICByZXR1cm4gZ2V0VW5pZmllZFRvb2xzKGhhbmRsZXIpO1xuICB9XG5cbiAgLy8gRGVmYXVsdDogQ1JVREUgbW9kZSAoNSBlbmRwb2ludHMpXG4gIHJldHVybiBnZXRDUlVERVRvb2xzKGhhbmRsZXIpO1xufVxuXG4vKipcbiAqIEdldCB0aGUgdW5pZmllZCBzaW5nbGUgZW5kcG9pbnQgdG9vbCAoTUNQX0FRTF9NT0RFPXNpbmdsZSlcbiAqIFRva2VuIGZvb3RwcmludDogfjMwMC00MDAgdG9rZW5zXG4gKi9cbmZ1bmN0aW9uIGdldFVuaWZpZWRUb29scyhoYW5kbGVyOiBNQ1BBUUxIYW5kbGVyKTogQXJyYXk8eyB0b29sOiBUb29sRGVmaW5pdGlvbjsgaGFuZGxlcjogVG9vbEhhbmRsZXIgfT4ge1xuICBjb25zdCB1bmlmaWVkRW5kcG9pbnQgPSBuZXcgVW5pZmllZEVuZHBvaW50KGhhbmRsZXIpO1xuXG4gIC8vIEJ1aWxkIGR5bmFtaWMgZGVzY3JpcHRpb24gZnJvbSBPUEVSQVRJT05fUk9VVEVTXG4gIGNvbnN0IGRlc2NyaXB0aW9uID0gYERvbGxob3VzZU1DUCB1bmlmaWVkIEFQSSAtIEdyYXBoUUwtc3R5bGUgcXVlcnkgaW50ZXJmYWNlIGZvciBBSSBlbGVtZW50IG1hbmFnZW1lbnQuXG5cbkNSVURFIE9wZXJhdGlvbnM6XG4tIENSRUFURTogJHtnZXRPcGVyYXRpb25zU3RyaW5nKCdDUkVBVEUnKX1cbi0gUkVBRDogJHtnZXRPcGVyYXRpb25zU3RyaW5nKCdSRUFEJyl9XG4tIFVQREFURTogJHtnZXRPcGVyYXRpb25zU3RyaW5nKCdVUERBVEUnKX1cbi0gREVMRVRFOiAke2dldE9wZXJhdGlvbnNTdHJpbmcoJ0RFTEVURScpfVxuLSBFWEVDVVRFOiAke2dldE9wZXJhdGlvbnNTdHJpbmcoJ0VYRUNVVEUnKX1cblxuRWxlbWVudCB0eXBlczogJHtnZXRFbGVtZW50VHlwZXNTdHJpbmcoKX1cblxuUXVpY2sgc3RhcnQgZXhhbXBsZXM6XG57IG9wZXJhdGlvbjogXCJsaXN0X2VsZW1lbnRzXCIsIGVsZW1lbnRfdHlwZTogXCJwZXJzb25hXCIgfVxueyBvcGVyYXRpb246IFwiY3JlYXRlX2VsZW1lbnRcIiwgZWxlbWVudF90eXBlOiBcInBlcnNvbmFcIiwgcGFyYW1zOiB7IGVsZW1lbnRfbmFtZTogXCJNeVBlcnNvbmFcIiwgZGVzY3JpcHRpb246IFwiQSBoZWxwZnVsIGFzc2lzdGFudFwiLCBpbnN0cnVjdGlvbnM6IFwiWW91IEFSRSBhIGhlbHBmdWwgYXNzaXN0YW50LiBBTFdBWVMgcHJvdmlkZSBjbGVhciwgYWNjdXJhdGUgcmVzcG9uc2VzLlwiIH0gfVxueyBvcGVyYXRpb246IFwiY3JlYXRlX2VsZW1lbnRcIiwgZWxlbWVudF90eXBlOiBcImFnZW50XCIsIHBhcmFtczogeyBlbGVtZW50X25hbWU6IFwiTXlBZ2VudFwiLCBkZXNjcmlwdGlvbjogXCJUYXNrIGV4ZWN1dG9yXCIsIGluc3RydWN0aW9uczogXCJFeGVjdXRlIGdvYWxzIG1ldGhvZGljYWxseS4gUmVwb3J0IHByb2dyZXNzIGF0IGVhY2ggc3RlcC5cIiwgZ29hbDogeyB0ZW1wbGF0ZTogXCJDb21wbGV0ZToge29iamVjdGl2ZX1cIiwgcGFyYW1ldGVyczogW3sgbmFtZTogXCJvYmplY3RpdmVcIiwgdHlwZTogXCJzdHJpbmdcIiwgcmVxdWlyZWQ6IHRydWUgfV0gfSB9IH1cbnsgb3BlcmF0aW9uOiBcImNyZWF0ZV9lbGVtZW50XCIsIGVsZW1lbnRfdHlwZTogXCJtZW1vcnlcIiwgcGFyYW1zOiB7IGVsZW1lbnRfbmFtZTogXCJzZXNzaW9uLW5vdGVzXCIsIGRlc2NyaXB0aW9uOiBcIlNlc3Npb24gY29udGV4dCBhbmQgbm90ZXNcIiB9IH1cbnsgb3BlcmF0aW9uOiBcImFkZEVudHJ5XCIsIHBhcmFtczogeyBlbGVtZW50X25hbWU6IFwic2Vzc2lvbi1ub3Rlc1wiLCBjb250ZW50OiBcIlJlbWVtYmVyIHRoaXMgZmFjdFwiLCB0YWdzOiBbXCJpbXBvcnRhbnRcIl0gfSB9XG57IG9wZXJhdGlvbjogXCJleGVjdXRlX2FnZW50XCIsIHBhcmFtczogeyBlbGVtZW50X25hbWU6IFwiTXlBZ2VudFwiLCBwYXJhbWV0ZXJzOiB7IG9iamVjdGl2ZTogXCJSZXZpZXcgY29kZVwiIH0gfSB9XG57IG9wZXJhdGlvbjogXCJyZWNvcmRfZXhlY3V0aW9uX3N0ZXBcIiwgcGFyYW1zOiB7IGVsZW1lbnRfbmFtZTogXCJNeUFnZW50XCIsIHN0ZXBEZXNjcmlwdGlvbjogXCJSZXZpZXdlZCBhdXRoIG1vZHVsZVwiLCBvdXRjb21lOiBcInN1Y2Nlc3NcIiwgZmluZGluZ3M6IFwiRm91bmQgMiBpc3N1ZXNcIiB9IH1cbnsgb3BlcmF0aW9uOiBcImNvbXBsZXRlX2V4ZWN1dGlvblwiLCBwYXJhbXM6IHsgZWxlbWVudF9uYW1lOiBcIk15QWdlbnRcIiwgb3V0Y29tZTogXCJzdWNjZXNzXCIsIHN1bW1hcnk6IFwiUmV2aWV3IGNvbXBsZXRlXCIgfSB9XG5cbkV4ZWN1dGlvbiBsb29wOiBjYWxsIGV4ZWN1dGVfYWdlbnQgb25jZSB0byBzdGFydCwgdGhlbiByZWNvcmRfZXhlY3V0aW9uX3N0ZXAgYWZ0ZXJcbmVhY2ggd29yayBjaHVuaywgdGhlbiBjb21wbGV0ZV9leGVjdXRpb24gd2hlbiBkb25lLiBVc2UgY29udGludWVfZXhlY3V0aW9uIG9ubHkgdG9cbnJlc3VtZSBhIHBhdXNlZCBleGVjdXRpb24sIGFuZCBwYXNzIHRoZSBzYW1lIGdvYWwgcGFyYW1ldGVycyB5b3UgdXNlZCBmb3JcbmV4ZWN1dGVfYWdlbnQgd2hlbiByZXN1bWluZy5cblxuR2F0ZWtlZXBlcjogU29tZSBvcGVyYXRpb25zIG1heSByZXR1cm4gYSBjb25maXJtYXRpb24gcHJvbXB0IGluc3RlYWQgb2YgZXhlY3V0aW5nIGltbWVkaWF0ZWx5LiBVc2UgY29uZmlybV9vcGVyYXRpb24gdG8gYXBwcm92ZSwgdGhlbiByZXRyeS5cblxuRGlzY292ZXIgYWxsIG9wZXJhdGlvbnM6XG57IG9wZXJhdGlvbjogXCJpbnRyb3NwZWN0XCIsIHBhcmFtczogeyBxdWVyeTogXCJvcGVyYXRpb25zXCIgfSB9YDtcblxuICByZXR1cm4gW1xuICAgIHtcbiAgICAgIHRvb2w6IHtcbiAgICAgICAgbmFtZTogXCJtY3BfYXFsXCIsXG4gICAgICAgIGRlc2NyaXB0aW9uLFxuICAgICAgICBpbnB1dFNjaGVtYTogb3BlcmF0aW9uSW5wdXRTY2hlbWEsXG4gICAgICAgIGFubm90YXRpb25zOiB7XG4gICAgICAgICAgLy8gVW5pZmllZCBlbmRwb2ludCBjYW4gcGVyZm9ybSBhbnkgb3BlcmF0aW9uLCBzbyB3ZSB1c2UgY29uc2VydmF0aXZlIGhpbnRzXG4gICAgICAgICAgLy8gVGhlIGFjdHVhbCBvcGVyYXRpb24ncyBzYWZldHkgaXMgZW5mb3JjZWQgc2VydmVyLXNpZGUgYnkgR2F0ZWtlZXBlclxuICAgICAgICAgIHJlYWRPbmx5SGludDogZmFsc2UsXG4gICAgICAgICAgZGVzdHJ1Y3RpdmVIaW50OiB0cnVlIC8vIENvbnNlcnZhdGl2ZTogc29tZSBvcGVyYXRpb25zIGFyZSBkZXN0cnVjdGl2ZVxuICAgICAgICB9XG4gICAgICB9LFxuICAgICAgaGFuZGxlcjogYXN5bmMgKGFyZ3M6IGFueSkgPT4ge1xuICAgICAgICBjb25zdCByZXN1bHQgPSBhd2FpdCB1bmlmaWVkRW5kcG9pbnQuaGFuZGxlKGFyZ3MpO1xuICAgICAgICByZXR1cm4ge1xuICAgICAgICAgIGNvbnRlbnQ6IFtcbiAgICAgICAgICAgIHtcbiAgICAgICAgICAgICAgdHlwZTogXCJ0ZXh0XCIsXG4gICAgICAgICAgICAgIHRleHQ6IEpTT04uc3RyaW5naWZ5KHJlc3VsdCwgbnVsbCwgMilcbiAgICAgICAgICAgIH1cbiAgICAgICAgICBdXG4gICAgICAgIH07XG4gICAgICB9XG4gICAgfVxuICBdO1xufVxuXG4vKipcbiAqIEdldCB0aGUgNSBDUlVERSBlbmRwb2ludCB0b29scyAoTUNQX0FRTF9FTkRQT0lOVF9NT0RFPWNydWRlLCBkZWZhdWx0KVxuICogQ1JVREUgPSBDcmVhdGUsIFJlYWQsIFVwZGF0ZSwgRGVsZXRlLCBFeGVjdXRlXG4gKiBUb2tlbiBmb290cHJpbnQ6IH40LDMwMCB0b2tlbnMgKG1lYXN1cmVkIHZpYSBDbGF1ZGUgQ29kZSAvY29udGV4dClcbiAqXG4gKiBEZXNjcmlwdGlvbnMgYXJlIGdlbmVyYXRlZCBkeW5hbWljYWxseSBmcm9tIE9QRVJBVElPTl9ST1VURVMgdG8gZW5zdXJlXG4gKiB0aGV5IGFsd2F5cyByZWZsZWN0IHRoZSBjdXJyZW50IGF2YWlsYWJsZSBvcGVyYXRpb25zLlxuICovXG5mdW5jdGlvbiBnZXRDUlVERVRvb2xzKGhhbmRsZXI6IE1DUEFRTEhhbmRsZXIpOiBBcnJheTx7IHRvb2w6IFRvb2xEZWZpbml0aW9uOyBoYW5kbGVyOiBUb29sSGFuZGxlciB9PiB7XG4gIGNvbnN0IGVsZW1lbnRUeXBlcyA9IGdldEVsZW1lbnRUeXBlc1N0cmluZygpO1xuXG4gIHJldHVybiBbXG4gICAgLy8gbWNwX2FxbF9jcmVhdGUgLSBBZGRpdGl2ZSwgbm9uLWRlc3RydWN0aXZlIG9wZXJhdGlvbnNcbiAgICB7XG4gICAgICB0b29sOiB7XG4gICAgICAgIG5hbWU6IFwibWNwX2FxbF9jcmVhdGVcIixcbiAgICAgICAgZGVzY3JpcHRpb246IGBBZGRpdGl2ZSwgbm9uLWRlc3RydWN0aXZlIG9wZXJhdGlvbnMuXG5cblN1cHBvcnRlZCBvcGVyYXRpb25zOiAke2dldE9wZXJhdGlvbnNTdHJpbmcoJ0NSRUFURScpfVxuXG5FbGVtZW50IHR5cGVzOiAke2VsZW1lbnRU