claude-mpm 4.21.0__py3-none-any.whl → 5.0.2__py3-none-any.whl
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.
- claude_mpm/VERSION +1 -1
- claude_mpm/agents/BASE_PM.md +12 -0
- claude_mpm/agents/OUTPUT_STYLE.md +3 -48
- claude_mpm/agents/PM_INSTRUCTIONS.md +632 -334
- claude_mpm/agents/WORKFLOW.md +75 -2
- claude_mpm/agents/__init__.py +6 -0
- claude_mpm/agents/agent_loader.py +1 -4
- claude_mpm/agents/base_agent.json +6 -3
- claude_mpm/agents/frontmatter_validator.py +1 -1
- claude_mpm/agents/templates/{circuit_breakers.md → circuit-breakers.md} +370 -3
- claude_mpm/agents/templates/context-management-examples.md +544 -0
- claude_mpm/agents/templates/{pm_red_flags.md → pm-red-flags.md} +89 -19
- claude_mpm/agents/templates/pr-workflow-examples.md +427 -0
- claude_mpm/agents/templates/research-gate-examples.md +669 -0
- claude_mpm/agents/templates/structured-questions-examples.md +615 -0
- claude_mpm/agents/templates/ticket-completeness-examples.md +139 -0
- claude_mpm/agents/templates/ticketing-examples.md +277 -0
- claude_mpm/cli/__init__.py +38 -2
- claude_mpm/cli/commands/agent_source.py +774 -0
- claude_mpm/cli/commands/agent_state_manager.py +125 -20
- claude_mpm/cli/commands/agents.py +684 -13
- claude_mpm/cli/commands/agents_cleanup.py +210 -0
- claude_mpm/cli/commands/agents_discover.py +338 -0
- claude_mpm/cli/commands/aggregate.py +1 -1
- claude_mpm/cli/commands/analyze.py +3 -3
- claude_mpm/cli/commands/auto_configure.py +2 -6
- claude_mpm/cli/commands/cleanup.py +1 -1
- claude_mpm/cli/commands/config.py +7 -4
- claude_mpm/cli/commands/configure.py +478 -44
- claude_mpm/cli/commands/configure_agent_display.py +4 -4
- claude_mpm/cli/commands/configure_navigation.py +63 -46
- claude_mpm/cli/commands/debug.py +12 -12
- claude_mpm/cli/commands/doctor.py +10 -2
- claude_mpm/cli/commands/hook_errors.py +277 -0
- claude_mpm/cli/commands/local_deploy.py +1 -4
- claude_mpm/cli/commands/mcp_install_commands.py +1 -1
- claude_mpm/cli/commands/mpm_init/core.py +50 -2
- claude_mpm/cli/commands/mpm_init/git_activity.py +10 -10
- claude_mpm/cli/commands/mpm_init/prompts.py +6 -6
- claude_mpm/cli/commands/run.py +124 -128
- claude_mpm/cli/commands/skill_source.py +694 -0
- claude_mpm/cli/commands/skills.py +435 -1
- claude_mpm/cli/executor.py +78 -3
- claude_mpm/cli/interactive/agent_wizard.py +919 -41
- claude_mpm/cli/parsers/agent_source_parser.py +171 -0
- claude_mpm/cli/parsers/agents_parser.py +173 -4
- claude_mpm/cli/parsers/base_parser.py +49 -0
- claude_mpm/cli/parsers/config_parser.py +96 -43
- claude_mpm/cli/parsers/skill_source_parser.py +169 -0
- claude_mpm/cli/parsers/skills_parser.py +138 -0
- claude_mpm/cli/parsers/source_parser.py +138 -0
- claude_mpm/cli/startup.py +499 -84
- claude_mpm/cli/startup_display.py +480 -0
- claude_mpm/cli/utils.py +1 -1
- claude_mpm/cli_module/commands.py +1 -1
- claude_mpm/commands/{mpm-auto-configure.md → mpm-agents-auto-configure.md} +9 -0
- claude_mpm/commands/mpm-agents-detect.md +9 -0
- claude_mpm/commands/{mpm-agents.md → mpm-agents-list.md} +9 -0
- claude_mpm/commands/mpm-agents-recommend.md +9 -0
- claude_mpm/commands/{mpm-config.md → mpm-config-view.md} +9 -0
- claude_mpm/commands/mpm-doctor.md +9 -0
- claude_mpm/commands/mpm-help.md +14 -2
- claude_mpm/commands/mpm-init.md +27 -2
- claude_mpm/commands/mpm-monitor.md +9 -0
- claude_mpm/commands/mpm-session-resume.md +381 -0
- claude_mpm/commands/mpm-status.md +9 -0
- claude_mpm/commands/{mpm-organize.md → mpm-ticket-organize.md} +9 -0
- claude_mpm/commands/mpm-ticket-view.md +552 -0
- claude_mpm/commands/mpm-version.md +9 -0
- claude_mpm/commands/mpm.md +11 -0
- claude_mpm/config/agent_presets.py +258 -0
- claude_mpm/config/agent_sources.py +325 -0
- claude_mpm/config/skill_sources.py +590 -0
- claude_mpm/constants.py +12 -0
- claude_mpm/core/api_validator.py +1 -1
- claude_mpm/core/claude_runner.py +17 -10
- claude_mpm/core/config.py +24 -0
- claude_mpm/core/constants.py +1 -1
- claude_mpm/core/framework/__init__.py +3 -16
- claude_mpm/core/framework/loaders/instruction_loader.py +25 -5
- claude_mpm/core/framework/processors/metadata_processor.py +1 -1
- claude_mpm/core/hook_error_memory.py +381 -0
- claude_mpm/core/hook_manager.py +41 -2
- claude_mpm/core/interactive_session.py +112 -5
- claude_mpm/core/logger.py +3 -1
- claude_mpm/core/oneshot_session.py +94 -4
- claude_mpm/dashboard/static/css/activity.css +69 -69
- claude_mpm/dashboard/static/css/connection-status.css +10 -10
- claude_mpm/dashboard/static/css/dashboard.css +15 -15
- claude_mpm/dashboard/static/js/components/activity-tree.js +178 -178
- claude_mpm/dashboard/static/js/components/agent-hierarchy.js +101 -101
- claude_mpm/dashboard/static/js/components/agent-inference.js +31 -31
- claude_mpm/dashboard/static/js/components/build-tracker.js +59 -59
- claude_mpm/dashboard/static/js/components/code-simple.js +107 -107
- claude_mpm/dashboard/static/js/components/connection-debug.js +101 -101
- claude_mpm/dashboard/static/js/components/diff-viewer.js +113 -113
- claude_mpm/dashboard/static/js/components/event-viewer.js +12 -12
- claude_mpm/dashboard/static/js/components/file-change-tracker.js +57 -57
- claude_mpm/dashboard/static/js/components/file-change-viewer.js +74 -74
- claude_mpm/dashboard/static/js/components/file-tool-tracker.js +6 -6
- claude_mpm/dashboard/static/js/components/file-viewer.js +42 -42
- claude_mpm/dashboard/static/js/components/module-viewer.js +27 -27
- claude_mpm/dashboard/static/js/components/session-manager.js +14 -14
- claude_mpm/dashboard/static/js/components/socket-manager.js +1 -1
- claude_mpm/dashboard/static/js/components/ui-state-manager.js +14 -14
- claude_mpm/dashboard/static/js/components/unified-data-viewer.js +110 -110
- claude_mpm/dashboard/static/js/components/working-directory.js +8 -8
- claude_mpm/dashboard/static/js/connection-manager.js +76 -76
- claude_mpm/dashboard/static/js/dashboard.js +76 -58
- claude_mpm/dashboard/static/js/extension-error-handler.js +22 -22
- claude_mpm/dashboard/static/js/socket-client.js +138 -121
- claude_mpm/dashboard/templates/code_simple.html +23 -23
- claude_mpm/dashboard/templates/index.html +18 -18
- claude_mpm/experimental/cli_enhancements.py +1 -5
- claude_mpm/hooks/claude_hooks/event_handlers.py +3 -1
- claude_mpm/hooks/claude_hooks/hook_handler.py +24 -7
- claude_mpm/hooks/claude_hooks/installer.py +45 -0
- claude_mpm/hooks/failure_learning/__init__.py +2 -8
- claude_mpm/hooks/failure_learning/failure_detection_hook.py +1 -6
- claude_mpm/hooks/failure_learning/fix_detection_hook.py +1 -6
- claude_mpm/hooks/failure_learning/learning_extraction_hook.py +1 -6
- claude_mpm/hooks/kuzu_response_hook.py +1 -5
- claude_mpm/hooks/templates/pre_tool_use_simple.py +78 -0
- claude_mpm/hooks/templates/pre_tool_use_template.py +323 -0
- claude_mpm/models/git_repository.py +198 -0
- claude_mpm/scripts/claude-hook-handler.sh +3 -3
- claude_mpm/scripts/start_activity_logging.py +3 -1
- claude_mpm/services/agents/agent_builder.py +45 -9
- claude_mpm/services/agents/agent_preset_service.py +238 -0
- claude_mpm/services/agents/agent_selection_service.py +484 -0
- claude_mpm/services/agents/auto_deploy_index_parser.py +569 -0
- claude_mpm/services/agents/deployment/agent_deployment.py +126 -2
- claude_mpm/services/agents/deployment/agent_discovery_service.py +105 -73
- claude_mpm/services/agents/deployment/agent_format_converter.py +1 -1
- claude_mpm/services/agents/deployment/agent_lifecycle_manager.py +1 -5
- claude_mpm/services/agents/deployment/agent_metrics_collector.py +3 -3
- claude_mpm/services/agents/deployment/agent_restore_handler.py +1 -4
- claude_mpm/services/agents/deployment/agent_template_builder.py +236 -15
- claude_mpm/services/agents/deployment/agents_directory_resolver.py +101 -15
- claude_mpm/services/agents/deployment/async_agent_deployment.py +2 -1
- claude_mpm/services/agents/deployment/facade/deployment_facade.py +3 -3
- claude_mpm/services/agents/deployment/multi_source_deployment_service.py +115 -15
- claude_mpm/services/agents/deployment/pipeline/pipeline_executor.py +2 -2
- claude_mpm/services/agents/deployment/refactored_agent_deployment_service.py +1 -4
- claude_mpm/services/agents/deployment/remote_agent_discovery_service.py +363 -0
- claude_mpm/services/agents/deployment/single_agent_deployer.py +2 -2
- claude_mpm/services/agents/deployment/system_instructions_deployer.py +168 -46
- claude_mpm/services/agents/deployment/validation/deployment_validator.py +2 -2
- claude_mpm/services/agents/git_source_manager.py +629 -0
- claude_mpm/services/agents/loading/framework_agent_loader.py +9 -12
- claude_mpm/services/agents/local_template_manager.py +50 -10
- claude_mpm/services/agents/single_tier_deployment_service.py +696 -0
- claude_mpm/services/agents/sources/__init__.py +13 -0
- claude_mpm/services/agents/sources/agent_sync_state.py +516 -0
- claude_mpm/services/agents/sources/git_source_sync_service.py +1055 -0
- claude_mpm/services/agents/startup_sync.py +239 -0
- claude_mpm/services/agents/toolchain_detector.py +474 -0
- claude_mpm/services/cli/session_pause_manager.py +1 -1
- claude_mpm/services/cli/unified_dashboard_manager.py +1 -1
- claude_mpm/services/command_deployment_service.py +92 -1
- claude_mpm/services/core/base.py +26 -11
- claude_mpm/services/core/interfaces/__init__.py +1 -3
- claude_mpm/services/core/interfaces/health.py +1 -4
- claude_mpm/services/core/models/__init__.py +2 -11
- claude_mpm/services/diagnostics/checks/__init__.py +4 -0
- claude_mpm/services/diagnostics/checks/agent_check.py +0 -2
- claude_mpm/services/diagnostics/checks/agent_sources_check.py +577 -0
- claude_mpm/services/diagnostics/checks/instructions_check.py +1 -2
- claude_mpm/services/diagnostics/checks/mcp_check.py +0 -1
- claude_mpm/services/diagnostics/checks/monitor_check.py +0 -1
- claude_mpm/services/diagnostics/checks/skill_sources_check.py +587 -0
- claude_mpm/services/diagnostics/diagnostic_runner.py +9 -0
- claude_mpm/services/diagnostics/doctor_reporter.py +40 -10
- claude_mpm/services/event_bus/direct_relay.py +3 -3
- claude_mpm/services/event_bus/event_bus.py +36 -3
- claude_mpm/services/event_bus/relay.py +23 -7
- claude_mpm/services/events/consumers/logging.py +1 -2
- claude_mpm/services/git/__init__.py +21 -0
- claude_mpm/services/git/git_operations_service.py +494 -0
- claude_mpm/services/github/__init__.py +21 -0
- claude_mpm/services/github/github_cli_service.py +397 -0
- claude_mpm/services/infrastructure/monitoring/__init__.py +1 -5
- claude_mpm/services/infrastructure/monitoring/aggregator.py +1 -6
- claude_mpm/services/infrastructure/monitoring/resources.py +1 -1
- claude_mpm/services/instructions/__init__.py +9 -0
- claude_mpm/services/instructions/instruction_cache_service.py +374 -0
- claude_mpm/services/local_ops/__init__.py +3 -13
- claude_mpm/services/local_ops/health_checks/__init__.py +1 -3
- claude_mpm/services/local_ops/health_manager.py +1 -4
- claude_mpm/services/local_ops/process_manager.py +1 -1
- claude_mpm/services/local_ops/resource_monitor.py +2 -2
- claude_mpm/services/mcp_gateway/config/configuration.py +1 -1
- claude_mpm/services/mcp_gateway/server/mcp_gateway.py +1 -6
- claude_mpm/services/mcp_gateway/server/stdio_server.py +0 -2
- claude_mpm/services/mcp_gateway/tools/document_summarizer.py +1 -1
- claude_mpm/services/mcp_gateway/tools/kuzu_memory_service.py +6 -2
- claude_mpm/services/memory/failure_tracker.py +19 -4
- claude_mpm/services/memory/optimizer.py +1 -1
- claude_mpm/services/model/model_router.py +8 -9
- claude_mpm/services/monitor/daemon.py +1 -1
- claude_mpm/services/monitor/server.py +2 -2
- claude_mpm/services/native_agent_converter.py +356 -0
- claude_mpm/services/port_manager.py +1 -1
- claude_mpm/services/pr/__init__.py +14 -0
- claude_mpm/services/pr/pr_template_service.py +329 -0
- claude_mpm/services/project/documentation_manager.py +2 -1
- claude_mpm/services/project/toolchain_analyzer.py +3 -1
- claude_mpm/services/runner_configuration_service.py +1 -0
- claude_mpm/services/self_upgrade_service.py +165 -7
- claude_mpm/services/skills/__init__.py +18 -0
- claude_mpm/services/skills/git_skill_source_manager.py +1169 -0
- claude_mpm/services/skills/skill_discovery_service.py +568 -0
- claude_mpm/services/skills_config.py +547 -0
- claude_mpm/services/skills_deployer.py +955 -0
- claude_mpm/services/socketio/handlers/connection.py +1 -1
- claude_mpm/services/socketio/handlers/git.py +2 -2
- claude_mpm/services/socketio/server/core.py +1 -4
- claude_mpm/services/socketio/server/main.py +1 -3
- claude_mpm/services/system_instructions_service.py +1 -3
- claude_mpm/services/unified/analyzer_strategies/performance_analyzer.py +0 -3
- claude_mpm/services/unified/analyzer_strategies/security_analyzer.py +0 -1
- claude_mpm/services/unified/deployment_strategies/cloud_strategies.py +1 -1
- claude_mpm/services/unified/deployment_strategies/vercel.py +1 -5
- claude_mpm/services/unified/unified_deployment.py +1 -5
- claude_mpm/services/version_control/conflict_resolution.py +6 -4
- claude_mpm/services/visualization/__init__.py +1 -5
- claude_mpm/services/visualization/mermaid_generator.py +2 -3
- claude_mpm/skills/bundled/infrastructure/env-manager/scripts/validate_env.py +576 -0
- claude_mpm/skills/bundled/performance-profiling.md +6 -0
- claude_mpm/skills/bundled/testing/webapp-testing/scripts/with_server.py +2 -2
- claude_mpm/skills/skills_registry.py +0 -1
- claude_mpm/templates/questions/__init__.py +38 -0
- claude_mpm/templates/questions/base.py +193 -0
- claude_mpm/templates/questions/pr_strategy.py +311 -0
- claude_mpm/templates/questions/project_init.py +385 -0
- claude_mpm/templates/questions/ticket_mgmt.py +394 -0
- claude_mpm/tools/__main__.py +8 -8
- claude_mpm/tools/code_tree_analyzer/__init__.py +45 -0
- claude_mpm/tools/code_tree_analyzer/analysis.py +299 -0
- claude_mpm/tools/code_tree_analyzer/cache.py +131 -0
- claude_mpm/tools/code_tree_analyzer/core.py +380 -0
- claude_mpm/tools/code_tree_analyzer/discovery.py +403 -0
- claude_mpm/tools/code_tree_analyzer/events.py +168 -0
- claude_mpm/tools/code_tree_analyzer/gitignore.py +308 -0
- claude_mpm/tools/code_tree_analyzer/models.py +39 -0
- claude_mpm/tools/code_tree_analyzer/multilang_analyzer.py +224 -0
- claude_mpm/tools/code_tree_analyzer/python_analyzer.py +284 -0
- claude_mpm/utils/agent_dependency_loader.py +80 -13
- claude_mpm/utils/dependency_cache.py +3 -1
- claude_mpm/utils/gitignore.py +241 -0
- claude_mpm/utils/log_cleanup.py +3 -3
- claude_mpm/utils/progress.py +383 -0
- claude_mpm/utils/robust_installer.py +3 -5
- claude_mpm/utils/structured_questions.py +619 -0
- {claude_mpm-4.21.0.dist-info → claude_mpm-5.0.2.dist-info}/METADATA +429 -59
- {claude_mpm-4.21.0.dist-info → claude_mpm-5.0.2.dist-info}/RECORD +264 -427
- claude_mpm/agents/templates/.claude-mpm/memories/README.md +0 -17
- claude_mpm/agents/templates/.claude-mpm/memories/engineer_memories.md +0 -3
- claude_mpm/agents/templates/agent-manager.json +0 -273
- claude_mpm/agents/templates/agentic-coder-optimizer.json +0 -248
- claude_mpm/agents/templates/api_qa.json +0 -180
- claude_mpm/agents/templates/clerk-ops.json +0 -235
- claude_mpm/agents/templates/code_analyzer.json +0 -101
- claude_mpm/agents/templates/content-agent.json +0 -358
- claude_mpm/agents/templates/dart_engineer.json +0 -307
- claude_mpm/agents/templates/data_engineer.json +0 -225
- claude_mpm/agents/templates/documentation.json +0 -211
- claude_mpm/agents/templates/engineer.json +0 -210
- claude_mpm/agents/templates/gcp_ops_agent.json +0 -253
- claude_mpm/agents/templates/golang_engineer.json +0 -270
- claude_mpm/agents/templates/imagemagick.json +0 -264
- claude_mpm/agents/templates/java_engineer.json +0 -346
- claude_mpm/agents/templates/local_ops_agent.json +0 -1840
- claude_mpm/agents/templates/logs/prompts/agent_engineer_20250826_014258_728.md +0 -39
- claude_mpm/agents/templates/logs/prompts/agent_engineer_20250901_010124_142.md +0 -400
- claude_mpm/agents/templates/memory_manager.json +0 -158
- claude_mpm/agents/templates/nextjs_engineer.json +0 -285
- claude_mpm/agents/templates/ops.json +0 -185
- claude_mpm/agents/templates/php-engineer.json +0 -287
- claude_mpm/agents/templates/product_owner.json +0 -338
- claude_mpm/agents/templates/project_organizer.json +0 -140
- claude_mpm/agents/templates/prompt-engineer.json +0 -737
- claude_mpm/agents/templates/python_engineer.json +0 -387
- claude_mpm/agents/templates/qa.json +0 -242
- claude_mpm/agents/templates/react_engineer.json +0 -238
- claude_mpm/agents/templates/refactoring_engineer.json +0 -276
- claude_mpm/agents/templates/research.json +0 -188
- claude_mpm/agents/templates/ruby-engineer.json +0 -280
- claude_mpm/agents/templates/rust_engineer.json +0 -275
- claude_mpm/agents/templates/security.json +0 -202
- claude_mpm/agents/templates/svelte-engineer.json +0 -225
- claude_mpm/agents/templates/ticketing.json +0 -177
- claude_mpm/agents/templates/typescript_engineer.json +0 -285
- claude_mpm/agents/templates/vercel_ops_agent.json +0 -412
- claude_mpm/agents/templates/version_control.json +0 -157
- claude_mpm/agents/templates/web_qa.json +0 -399
- claude_mpm/agents/templates/web_ui.json +0 -189
- claude_mpm/commands/mpm-tickets.md +0 -102
- claude_mpm/dashboard/.claude-mpm/socketio-instances.json +0 -1
- claude_mpm/dashboard/react/components/DataInspector/DataInspector.module.css +0 -188
- claude_mpm/dashboard/react/components/EventViewer/EventViewer.module.css +0 -156
- claude_mpm/dashboard/react/components/shared/ConnectionStatus.module.css +0 -38
- claude_mpm/dashboard/react/components/shared/FilterBar.module.css +0 -92
- claude_mpm/dashboard/static/archive/activity_dashboard_fixed.html +0 -248
- claude_mpm/dashboard/static/archive/activity_dashboard_test.html +0 -61
- claude_mpm/dashboard/static/archive/test_activity_connection.html +0 -179
- claude_mpm/dashboard/static/archive/test_claude_tree_tab.html +0 -68
- claude_mpm/dashboard/static/archive/test_dashboard.html +0 -409
- claude_mpm/dashboard/static/archive/test_dashboard_fixed.html +0 -519
- claude_mpm/dashboard/static/archive/test_dashboard_verification.html +0 -181
- claude_mpm/dashboard/static/archive/test_file_data.html +0 -315
- claude_mpm/dashboard/static/archive/test_file_tree_empty_state.html +0 -243
- claude_mpm/dashboard/static/archive/test_file_tree_fix.html +0 -234
- claude_mpm/dashboard/static/archive/test_file_tree_rename.html +0 -117
- claude_mpm/dashboard/static/archive/test_file_tree_tab.html +0 -115
- claude_mpm/dashboard/static/archive/test_file_viewer.html +0 -224
- claude_mpm/dashboard/static/archive/test_final_activity.html +0 -220
- claude_mpm/dashboard/static/archive/test_tab_fix.html +0 -139
- claude_mpm/dashboard/static/built/assets/events.DjpNxWNo.css +0 -1
- claude_mpm/dashboard/static/built/components/activity-tree.js +0 -2
- claude_mpm/dashboard/static/built/components/agent-hierarchy.js +0 -777
- claude_mpm/dashboard/static/built/components/agent-inference.js +0 -2
- claude_mpm/dashboard/static/built/components/build-tracker.js +0 -333
- claude_mpm/dashboard/static/built/components/code-simple.js +0 -857
- claude_mpm/dashboard/static/built/components/code-tree/tree-breadcrumb.js +0 -353
- claude_mpm/dashboard/static/built/components/code-tree/tree-constants.js +0 -235
- claude_mpm/dashboard/static/built/components/code-tree/tree-search.js +0 -409
- claude_mpm/dashboard/static/built/components/code-tree/tree-utils.js +0 -435
- claude_mpm/dashboard/static/built/components/code-tree.js +0 -2
- claude_mpm/dashboard/static/built/components/code-viewer.js +0 -2
- claude_mpm/dashboard/static/built/components/connection-debug.js +0 -654
- claude_mpm/dashboard/static/built/components/diff-viewer.js +0 -891
- claude_mpm/dashboard/static/built/components/event-processor.js +0 -2
- claude_mpm/dashboard/static/built/components/event-viewer.js +0 -2
- claude_mpm/dashboard/static/built/components/export-manager.js +0 -2
- claude_mpm/dashboard/static/built/components/file-change-tracker.js +0 -443
- claude_mpm/dashboard/static/built/components/file-change-viewer.js +0 -690
- claude_mpm/dashboard/static/built/components/file-tool-tracker.js +0 -2
- claude_mpm/dashboard/static/built/components/file-viewer.js +0 -2
- claude_mpm/dashboard/static/built/components/hud-library-loader.js +0 -2
- claude_mpm/dashboard/static/built/components/hud-manager.js +0 -2
- claude_mpm/dashboard/static/built/components/hud-visualizer.js +0 -2
- claude_mpm/dashboard/static/built/components/module-viewer.js +0 -2
- claude_mpm/dashboard/static/built/components/nav-bar.js +0 -145
- claude_mpm/dashboard/static/built/components/page-structure.js +0 -429
- claude_mpm/dashboard/static/built/components/session-manager.js +0 -2
- claude_mpm/dashboard/static/built/components/socket-manager.js +0 -2
- claude_mpm/dashboard/static/built/components/ui-state-manager.js +0 -2
- claude_mpm/dashboard/static/built/components/unified-data-viewer.js +0 -2
- claude_mpm/dashboard/static/built/components/working-directory.js +0 -2
- claude_mpm/dashboard/static/built/connection-manager.js +0 -536
- claude_mpm/dashboard/static/built/dashboard.js +0 -2
- claude_mpm/dashboard/static/built/extension-error-handler.js +0 -164
- claude_mpm/dashboard/static/built/react/events.js +0 -30
- claude_mpm/dashboard/static/built/shared/dom-helpers.js +0 -396
- claude_mpm/dashboard/static/built/shared/event-bus.js +0 -330
- claude_mpm/dashboard/static/built/shared/event-filter-service.js +0 -540
- claude_mpm/dashboard/static/built/shared/logger.js +0 -385
- claude_mpm/dashboard/static/built/shared/page-structure.js +0 -249
- claude_mpm/dashboard/static/built/shared/tooltip-service.js +0 -253
- claude_mpm/dashboard/static/built/socket-client.js +0 -2
- claude_mpm/dashboard/static/built/tab-isolation-fix.js +0 -185
- claude_mpm/dashboard/static/dist/assets/events.DjpNxWNo.css +0 -1
- claude_mpm/dashboard/static/dist/components/activity-tree.js +0 -2
- claude_mpm/dashboard/static/dist/components/agent-inference.js +0 -2
- claude_mpm/dashboard/static/dist/components/code-tree.js +0 -2
- claude_mpm/dashboard/static/dist/components/code-viewer.js +0 -2
- claude_mpm/dashboard/static/dist/components/event-processor.js +0 -2
- claude_mpm/dashboard/static/dist/components/event-viewer.js +0 -2
- claude_mpm/dashboard/static/dist/components/export-manager.js +0 -2
- claude_mpm/dashboard/static/dist/components/file-tool-tracker.js +0 -2
- claude_mpm/dashboard/static/dist/components/file-viewer.js +0 -2
- claude_mpm/dashboard/static/dist/components/hud-library-loader.js +0 -2
- claude_mpm/dashboard/static/dist/components/hud-manager.js +0 -2
- claude_mpm/dashboard/static/dist/components/hud-visualizer.js +0 -2
- claude_mpm/dashboard/static/dist/components/module-viewer.js +0 -2
- claude_mpm/dashboard/static/dist/components/session-manager.js +0 -2
- claude_mpm/dashboard/static/dist/components/socket-manager.js +0 -2
- claude_mpm/dashboard/static/dist/components/ui-state-manager.js +0 -2
- claude_mpm/dashboard/static/dist/components/unified-data-viewer.js +0 -2
- claude_mpm/dashboard/static/dist/components/working-directory.js +0 -2
- claude_mpm/dashboard/static/dist/dashboard.js +0 -2
- claude_mpm/dashboard/static/dist/react/events.js +0 -30
- claude_mpm/dashboard/static/dist/socket-client.js +0 -2
- claude_mpm/dashboard/static/events.html +0 -607
- claude_mpm/dashboard/static/index.html +0 -635
- claude_mpm/dashboard/static/js/shared/dom-helpers.js +0 -396
- claude_mpm/dashboard/static/js/shared/event-bus.js +0 -330
- claude_mpm/dashboard/static/js/shared/logger.js +0 -385
- claude_mpm/dashboard/static/js/shared/tooltip-service.js +0 -253
- claude_mpm/dashboard/static/js/stores/dashboard-store.js +0 -562
- claude_mpm/dashboard/static/legacy/activity.html +0 -736
- claude_mpm/dashboard/static/legacy/agents.html +0 -786
- claude_mpm/dashboard/static/legacy/files.html +0 -747
- claude_mpm/dashboard/static/legacy/tools.html +0 -831
- claude_mpm/dashboard/static/monitors.html +0 -431
- claude_mpm/dashboard/static/production/events.html +0 -659
- claude_mpm/dashboard/static/production/main.html +0 -698
- claude_mpm/dashboard/static/production/monitors.html +0 -483
- claude_mpm/dashboard/static/test-archive/dashboard.html +0 -635
- claude_mpm/dashboard/static/test-archive/debug-events.html +0 -147
- claude_mpm/dashboard/static/test-archive/test-navigation.html +0 -256
- claude_mpm/dashboard/static/test-archive/test-react-exports.html +0 -180
- claude_mpm/dashboard/static/test-archive/test_debug.html +0 -25
- claude_mpm/skills/bundled/collaboration/brainstorming/SKILL.md +0 -79
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/SKILL.md +0 -178
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/agent-prompts.md +0 -577
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/coordination-patterns.md +0 -467
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/examples.md +0 -537
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/troubleshooting.md +0 -730
- claude_mpm/skills/bundled/collaboration/requesting-code-review/SKILL.md +0 -112
- claude_mpm/skills/bundled/collaboration/requesting-code-review/references/code-reviewer-template.md +0 -146
- claude_mpm/skills/bundled/collaboration/requesting-code-review/references/review-examples.md +0 -412
- claude_mpm/skills/bundled/collaboration/writing-plans/SKILL.md +0 -81
- claude_mpm/skills/bundled/collaboration/writing-plans/references/best-practices.md +0 -362
- claude_mpm/skills/bundled/collaboration/writing-plans/references/plan-structure-templates.md +0 -312
- claude_mpm/skills/bundled/debugging/root-cause-tracing/SKILL.md +0 -152
- claude_mpm/skills/bundled/debugging/root-cause-tracing/references/advanced-techniques.md +0 -668
- claude_mpm/skills/bundled/debugging/root-cause-tracing/references/examples.md +0 -587
- claude_mpm/skills/bundled/debugging/root-cause-tracing/references/integration.md +0 -438
- claude_mpm/skills/bundled/debugging/root-cause-tracing/references/tracing-techniques.md +0 -391
- claude_mpm/skills/bundled/debugging/systematic-debugging/CREATION-LOG.md +0 -119
- claude_mpm/skills/bundled/debugging/systematic-debugging/SKILL.md +0 -148
- claude_mpm/skills/bundled/debugging/systematic-debugging/references/anti-patterns.md +0 -483
- claude_mpm/skills/bundled/debugging/systematic-debugging/references/examples.md +0 -452
- claude_mpm/skills/bundled/debugging/systematic-debugging/references/troubleshooting.md +0 -449
- claude_mpm/skills/bundled/debugging/systematic-debugging/references/workflow.md +0 -411
- claude_mpm/skills/bundled/debugging/systematic-debugging/test-academic.md +0 -14
- claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-1.md +0 -58
- claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-2.md +0 -68
- claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-3.md +0 -69
- claude_mpm/skills/bundled/debugging/verification-before-completion/SKILL.md +0 -131
- claude_mpm/skills/bundled/debugging/verification-before-completion/references/gate-function.md +0 -325
- claude_mpm/skills/bundled/debugging/verification-before-completion/references/integration-and-workflows.md +0 -490
- claude_mpm/skills/bundled/debugging/verification-before-completion/references/red-flags-and-failures.md +0 -425
- claude_mpm/skills/bundled/debugging/verification-before-completion/references/verification-patterns.md +0 -499
- claude_mpm/skills/bundled/main/artifacts-builder/SKILL.md +0 -86
- claude_mpm/skills/bundled/main/internal-comms/SKILL.md +0 -43
- claude_mpm/skills/bundled/main/internal-comms/examples/3p-updates.md +0 -47
- claude_mpm/skills/bundled/main/internal-comms/examples/company-newsletter.md +0 -65
- claude_mpm/skills/bundled/main/internal-comms/examples/faq-answers.md +0 -30
- claude_mpm/skills/bundled/main/internal-comms/examples/general-comms.md +0 -16
- claude_mpm/skills/bundled/main/mcp-builder/SKILL.md +0 -160
- claude_mpm/skills/bundled/main/mcp-builder/reference/design_principles.md +0 -412
- claude_mpm/skills/bundled/main/mcp-builder/reference/evaluation.md +0 -602
- claude_mpm/skills/bundled/main/mcp-builder/reference/mcp_best_practices.md +0 -915
- claude_mpm/skills/bundled/main/mcp-builder/reference/node_mcp_server.md +0 -916
- claude_mpm/skills/bundled/main/mcp-builder/reference/python_mcp_server.md +0 -752
- claude_mpm/skills/bundled/main/mcp-builder/reference/workflow.md +0 -1237
- claude_mpm/skills/bundled/main/skill-creator/SKILL.md +0 -189
- claude_mpm/skills/bundled/main/skill-creator/references/best-practices.md +0 -500
- claude_mpm/skills/bundled/main/skill-creator/references/creation-workflow.md +0 -464
- claude_mpm/skills/bundled/main/skill-creator/references/examples.md +0 -619
- claude_mpm/skills/bundled/main/skill-creator/references/progressive-disclosure.md +0 -437
- claude_mpm/skills/bundled/main/skill-creator/references/skill-structure.md +0 -231
- claude_mpm/skills/bundled/php/espocrm-development/SKILL.md +0 -170
- claude_mpm/skills/bundled/php/espocrm-development/references/architecture.md +0 -602
- claude_mpm/skills/bundled/php/espocrm-development/references/common-tasks.md +0 -821
- claude_mpm/skills/bundled/php/espocrm-development/references/development-workflow.md +0 -742
- claude_mpm/skills/bundled/php/espocrm-development/references/frontend-customization.md +0 -726
- claude_mpm/skills/bundled/php/espocrm-development/references/hooks-and-services.md +0 -764
- claude_mpm/skills/bundled/php/espocrm-development/references/testing-debugging.md +0 -831
- claude_mpm/skills/bundled/rust/desktop-applications/SKILL.md +0 -226
- claude_mpm/skills/bundled/rust/desktop-applications/references/architecture-patterns.md +0 -901
- claude_mpm/skills/bundled/rust/desktop-applications/references/native-gui-frameworks.md +0 -901
- claude_mpm/skills/bundled/rust/desktop-applications/references/platform-integration.md +0 -775
- claude_mpm/skills/bundled/rust/desktop-applications/references/state-management.md +0 -937
- claude_mpm/skills/bundled/rust/desktop-applications/references/tauri-framework.md +0 -770
- claude_mpm/skills/bundled/rust/desktop-applications/references/testing-deployment.md +0 -961
- claude_mpm/skills/bundled/testing/condition-based-waiting/SKILL.md +0 -119
- claude_mpm/skills/bundled/testing/condition-based-waiting/references/patterns-and-implementation.md +0 -253
- claude_mpm/skills/bundled/testing/test-driven-development/SKILL.md +0 -145
- claude_mpm/skills/bundled/testing/test-driven-development/references/anti-patterns.md +0 -543
- claude_mpm/skills/bundled/testing/test-driven-development/references/examples.md +0 -741
- claude_mpm/skills/bundled/testing/test-driven-development/references/integration.md +0 -470
- claude_mpm/skills/bundled/testing/test-driven-development/references/philosophy.md +0 -458
- claude_mpm/skills/bundled/testing/test-driven-development/references/workflow.md +0 -639
- claude_mpm/skills/bundled/testing/testing-anti-patterns/SKILL.md +0 -140
- claude_mpm/skills/bundled/testing/testing-anti-patterns/references/completeness-anti-patterns.md +0 -572
- claude_mpm/skills/bundled/testing/testing-anti-patterns/references/core-anti-patterns.md +0 -411
- claude_mpm/skills/bundled/testing/testing-anti-patterns/references/detection-guide.md +0 -569
- claude_mpm/skills/bundled/testing/testing-anti-patterns/references/tdd-connection.md +0 -695
- claude_mpm/skills/bundled/testing/webapp-testing/SKILL.md +0 -184
- claude_mpm/skills/bundled/testing/webapp-testing/decision-tree.md +0 -459
- claude_mpm/skills/bundled/testing/webapp-testing/playwright-patterns.md +0 -479
- claude_mpm/skills/bundled/testing/webapp-testing/reconnaissance-pattern.md +0 -687
- claude_mpm/skills/bundled/testing/webapp-testing/server-management.md +0 -758
- claude_mpm/skills/bundled/testing/webapp-testing/troubleshooting.md +0 -868
- claude_mpm/tools/code_tree_analyzer.py +0 -1825
- /claude_mpm/agents/templates/{git_file_tracking.md → git-file-tracking.md} +0 -0
- /claude_mpm/agents/templates/{pm_examples.md → pm-examples.md} +0 -0
- /claude_mpm/agents/templates/{response_format.md → response-format.md} +0 -0
- /claude_mpm/agents/templates/{validation_templates.md → validation-templates.md} +0 -0
- {claude_mpm-4.21.0.dist-info → claude_mpm-5.0.2.dist-info}/WHEEL +0 -0
- {claude_mpm-4.21.0.dist-info → claude_mpm-5.0.2.dist-info}/entry_points.txt +0 -0
- {claude_mpm-4.21.0.dist-info → claude_mpm-5.0.2.dist-info}/licenses/LICENSE +0 -0
- {claude_mpm-4.21.0.dist-info → claude_mpm-5.0.2.dist-info}/top_level.txt +0 -0
|
@@ -0,0 +1,619 @@
|
|
|
1
|
+
"""Structured questions framework for AskUserQuestion tool integration.
|
|
2
|
+
|
|
3
|
+
This module provides a type-safe, validated approach to building structured questions
|
|
4
|
+
for use with Claude's AskUserQuestion tool. It enables agents (particularly PM) to
|
|
5
|
+
gather user input in a structured, consistent way.
|
|
6
|
+
|
|
7
|
+
Design Principles:
|
|
8
|
+
- Type-safe with full type hints
|
|
9
|
+
- Validation at construction time
|
|
10
|
+
- Fluent API for ease of use
|
|
11
|
+
- Clear error messages
|
|
12
|
+
- Immutable question structures
|
|
13
|
+
|
|
14
|
+
Example Usage:
|
|
15
|
+
>>> question = (
|
|
16
|
+
... QuestionBuilder()
|
|
17
|
+
... .ask("Which database should we use?")
|
|
18
|
+
... .header("Database")
|
|
19
|
+
... .add_option("PostgreSQL", "Robust, feature-rich relational database")
|
|
20
|
+
... .add_option("MongoDB", "Flexible NoSQL document database")
|
|
21
|
+
... .build()
|
|
22
|
+
... )
|
|
23
|
+
>>> question_set = QuestionSet([question])
|
|
24
|
+
>>> params = question_set.to_ask_user_question_params()
|
|
25
|
+
"""
|
|
26
|
+
|
|
27
|
+
from __future__ import annotations
|
|
28
|
+
|
|
29
|
+
import sys
|
|
30
|
+
from dataclasses import dataclass, field
|
|
31
|
+
from typing import Any
|
|
32
|
+
|
|
33
|
+
|
|
34
|
+
class QuestionValidationError(Exception):
|
|
35
|
+
"""Raised when question validation fails."""
|
|
36
|
+
|
|
37
|
+
|
|
38
|
+
@dataclass(frozen=True)
|
|
39
|
+
class QuestionOption:
|
|
40
|
+
"""Represents a single option in a structured question.
|
|
41
|
+
|
|
42
|
+
Attributes:
|
|
43
|
+
label: Display text shown to user (1-5 words recommended)
|
|
44
|
+
description: Explanation of what this option means or implies
|
|
45
|
+
"""
|
|
46
|
+
|
|
47
|
+
label: str
|
|
48
|
+
description: str
|
|
49
|
+
|
|
50
|
+
def __post_init__(self) -> None:
|
|
51
|
+
"""Validate option constraints."""
|
|
52
|
+
if not self.label or not self.label.strip():
|
|
53
|
+
raise QuestionValidationError("Option label cannot be empty")
|
|
54
|
+
if not self.description or not self.description.strip():
|
|
55
|
+
raise QuestionValidationError("Option description cannot be empty")
|
|
56
|
+
if len(self.label) > 50:
|
|
57
|
+
raise QuestionValidationError(
|
|
58
|
+
f"Option label too long ({len(self.label)} chars): {self.label[:30]}..."
|
|
59
|
+
)
|
|
60
|
+
|
|
61
|
+
def to_dict(self) -> dict[str, str]:
|
|
62
|
+
"""Convert option to AskUserQuestion format."""
|
|
63
|
+
return {"label": self.label, "description": self.description}
|
|
64
|
+
|
|
65
|
+
|
|
66
|
+
@dataclass(frozen=True)
|
|
67
|
+
class StructuredQuestion:
|
|
68
|
+
"""Represents a single structured question with validation.
|
|
69
|
+
|
|
70
|
+
Attributes:
|
|
71
|
+
question: The complete question text (should end with '?')
|
|
72
|
+
header: Short label displayed as chip/tag (max 12 chars)
|
|
73
|
+
options: List of 2-4 QuestionOption objects
|
|
74
|
+
multi_select: Whether user can select multiple options
|
|
75
|
+
"""
|
|
76
|
+
|
|
77
|
+
question: str
|
|
78
|
+
header: str
|
|
79
|
+
options: list[QuestionOption]
|
|
80
|
+
multi_select: bool = False
|
|
81
|
+
|
|
82
|
+
def __post_init__(self) -> None:
|
|
83
|
+
"""Validate question constraints."""
|
|
84
|
+
# Question text validation
|
|
85
|
+
if not self.question or not self.question.strip():
|
|
86
|
+
raise QuestionValidationError("Question text cannot be empty")
|
|
87
|
+
if not self.question.strip().endswith("?"):
|
|
88
|
+
raise QuestionValidationError(
|
|
89
|
+
f"Question should end with '?': {self.question[:50]}"
|
|
90
|
+
)
|
|
91
|
+
|
|
92
|
+
# Header validation
|
|
93
|
+
if not self.header or not self.header.strip():
|
|
94
|
+
raise QuestionValidationError("Header cannot be empty")
|
|
95
|
+
if len(self.header) > 12:
|
|
96
|
+
raise QuestionValidationError(
|
|
97
|
+
f"Header too long ({len(self.header)} chars, max 12): {self.header}"
|
|
98
|
+
)
|
|
99
|
+
|
|
100
|
+
# Options validation
|
|
101
|
+
if not self.options:
|
|
102
|
+
raise QuestionValidationError("Question must have at least 2 options")
|
|
103
|
+
if len(self.options) < 2:
|
|
104
|
+
raise QuestionValidationError(
|
|
105
|
+
f"Question must have at least 2 options, got {len(self.options)}"
|
|
106
|
+
)
|
|
107
|
+
if len(self.options) > 4:
|
|
108
|
+
raise QuestionValidationError(
|
|
109
|
+
f"Question must have at most 4 options, got {len(self.options)}"
|
|
110
|
+
)
|
|
111
|
+
|
|
112
|
+
# Validate all options are QuestionOption instances
|
|
113
|
+
if not all(isinstance(opt, QuestionOption) for opt in self.options):
|
|
114
|
+
raise QuestionValidationError(
|
|
115
|
+
"All options must be QuestionOption instances"
|
|
116
|
+
)
|
|
117
|
+
|
|
118
|
+
def to_dict(self) -> dict[str, Any]:
|
|
119
|
+
"""Convert question to AskUserQuestion format."""
|
|
120
|
+
return {
|
|
121
|
+
"question": self.question,
|
|
122
|
+
"header": self.header,
|
|
123
|
+
"options": [opt.to_dict() for opt in self.options],
|
|
124
|
+
"multiSelect": self.multi_select,
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
|
|
128
|
+
class QuestionBuilder:
|
|
129
|
+
"""Fluent API for building StructuredQuestion objects.
|
|
130
|
+
|
|
131
|
+
Example:
|
|
132
|
+
>>> question = (
|
|
133
|
+
... QuestionBuilder()
|
|
134
|
+
... .ask("Which testing framework?")
|
|
135
|
+
... .header("Testing")
|
|
136
|
+
... .add_option("pytest", "Python's most popular testing framework")
|
|
137
|
+
... .add_option("unittest", "Python's built-in testing framework")
|
|
138
|
+
... .multi_select()
|
|
139
|
+
... .build()
|
|
140
|
+
... )
|
|
141
|
+
"""
|
|
142
|
+
|
|
143
|
+
def __init__(self) -> None:
|
|
144
|
+
"""Initialize builder with empty state."""
|
|
145
|
+
self._question: str | None = None
|
|
146
|
+
self._header: str | None = None
|
|
147
|
+
self._options: list[QuestionOption] = []
|
|
148
|
+
self._multi_select: bool = False
|
|
149
|
+
|
|
150
|
+
def ask(self, question: str) -> QuestionBuilder:
|
|
151
|
+
"""Set the question text.
|
|
152
|
+
|
|
153
|
+
Args:
|
|
154
|
+
question: The question text (should end with '?')
|
|
155
|
+
|
|
156
|
+
Returns:
|
|
157
|
+
Self for method chaining
|
|
158
|
+
"""
|
|
159
|
+
self._question = question
|
|
160
|
+
return self
|
|
161
|
+
|
|
162
|
+
def header(self, header: str) -> QuestionBuilder:
|
|
163
|
+
"""Set the header label.
|
|
164
|
+
|
|
165
|
+
Args:
|
|
166
|
+
header: Short label (max 12 chars)
|
|
167
|
+
|
|
168
|
+
Returns:
|
|
169
|
+
Self for method chaining
|
|
170
|
+
"""
|
|
171
|
+
self._header = header
|
|
172
|
+
return self
|
|
173
|
+
|
|
174
|
+
def add_option(self, label: str, description: str) -> QuestionBuilder:
|
|
175
|
+
"""Add an option to the question.
|
|
176
|
+
|
|
177
|
+
Args:
|
|
178
|
+
label: Display text for the option
|
|
179
|
+
description: Explanation of the option
|
|
180
|
+
|
|
181
|
+
Returns:
|
|
182
|
+
Self for method chaining
|
|
183
|
+
"""
|
|
184
|
+
self._options.append(QuestionOption(label=label, description=description))
|
|
185
|
+
return self
|
|
186
|
+
|
|
187
|
+
def with_options(self, options: list[QuestionOption]) -> QuestionBuilder:
|
|
188
|
+
"""Set all options at once.
|
|
189
|
+
|
|
190
|
+
Args:
|
|
191
|
+
options: List of QuestionOption objects
|
|
192
|
+
|
|
193
|
+
Returns:
|
|
194
|
+
Self for method chaining
|
|
195
|
+
"""
|
|
196
|
+
self._options = list(options)
|
|
197
|
+
return self
|
|
198
|
+
|
|
199
|
+
def multi_select(self, enabled: bool = True) -> QuestionBuilder:
|
|
200
|
+
"""Enable or disable multi-select mode.
|
|
201
|
+
|
|
202
|
+
Args:
|
|
203
|
+
enabled: Whether to allow multiple selections
|
|
204
|
+
|
|
205
|
+
Returns:
|
|
206
|
+
Self for method chaining
|
|
207
|
+
"""
|
|
208
|
+
self._multi_select = enabled
|
|
209
|
+
return self
|
|
210
|
+
|
|
211
|
+
def build(self) -> StructuredQuestion:
|
|
212
|
+
"""Build and validate the StructuredQuestion.
|
|
213
|
+
|
|
214
|
+
Returns:
|
|
215
|
+
Validated StructuredQuestion instance
|
|
216
|
+
|
|
217
|
+
Raises:
|
|
218
|
+
QuestionValidationError: If validation fails
|
|
219
|
+
"""
|
|
220
|
+
if self._question is None:
|
|
221
|
+
raise QuestionValidationError("Question text is required (use .ask())")
|
|
222
|
+
if self._header is None:
|
|
223
|
+
raise QuestionValidationError("Header is required (use .header())")
|
|
224
|
+
|
|
225
|
+
return StructuredQuestion(
|
|
226
|
+
question=self._question,
|
|
227
|
+
header=self._header,
|
|
228
|
+
options=self._options,
|
|
229
|
+
multi_select=self._multi_select,
|
|
230
|
+
)
|
|
231
|
+
|
|
232
|
+
|
|
233
|
+
@dataclass
|
|
234
|
+
class QuestionSet:
|
|
235
|
+
"""Collection of structured questions for a single AskUserQuestion call.
|
|
236
|
+
|
|
237
|
+
Attributes:
|
|
238
|
+
questions: List of 1-4 StructuredQuestion objects
|
|
239
|
+
"""
|
|
240
|
+
|
|
241
|
+
questions: list[StructuredQuestion] = field(default_factory=list)
|
|
242
|
+
|
|
243
|
+
def __post_init__(self) -> None:
|
|
244
|
+
"""Validate question set constraints."""
|
|
245
|
+
if not self.questions:
|
|
246
|
+
raise QuestionValidationError(
|
|
247
|
+
"QuestionSet must contain at least 1 question"
|
|
248
|
+
)
|
|
249
|
+
if len(self.questions) > 4:
|
|
250
|
+
raise QuestionValidationError(
|
|
251
|
+
f"QuestionSet can have at most 4 questions, got {len(self.questions)}"
|
|
252
|
+
)
|
|
253
|
+
if not all(isinstance(q, StructuredQuestion) for q in self.questions):
|
|
254
|
+
raise QuestionValidationError(
|
|
255
|
+
"All questions must be StructuredQuestion instances"
|
|
256
|
+
)
|
|
257
|
+
|
|
258
|
+
def add(self, question: StructuredQuestion) -> QuestionSet:
|
|
259
|
+
"""Add a question to the set.
|
|
260
|
+
|
|
261
|
+
Args:
|
|
262
|
+
question: StructuredQuestion to add
|
|
263
|
+
|
|
264
|
+
Returns:
|
|
265
|
+
Self for method chaining
|
|
266
|
+
|
|
267
|
+
Raises:
|
|
268
|
+
QuestionValidationError: If adding would exceed 4 questions
|
|
269
|
+
"""
|
|
270
|
+
if len(self.questions) >= 4:
|
|
271
|
+
raise QuestionValidationError("Cannot add more than 4 questions")
|
|
272
|
+
self.questions.append(question)
|
|
273
|
+
return self
|
|
274
|
+
|
|
275
|
+
def to_ask_user_question_params(self) -> dict[str, Any]:
|
|
276
|
+
"""Convert question set to AskUserQuestion tool parameters.
|
|
277
|
+
|
|
278
|
+
Returns:
|
|
279
|
+
Dictionary suitable for AskUserQuestion tool parameters
|
|
280
|
+
"""
|
|
281
|
+
return {"questions": [q.to_dict() for q in self.questions]}
|
|
282
|
+
|
|
283
|
+
def execute(
|
|
284
|
+
self,
|
|
285
|
+
response: dict[str, Any] | None = None,
|
|
286
|
+
use_fallback_if_needed: bool = True,
|
|
287
|
+
) -> ParsedResponse:
|
|
288
|
+
"""Execute questions with automatic fallback on AskUserQuestion failure.
|
|
289
|
+
|
|
290
|
+
This method provides graceful degradation when AskUserQuestion tool fails
|
|
291
|
+
or returns empty/invalid responses. It will automatically fall back to
|
|
292
|
+
text-based questions that display in the console.
|
|
293
|
+
|
|
294
|
+
Args:
|
|
295
|
+
response: Response from AskUserQuestion tool (if already obtained).
|
|
296
|
+
If None, assumes tool failed and uses fallback.
|
|
297
|
+
use_fallback_if_needed: Auto-fallback if AskUserQuestion fails (default: True)
|
|
298
|
+
|
|
299
|
+
Returns:
|
|
300
|
+
ParsedResponse with user answers
|
|
301
|
+
|
|
302
|
+
Example:
|
|
303
|
+
>>> # When AskUserQuestion fails
|
|
304
|
+
>>> response = {} # Empty/failed response
|
|
305
|
+
>>> parsed = question_set.execute(response)
|
|
306
|
+
>>> # Automatically displays text-based questions
|
|
307
|
+
"""
|
|
308
|
+
# If no response provided, use fallback immediately
|
|
309
|
+
if response is None:
|
|
310
|
+
if use_fallback_if_needed:
|
|
311
|
+
return self._execute_text_fallback()
|
|
312
|
+
raise QuestionValidationError("No response provided and fallback disabled")
|
|
313
|
+
|
|
314
|
+
# Check if response indicates failure
|
|
315
|
+
if use_fallback_if_needed and self._should_use_fallback(response):
|
|
316
|
+
return self._execute_text_fallback()
|
|
317
|
+
|
|
318
|
+
# Use standard parser for valid responses
|
|
319
|
+
return ParsedResponse(self, response.get("answers", {}))
|
|
320
|
+
|
|
321
|
+
def _should_use_fallback(self, response: dict[str, Any]) -> bool:
|
|
322
|
+
"""Detect if AskUserQuestion failed or returned invalid response.
|
|
323
|
+
|
|
324
|
+
Args:
|
|
325
|
+
response: Response from AskUserQuestion tool
|
|
326
|
+
|
|
327
|
+
Returns:
|
|
328
|
+
True if fallback should be used, False if response is valid
|
|
329
|
+
"""
|
|
330
|
+
# Check for empty/missing response
|
|
331
|
+
if not response or not isinstance(response, dict):
|
|
332
|
+
return True
|
|
333
|
+
|
|
334
|
+
# Check for missing answers key
|
|
335
|
+
if "answers" not in response:
|
|
336
|
+
return True
|
|
337
|
+
|
|
338
|
+
answers = response.get("answers", {})
|
|
339
|
+
|
|
340
|
+
# Check for empty answers
|
|
341
|
+
if not answers or answers == {}:
|
|
342
|
+
return True
|
|
343
|
+
|
|
344
|
+
# Check for fake/placeholder responses (like ".")
|
|
345
|
+
answer_values = list(answers.values())
|
|
346
|
+
if answer_values == ["."] or all(v in {".", ""} for v in answer_values):
|
|
347
|
+
return True
|
|
348
|
+
|
|
349
|
+
return False
|
|
350
|
+
|
|
351
|
+
def _execute_text_fallback(self) -> ParsedResponse:
|
|
352
|
+
"""Execute questions as formatted text when AskUserQuestion fails.
|
|
353
|
+
|
|
354
|
+
Displays questions in a clear, numbered format and collects user input
|
|
355
|
+
via stdin. Supports numeric selection, label matching, and custom answers.
|
|
356
|
+
|
|
357
|
+
Returns:
|
|
358
|
+
ParsedResponse with user answers
|
|
359
|
+
"""
|
|
360
|
+
print("\n" + "=" * 60, file=sys.stderr)
|
|
361
|
+
print("📋 USER INPUT REQUIRED", file=sys.stderr)
|
|
362
|
+
print(
|
|
363
|
+
"(AskUserQuestion tool unavailable - using text fallback)", file=sys.stderr
|
|
364
|
+
)
|
|
365
|
+
print("=" * 60 + "\n", file=sys.stderr)
|
|
366
|
+
|
|
367
|
+
answers = {}
|
|
368
|
+
for i, question in enumerate(self.questions, 1):
|
|
369
|
+
# Display question header
|
|
370
|
+
print(f"=== Question {i} of {len(self.questions)} ===", file=sys.stderr)
|
|
371
|
+
print(f"[{question.header}] {question.question}\n", file=sys.stderr)
|
|
372
|
+
|
|
373
|
+
# Display options
|
|
374
|
+
print("Options:", file=sys.stderr)
|
|
375
|
+
for j, opt in enumerate(question.options, 1):
|
|
376
|
+
print(f"{j}. {opt.label} - {opt.description}", file=sys.stderr)
|
|
377
|
+
|
|
378
|
+
# Show multi-select hint
|
|
379
|
+
if question.multi_select:
|
|
380
|
+
print(
|
|
381
|
+
"\n(You may select multiple - separate with commas)",
|
|
382
|
+
file=sys.stderr,
|
|
383
|
+
)
|
|
384
|
+
|
|
385
|
+
# Get user input
|
|
386
|
+
print("\nYour answer: ", file=sys.stderr, end="")
|
|
387
|
+
user_input = input().strip()
|
|
388
|
+
|
|
389
|
+
# Parse response
|
|
390
|
+
answer = self._parse_text_response(
|
|
391
|
+
user_input, question.options, question.multi_select
|
|
392
|
+
)
|
|
393
|
+
|
|
394
|
+
answers[question.header] = answer
|
|
395
|
+
print(file=sys.stderr) # Blank line between questions
|
|
396
|
+
|
|
397
|
+
print("=" * 60 + "\n", file=sys.stderr)
|
|
398
|
+
return ParsedResponse(self, answers)
|
|
399
|
+
|
|
400
|
+
def _parse_text_response(
|
|
401
|
+
self,
|
|
402
|
+
user_input: str,
|
|
403
|
+
options: list[QuestionOption],
|
|
404
|
+
multi_select: bool,
|
|
405
|
+
) -> str | list[str]:
|
|
406
|
+
"""Parse user's text response into option label(s).
|
|
407
|
+
|
|
408
|
+
Supports multiple input formats:
|
|
409
|
+
- Numeric: "1", "2", "3"
|
|
410
|
+
- Label matching: "OAuth", "PostgreSQL"
|
|
411
|
+
- Multi-select: "1,2" or "OAuth, JWT"
|
|
412
|
+
- Custom answers: Any text not matching options
|
|
413
|
+
|
|
414
|
+
Args:
|
|
415
|
+
user_input: Raw user input string
|
|
416
|
+
options: Available question options
|
|
417
|
+
multi_select: Whether question allows multiple selections
|
|
418
|
+
|
|
419
|
+
Returns:
|
|
420
|
+
Selected option label(s) or custom answer
|
|
421
|
+
"""
|
|
422
|
+
# Handle multi-select (comma-separated)
|
|
423
|
+
if multi_select and "," in user_input:
|
|
424
|
+
inputs = [x.strip() for x in user_input.split(",")]
|
|
425
|
+
results = []
|
|
426
|
+
for inp in inputs:
|
|
427
|
+
match = self._match_single_input(inp, options)
|
|
428
|
+
if match:
|
|
429
|
+
results.append(match)
|
|
430
|
+
else:
|
|
431
|
+
results.append(inp) # Include custom answer
|
|
432
|
+
return results if results else [user_input]
|
|
433
|
+
|
|
434
|
+
# Single selection
|
|
435
|
+
match = self._match_single_input(user_input, options)
|
|
436
|
+
if match:
|
|
437
|
+
return match
|
|
438
|
+
|
|
439
|
+
# Return custom answer
|
|
440
|
+
return user_input
|
|
441
|
+
|
|
442
|
+
def _match_single_input(
|
|
443
|
+
self, input_str: str, options: list[QuestionOption]
|
|
444
|
+
) -> str | None:
|
|
445
|
+
"""Match a single input string to an option.
|
|
446
|
+
|
|
447
|
+
Args:
|
|
448
|
+
input_str: User input to match
|
|
449
|
+
options: Available options
|
|
450
|
+
|
|
451
|
+
Returns:
|
|
452
|
+
Matched option label or None if no match
|
|
453
|
+
"""
|
|
454
|
+
input_lower = input_str.lower().strip()
|
|
455
|
+
|
|
456
|
+
# Handle numeric input
|
|
457
|
+
if input_str.isdigit():
|
|
458
|
+
idx = int(input_str) - 1
|
|
459
|
+
if 0 <= idx < len(options):
|
|
460
|
+
return options[idx].label
|
|
461
|
+
|
|
462
|
+
# Exact match (case-insensitive)
|
|
463
|
+
for opt in options:
|
|
464
|
+
if opt.label.lower() == input_lower:
|
|
465
|
+
return opt.label
|
|
466
|
+
|
|
467
|
+
# Partial match (case-insensitive)
|
|
468
|
+
for opt in options:
|
|
469
|
+
if input_lower in opt.label.lower():
|
|
470
|
+
return opt.label
|
|
471
|
+
|
|
472
|
+
return None
|
|
473
|
+
|
|
474
|
+
|
|
475
|
+
@dataclass
|
|
476
|
+
class ParsedResponse:
|
|
477
|
+
"""Wrapper for parsed question responses with convenient accessor methods.
|
|
478
|
+
|
|
479
|
+
Provides a clean interface for accessing user answers from both
|
|
480
|
+
AskUserQuestion tool and text fallback responses.
|
|
481
|
+
|
|
482
|
+
Attributes:
|
|
483
|
+
question_set: The QuestionSet that was asked
|
|
484
|
+
answers: Dictionary mapping question headers to selected labels
|
|
485
|
+
"""
|
|
486
|
+
|
|
487
|
+
question_set: QuestionSet
|
|
488
|
+
answers: dict[str, str | list[str]]
|
|
489
|
+
|
|
490
|
+
def get(self, header: str) -> str | list[str] | None:
|
|
491
|
+
"""Get answer for a specific question by header.
|
|
492
|
+
|
|
493
|
+
Args:
|
|
494
|
+
header: Question header to look up
|
|
495
|
+
|
|
496
|
+
Returns:
|
|
497
|
+
Selected option label(s) or None if not answered
|
|
498
|
+
"""
|
|
499
|
+
return self.answers.get(header)
|
|
500
|
+
|
|
501
|
+
def was_answered(self, header: str) -> bool:
|
|
502
|
+
"""Check if a question was answered.
|
|
503
|
+
|
|
504
|
+
Args:
|
|
505
|
+
header: Question header to check
|
|
506
|
+
|
|
507
|
+
Returns:
|
|
508
|
+
True if question was answered, False otherwise
|
|
509
|
+
"""
|
|
510
|
+
return header in self.answers
|
|
511
|
+
|
|
512
|
+
def get_all(self) -> dict[str, str | list[str]]:
|
|
513
|
+
"""Get all answers as a dictionary.
|
|
514
|
+
|
|
515
|
+
Returns:
|
|
516
|
+
Dictionary mapping question headers to selected labels
|
|
517
|
+
"""
|
|
518
|
+
return self.answers.copy()
|
|
519
|
+
|
|
520
|
+
|
|
521
|
+
class ResponseParser:
|
|
522
|
+
"""Parses and validates responses from AskUserQuestion tool.
|
|
523
|
+
|
|
524
|
+
Note: This class is maintained for backward compatibility.
|
|
525
|
+
New code should use QuestionSet.execute() which returns ParsedResponse directly.
|
|
526
|
+
|
|
527
|
+
Example:
|
|
528
|
+
>>> parser = ResponseParser(question_set)
|
|
529
|
+
>>> answers = parser.parse(tool_response)
|
|
530
|
+
>>> db_choice = answers.get("Database") # Returns selected option label(s)
|
|
531
|
+
"""
|
|
532
|
+
|
|
533
|
+
def __init__(self, question_set: QuestionSet) -> None:
|
|
534
|
+
"""Initialize parser with the question set that was asked.
|
|
535
|
+
|
|
536
|
+
Args:
|
|
537
|
+
question_set: The QuestionSet that was sent to AskUserQuestion
|
|
538
|
+
"""
|
|
539
|
+
self.question_set = question_set
|
|
540
|
+
|
|
541
|
+
def parse(self, response: dict[str, Any]) -> dict[str, str | list[str]]:
|
|
542
|
+
"""Parse AskUserQuestion response into header -> answer mapping.
|
|
543
|
+
|
|
544
|
+
Args:
|
|
545
|
+
response: Raw response from AskUserQuestion tool
|
|
546
|
+
Expected format: {"answers": {"header": "label", ...}}
|
|
547
|
+
|
|
548
|
+
Returns:
|
|
549
|
+
Dictionary mapping question headers to selected option labels
|
|
550
|
+
For multi-select questions, values are lists of labels
|
|
551
|
+
|
|
552
|
+
Raises:
|
|
553
|
+
QuestionValidationError: If response format is invalid
|
|
554
|
+
"""
|
|
555
|
+
if not isinstance(response, dict):
|
|
556
|
+
raise QuestionValidationError("Response must be a dictionary")
|
|
557
|
+
|
|
558
|
+
answers = response.get("answers", {})
|
|
559
|
+
if not isinstance(answers, dict):
|
|
560
|
+
raise QuestionValidationError("Response must contain 'answers' dictionary")
|
|
561
|
+
|
|
562
|
+
parsed: dict[str, str | list[str]] = {}
|
|
563
|
+
|
|
564
|
+
for question in self.question_set.questions:
|
|
565
|
+
header = question.header
|
|
566
|
+
answer = answers.get(header)
|
|
567
|
+
|
|
568
|
+
if answer is None:
|
|
569
|
+
# User didn't answer this question (optional question)
|
|
570
|
+
continue
|
|
571
|
+
|
|
572
|
+
if question.multi_select:
|
|
573
|
+
# Multi-select: answer should be list or comma-separated string
|
|
574
|
+
if isinstance(answer, list):
|
|
575
|
+
parsed[header] = answer
|
|
576
|
+
elif isinstance(answer, str):
|
|
577
|
+
# Parse comma-separated values
|
|
578
|
+
parsed[header] = [a.strip() for a in answer.split(",") if a.strip()]
|
|
579
|
+
else:
|
|
580
|
+
raise QuestionValidationError(
|
|
581
|
+
f"Multi-select answer for '{header}' must be list or string"
|
|
582
|
+
)
|
|
583
|
+
# Single select: answer should be string
|
|
584
|
+
elif isinstance(answer, str):
|
|
585
|
+
parsed[header] = answer
|
|
586
|
+
else:
|
|
587
|
+
raise QuestionValidationError(
|
|
588
|
+
f"Single-select answer for '{header}' must be string"
|
|
589
|
+
)
|
|
590
|
+
|
|
591
|
+
return parsed
|
|
592
|
+
|
|
593
|
+
def get_answer(
|
|
594
|
+
self, parsed_answers: dict[str, str | list[str]], header: str
|
|
595
|
+
) -> str | list[str] | None:
|
|
596
|
+
"""Get answer for a specific question by header.
|
|
597
|
+
|
|
598
|
+
Args:
|
|
599
|
+
parsed_answers: Result from parse()
|
|
600
|
+
header: Question header to look up
|
|
601
|
+
|
|
602
|
+
Returns:
|
|
603
|
+
Selected option label(s) or None if not answered
|
|
604
|
+
"""
|
|
605
|
+
return parsed_answers.get(header)
|
|
606
|
+
|
|
607
|
+
def was_answered(
|
|
608
|
+
self, parsed_answers: dict[str, str | list[str]], header: str
|
|
609
|
+
) -> bool:
|
|
610
|
+
"""Check if a question was answered.
|
|
611
|
+
|
|
612
|
+
Args:
|
|
613
|
+
parsed_answers: Result from parse()
|
|
614
|
+
header: Question header to check
|
|
615
|
+
|
|
616
|
+
Returns:
|
|
617
|
+
True if question was answered, False otherwise
|
|
618
|
+
"""
|
|
619
|
+
return header in parsed_answers
|