claude-mpm 4.21.3__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 +11 -2
- claude_mpm/commands/mpm-init.md +27 -2
- claude_mpm/commands/mpm-monitor.md +9 -0
- claude_mpm/commands/{mpm-resume.md → mpm-session-resume.md} +9 -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 +10 -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/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/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/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/analysis.py +1 -1
- 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.3.dist-info → claude_mpm-5.0.2.dist-info}/METADATA +429 -59
- {claude_mpm-4.21.3.dist-info → claude_mpm-5.0.2.dist-info}/RECORD +252 -425
- 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/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.3.dist-info → claude_mpm-5.0.2.dist-info}/WHEEL +0 -0
- {claude_mpm-4.21.3.dist-info → claude_mpm-5.0.2.dist-info}/entry_points.txt +0 -0
- {claude_mpm-4.21.3.dist-info → claude_mpm-5.0.2.dist-info}/licenses/LICENSE +0 -0
- {claude_mpm-4.21.3.dist-info → claude_mpm-5.0.2.dist-info}/top_level.txt +0 -0
|
@@ -0,0 +1,694 @@
|
|
|
1
|
+
"""Skill source command handlers for managing Git-based skill repositories.
|
|
2
|
+
|
|
3
|
+
WHY: This module implements CLI commands for managing skill source repositories
|
|
4
|
+
(Git repositories containing skill JSON files). Provides add, remove, list, update,
|
|
5
|
+
enable, disable, and show commands with user-friendly output.
|
|
6
|
+
|
|
7
|
+
DESIGN DECISION: Uses SkillSourceConfiguration for persistent storage and
|
|
8
|
+
GitSkillSourceManager for Git operations. Provides clear, emoji-enhanced feedback
|
|
9
|
+
for better UX. Handles errors gracefully with actionable messages.
|
|
10
|
+
"""
|
|
11
|
+
|
|
12
|
+
import json
|
|
13
|
+
import logging
|
|
14
|
+
import re
|
|
15
|
+
|
|
16
|
+
from ...config.skill_sources import SkillSource, SkillSourceConfiguration
|
|
17
|
+
from ...services.skills.git_skill_source_manager import GitSkillSourceManager
|
|
18
|
+
from ...services.skills.skill_discovery_service import SkillDiscoveryService
|
|
19
|
+
|
|
20
|
+
logger = logging.getLogger(__name__)
|
|
21
|
+
|
|
22
|
+
|
|
23
|
+
def _test_skill_repository_access(source: SkillSource) -> dict:
|
|
24
|
+
"""Test if skill repository is accessible via GitHub API.
|
|
25
|
+
|
|
26
|
+
Design Decision: Test via GitHub API, not Git clone
|
|
27
|
+
|
|
28
|
+
Rationale: GitHub API is faster and less resource-intensive than
|
|
29
|
+
cloning the repository. We can validate access and existence without
|
|
30
|
+
downloading any files.
|
|
31
|
+
|
|
32
|
+
Args:
|
|
33
|
+
source: SkillSource to test
|
|
34
|
+
|
|
35
|
+
Returns:
|
|
36
|
+
Dictionary with:
|
|
37
|
+
- accessible: bool (True if repo can be reached)
|
|
38
|
+
- error: str (error message if not accessible)
|
|
39
|
+
|
|
40
|
+
Example:
|
|
41
|
+
>>> source = SkillSource(id="custom", type="git", url="https://github.com/owner/repo")
|
|
42
|
+
>>> result = _test_skill_repository_access(source)
|
|
43
|
+
>>> print(result["accessible"])
|
|
44
|
+
True
|
|
45
|
+
"""
|
|
46
|
+
import requests
|
|
47
|
+
|
|
48
|
+
try:
|
|
49
|
+
# Parse GitHub URL
|
|
50
|
+
url = source.url.rstrip("/").replace(".git", "")
|
|
51
|
+
parts = url.split("github.com/")
|
|
52
|
+
if len(parts) != 2:
|
|
53
|
+
return {"accessible": False, "error": "Invalid GitHub URL format"}
|
|
54
|
+
|
|
55
|
+
repo_path = parts[1].strip("/")
|
|
56
|
+
owner_repo = "/".join(repo_path.split("/")[:2])
|
|
57
|
+
|
|
58
|
+
# Test GitHub API access
|
|
59
|
+
api_url = f"https://api.github.com/repos/{owner_repo}"
|
|
60
|
+
|
|
61
|
+
response = requests.get(api_url, timeout=10)
|
|
62
|
+
|
|
63
|
+
if response.status_code == 200:
|
|
64
|
+
return {"accessible": True, "error": None}
|
|
65
|
+
if response.status_code == 404:
|
|
66
|
+
return {
|
|
67
|
+
"accessible": False,
|
|
68
|
+
"error": f"Repository not found: {owner_repo}",
|
|
69
|
+
}
|
|
70
|
+
if response.status_code == 403:
|
|
71
|
+
return {
|
|
72
|
+
"accessible": False,
|
|
73
|
+
"error": "Access denied (private repository or rate limit)",
|
|
74
|
+
}
|
|
75
|
+
return {
|
|
76
|
+
"accessible": False,
|
|
77
|
+
"error": f"HTTP {response.status_code}: {response.reason}",
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
except Exception as e:
|
|
81
|
+
return {"accessible": False, "error": str(e)}
|
|
82
|
+
|
|
83
|
+
|
|
84
|
+
def _test_skill_repository_sync(source: SkillSource) -> dict:
|
|
85
|
+
"""Test syncing skill repository and discovering skills.
|
|
86
|
+
|
|
87
|
+
Design Decision: Use temporary cache for testing
|
|
88
|
+
|
|
89
|
+
Rationale: We want to test sync without polluting the main cache.
|
|
90
|
+
Use a temporary directory that gets cleaned up after testing.
|
|
91
|
+
|
|
92
|
+
Args:
|
|
93
|
+
source: SkillSource to test sync
|
|
94
|
+
|
|
95
|
+
Returns:
|
|
96
|
+
Dictionary with:
|
|
97
|
+
- synced: bool (True if sync successful)
|
|
98
|
+
- skills_discovered: int (number of skills found)
|
|
99
|
+
- error: str (error message if sync failed)
|
|
100
|
+
|
|
101
|
+
Example:
|
|
102
|
+
>>> source = SkillSource(id="custom", type="git", url="https://github.com/owner/repo")
|
|
103
|
+
>>> result = _test_skill_repository_sync(source)
|
|
104
|
+
>>> print(result["synced"])
|
|
105
|
+
True
|
|
106
|
+
>>> print(result["skills_discovered"])
|
|
107
|
+
5
|
|
108
|
+
"""
|
|
109
|
+
import tempfile
|
|
110
|
+
from pathlib import Path
|
|
111
|
+
|
|
112
|
+
try:
|
|
113
|
+
# Create temporary cache directory
|
|
114
|
+
with tempfile.TemporaryDirectory() as temp_cache:
|
|
115
|
+
temp_cache_path = Path(temp_cache)
|
|
116
|
+
|
|
117
|
+
# Create temporary config with just this source
|
|
118
|
+
temp_config_path = temp_cache_path / "skill_sources.yaml"
|
|
119
|
+
temp_config = SkillSourceConfiguration(config_path=temp_config_path)
|
|
120
|
+
|
|
121
|
+
# Save source to temp config
|
|
122
|
+
temp_config.save([source])
|
|
123
|
+
|
|
124
|
+
# Sync repository
|
|
125
|
+
manager = GitSkillSourceManager(
|
|
126
|
+
config=temp_config, cache_dir=temp_cache_path
|
|
127
|
+
)
|
|
128
|
+
sync_result = manager.sync_source(source.id, force=True)
|
|
129
|
+
|
|
130
|
+
if not sync_result.get("synced"):
|
|
131
|
+
return {
|
|
132
|
+
"synced": False,
|
|
133
|
+
"skills_discovered": 0,
|
|
134
|
+
"error": sync_result.get("error", "Unknown sync error"),
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
return {
|
|
138
|
+
"synced": True,
|
|
139
|
+
"skills_discovered": sync_result.get("skills_discovered", 0),
|
|
140
|
+
"error": None,
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
except Exception as e:
|
|
144
|
+
return {"synced": False, "skills_discovered": 0, "error": str(e)}
|
|
145
|
+
|
|
146
|
+
|
|
147
|
+
def _generate_source_id(url: str) -> str:
|
|
148
|
+
"""Generate source ID from Git URL.
|
|
149
|
+
|
|
150
|
+
Extracts repository name from various Git URL formats and sanitizes
|
|
151
|
+
it to create a valid identifier.
|
|
152
|
+
|
|
153
|
+
Args:
|
|
154
|
+
url: Git repository URL
|
|
155
|
+
|
|
156
|
+
Returns:
|
|
157
|
+
Source ID (sanitized repository name)
|
|
158
|
+
|
|
159
|
+
Examples:
|
|
160
|
+
https://github.com/owner/repo.git -> repo
|
|
161
|
+
https://github.com/owner/repo -> repo
|
|
162
|
+
git@github.com:owner/repo.git -> repo
|
|
163
|
+
"""
|
|
164
|
+
# Remove .git suffix
|
|
165
|
+
url_clean = url.rstrip("/").removesuffix(".git")
|
|
166
|
+
|
|
167
|
+
# Extract last path component (repo name)
|
|
168
|
+
if "://" in url_clean:
|
|
169
|
+
# HTTPS URL: https://github.com/owner/repo
|
|
170
|
+
repo_name = url_clean.split("/")[-1]
|
|
171
|
+
elif "@" in url_clean and ":" in url_clean:
|
|
172
|
+
# SSH URL: git@github.com:owner/repo
|
|
173
|
+
repo_name = url_clean.split(":")[-1].split("/")[-1]
|
|
174
|
+
else:
|
|
175
|
+
# Fallback: use last path component
|
|
176
|
+
repo_name = url_clean.split("/")[-1]
|
|
177
|
+
|
|
178
|
+
# Sanitize: only alphanumeric, dash, underscore
|
|
179
|
+
sanitized = re.sub(r"[^a-zA-Z0-9_-]", "-", repo_name)
|
|
180
|
+
|
|
181
|
+
# Remove leading/trailing dashes
|
|
182
|
+
sanitized = sanitized.strip("-")
|
|
183
|
+
|
|
184
|
+
return sanitized or "unnamed-repo"
|
|
185
|
+
|
|
186
|
+
|
|
187
|
+
def skill_source_command(args) -> int:
|
|
188
|
+
"""Main entry point for skill-source commands.
|
|
189
|
+
|
|
190
|
+
Routes to appropriate handler based on subcommand.
|
|
191
|
+
|
|
192
|
+
Args:
|
|
193
|
+
args: Parsed command arguments
|
|
194
|
+
|
|
195
|
+
Returns:
|
|
196
|
+
Exit code (0 = success, non-zero = error)
|
|
197
|
+
"""
|
|
198
|
+
handlers = {
|
|
199
|
+
"add": handle_add_skill_source,
|
|
200
|
+
"list": handle_list_skill_sources,
|
|
201
|
+
"remove": handle_remove_skill_source,
|
|
202
|
+
"update": handle_update_skill_sources,
|
|
203
|
+
"enable": handle_enable_skill_source,
|
|
204
|
+
"disable": handle_disable_skill_source,
|
|
205
|
+
"show": handle_show_skill_source,
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
handler = handlers.get(getattr(args, "skill_source_command", None))
|
|
209
|
+
if not handler:
|
|
210
|
+
print(f"❌ Unknown command: {getattr(args, 'skill_source_command', 'none')}")
|
|
211
|
+
print()
|
|
212
|
+
print("💡 Run 'claude-mpm skill-source --help' for available commands")
|
|
213
|
+
return 1
|
|
214
|
+
|
|
215
|
+
try:
|
|
216
|
+
return handler(args)
|
|
217
|
+
except Exception as e:
|
|
218
|
+
logger.error(f"Command failed: {e}", exc_info=True)
|
|
219
|
+
print(f"❌ Command failed: {e}")
|
|
220
|
+
return 1
|
|
221
|
+
|
|
222
|
+
|
|
223
|
+
def handle_add_skill_source(args) -> int:
|
|
224
|
+
"""Add a new skill source with immediate testing.
|
|
225
|
+
|
|
226
|
+
Args:
|
|
227
|
+
args: Parsed arguments with url, priority, branch, disabled, test, skip_test
|
|
228
|
+
|
|
229
|
+
Returns:
|
|
230
|
+
Exit code
|
|
231
|
+
|
|
232
|
+
Design Decision: Immediate testing on add (fail-fast approach)
|
|
233
|
+
|
|
234
|
+
Rationale: Adding a repository that can't be accessed or synced leads to
|
|
235
|
+
broken state at startup. By testing immediately, we provide instant feedback
|
|
236
|
+
and prevent configuration pollution.
|
|
237
|
+
|
|
238
|
+
Test Mode Behavior:
|
|
239
|
+
- --test: Test only, don't save to configuration
|
|
240
|
+
- --no-test: Skip testing entirely (not recommended)
|
|
241
|
+
- Default: Test and save if successful
|
|
242
|
+
"""
|
|
243
|
+
try:
|
|
244
|
+
# Load configuration
|
|
245
|
+
config = SkillSourceConfiguration()
|
|
246
|
+
|
|
247
|
+
# Generate source ID from URL
|
|
248
|
+
source_id = _generate_source_id(args.url)
|
|
249
|
+
|
|
250
|
+
# Check if already exists
|
|
251
|
+
existing = config.get_source(source_id)
|
|
252
|
+
if existing:
|
|
253
|
+
print(f"❌ Source '{source_id}' already exists")
|
|
254
|
+
print(f" URL: {existing.url}")
|
|
255
|
+
print()
|
|
256
|
+
print(f"💡 Remove it first: claude-mpm skill-source remove {source_id}")
|
|
257
|
+
return 1
|
|
258
|
+
|
|
259
|
+
# Validate priority range
|
|
260
|
+
if args.priority < 0 or args.priority > 1000:
|
|
261
|
+
print("❌ Priority must be between 0 and 1000")
|
|
262
|
+
return 1
|
|
263
|
+
|
|
264
|
+
# Create new source
|
|
265
|
+
enabled = not args.disabled
|
|
266
|
+
source = SkillSource(
|
|
267
|
+
id=source_id,
|
|
268
|
+
type="git",
|
|
269
|
+
url=args.url,
|
|
270
|
+
branch=args.branch,
|
|
271
|
+
priority=args.priority,
|
|
272
|
+
enabled=enabled,
|
|
273
|
+
)
|
|
274
|
+
|
|
275
|
+
# Determine if we should test
|
|
276
|
+
test_mode = getattr(args, "test", False)
|
|
277
|
+
skip_test = getattr(args, "skip_test", False)
|
|
278
|
+
|
|
279
|
+
# Test repository access unless explicitly skipped
|
|
280
|
+
if not skip_test:
|
|
281
|
+
print(f"🔍 Testing repository access: {args.url}")
|
|
282
|
+
print()
|
|
283
|
+
|
|
284
|
+
test_result = _test_skill_repository_access(source)
|
|
285
|
+
|
|
286
|
+
if not test_result["accessible"]:
|
|
287
|
+
print(f"❌ Repository not accessible: {test_result['error']}")
|
|
288
|
+
print()
|
|
289
|
+
print("💡 Check the URL and try again")
|
|
290
|
+
return 1
|
|
291
|
+
|
|
292
|
+
print("✅ Repository accessible")
|
|
293
|
+
|
|
294
|
+
# Test sync and discovery
|
|
295
|
+
print("🔍 Testing sync and skill discovery...")
|
|
296
|
+
print()
|
|
297
|
+
|
|
298
|
+
sync_result = _test_skill_repository_sync(source)
|
|
299
|
+
|
|
300
|
+
if not sync_result["synced"]:
|
|
301
|
+
print(f"❌ Sync failed: {sync_result['error']}")
|
|
302
|
+
print()
|
|
303
|
+
print("💡 Repository may be valid but sync failed")
|
|
304
|
+
print(
|
|
305
|
+
" You can still add it with --no-test if you want to troubleshoot later"
|
|
306
|
+
)
|
|
307
|
+
return 1
|
|
308
|
+
|
|
309
|
+
skills_count = sync_result.get("skills_discovered", 0)
|
|
310
|
+
print("✅ Sync successful")
|
|
311
|
+
print(f" Discovered {skills_count} skills")
|
|
312
|
+
print()
|
|
313
|
+
|
|
314
|
+
# If test mode, stop here
|
|
315
|
+
if test_mode:
|
|
316
|
+
print("✅ Test complete - repository is valid and accessible")
|
|
317
|
+
print()
|
|
318
|
+
print("💡 To add this repository, run without --test flag:")
|
|
319
|
+
print(f" claude-mpm skill-source add {args.url}")
|
|
320
|
+
return 0
|
|
321
|
+
|
|
322
|
+
# Check for priority conflicts
|
|
323
|
+
warnings = config.validate_priority_conflicts()
|
|
324
|
+
if warnings:
|
|
325
|
+
print("⚠️ Priority conflicts detected:")
|
|
326
|
+
for warning in warnings:
|
|
327
|
+
print(f" {warning}")
|
|
328
|
+
print()
|
|
329
|
+
print("💡 Lower priority number = higher precedence")
|
|
330
|
+
|
|
331
|
+
# Add source
|
|
332
|
+
config.add_source(source)
|
|
333
|
+
|
|
334
|
+
# Success message
|
|
335
|
+
status_emoji = "✅" if enabled else "⚠️ "
|
|
336
|
+
status_text = "enabled" if enabled else "disabled"
|
|
337
|
+
print(f"{status_emoji} Added skill source: {source_id}")
|
|
338
|
+
print(f" URL: {args.url}")
|
|
339
|
+
print(f" Branch: {args.branch}")
|
|
340
|
+
print(f" Priority: {args.priority}")
|
|
341
|
+
print(f" Status: {status_text}")
|
|
342
|
+
print()
|
|
343
|
+
|
|
344
|
+
if enabled:
|
|
345
|
+
print("💡 Repository configured and tested successfully")
|
|
346
|
+
print(" Skills from this source will be available on next startup")
|
|
347
|
+
else:
|
|
348
|
+
print(f"💡 Enable it: claude-mpm skill-source enable {source_id}")
|
|
349
|
+
|
|
350
|
+
return 0
|
|
351
|
+
|
|
352
|
+
except Exception as e:
|
|
353
|
+
logger.error(f"Failed to add skill source: {e}", exc_info=True)
|
|
354
|
+
print(f"❌ Failed to add skill source: {e}")
|
|
355
|
+
return 1
|
|
356
|
+
|
|
357
|
+
|
|
358
|
+
def handle_list_skill_sources(args) -> int:
|
|
359
|
+
"""List configured skill sources.
|
|
360
|
+
|
|
361
|
+
Args:
|
|
362
|
+
args: Parsed arguments with by_priority, enabled_only, json
|
|
363
|
+
|
|
364
|
+
Returns:
|
|
365
|
+
Exit code
|
|
366
|
+
"""
|
|
367
|
+
try:
|
|
368
|
+
config = SkillSourceConfiguration()
|
|
369
|
+
sources = config.load()
|
|
370
|
+
|
|
371
|
+
# Filter if requested
|
|
372
|
+
if args.enabled_only:
|
|
373
|
+
sources = [s for s in sources if s.enabled]
|
|
374
|
+
|
|
375
|
+
# Sort if requested
|
|
376
|
+
if args.by_priority:
|
|
377
|
+
sources = sorted(sources, key=lambda s: s.priority)
|
|
378
|
+
|
|
379
|
+
# Output format
|
|
380
|
+
if args.json:
|
|
381
|
+
# JSON output
|
|
382
|
+
output = [
|
|
383
|
+
{
|
|
384
|
+
"id": s.id,
|
|
385
|
+
"type": s.type,
|
|
386
|
+
"url": s.url,
|
|
387
|
+
"branch": s.branch,
|
|
388
|
+
"priority": s.priority,
|
|
389
|
+
"enabled": s.enabled,
|
|
390
|
+
}
|
|
391
|
+
for s in sources
|
|
392
|
+
]
|
|
393
|
+
print(json.dumps(output, indent=2))
|
|
394
|
+
else:
|
|
395
|
+
# Human-readable output
|
|
396
|
+
if not sources:
|
|
397
|
+
print("📚 No skill sources configured")
|
|
398
|
+
print()
|
|
399
|
+
print("💡 Add a source: claude-mpm skill-source add <git-url>")
|
|
400
|
+
return 0
|
|
401
|
+
|
|
402
|
+
filter_text = " (enabled only)" if args.enabled_only else ""
|
|
403
|
+
print(f"📚 Configured Skill Sources ({len(sources)} total{filter_text}):")
|
|
404
|
+
print()
|
|
405
|
+
|
|
406
|
+
for source in sources:
|
|
407
|
+
status = "✅" if source.enabled else "❌"
|
|
408
|
+
status_text = "Enabled" if source.enabled else "Disabled"
|
|
409
|
+
print(f" {status} {source.id} ({status_text})")
|
|
410
|
+
print(f" URL: {source.url}")
|
|
411
|
+
print(f" Branch: {source.branch}")
|
|
412
|
+
print(f" Priority: {source.priority}")
|
|
413
|
+
print()
|
|
414
|
+
|
|
415
|
+
return 0
|
|
416
|
+
|
|
417
|
+
except Exception as e:
|
|
418
|
+
logger.error(f"Failed to list skill sources: {e}", exc_info=True)
|
|
419
|
+
print(f"❌ Failed to list skill sources: {e}")
|
|
420
|
+
return 1
|
|
421
|
+
|
|
422
|
+
|
|
423
|
+
def handle_remove_skill_source(args) -> int:
|
|
424
|
+
"""Remove a skill source.
|
|
425
|
+
|
|
426
|
+
Args:
|
|
427
|
+
args: Parsed arguments with source_id, force
|
|
428
|
+
|
|
429
|
+
Returns:
|
|
430
|
+
Exit code
|
|
431
|
+
"""
|
|
432
|
+
try:
|
|
433
|
+
config = SkillSourceConfiguration()
|
|
434
|
+
source = config.get_source(args.source_id)
|
|
435
|
+
|
|
436
|
+
if not source:
|
|
437
|
+
print(f"❌ Source not found: {args.source_id}")
|
|
438
|
+
print()
|
|
439
|
+
print("💡 List sources: claude-mpm skill-source list")
|
|
440
|
+
return 1
|
|
441
|
+
|
|
442
|
+
# Confirmation prompt unless --force
|
|
443
|
+
if not args.force:
|
|
444
|
+
print(f"⚠️ Remove skill source: {args.source_id}")
|
|
445
|
+
print(f" URL: {source.url}")
|
|
446
|
+
print()
|
|
447
|
+
response = input(" Continue? (y/N): ").strip().lower()
|
|
448
|
+
if response not in ("y", "yes"):
|
|
449
|
+
print()
|
|
450
|
+
print("❌ Cancelled")
|
|
451
|
+
return 0
|
|
452
|
+
|
|
453
|
+
# Remove source
|
|
454
|
+
config.remove_source(args.source_id)
|
|
455
|
+
|
|
456
|
+
print()
|
|
457
|
+
print(f"✅ Removed skill source: {args.source_id}")
|
|
458
|
+
|
|
459
|
+
return 0
|
|
460
|
+
|
|
461
|
+
except Exception as e:
|
|
462
|
+
logger.error(f"Failed to remove skill source: {e}", exc_info=True)
|
|
463
|
+
print(f"❌ Failed to remove skill source: {e}")
|
|
464
|
+
return 1
|
|
465
|
+
|
|
466
|
+
|
|
467
|
+
def handle_update_skill_sources(args) -> int:
|
|
468
|
+
"""Update (sync) skill sources.
|
|
469
|
+
|
|
470
|
+
Args:
|
|
471
|
+
args: Parsed arguments with source_id (optional), force
|
|
472
|
+
|
|
473
|
+
Returns:
|
|
474
|
+
Exit code
|
|
475
|
+
"""
|
|
476
|
+
try:
|
|
477
|
+
config = SkillSourceConfiguration()
|
|
478
|
+
manager = GitSkillSourceManager(config)
|
|
479
|
+
|
|
480
|
+
if args.source_id:
|
|
481
|
+
# Update specific source
|
|
482
|
+
print(f"🔄 Updating skill source: {args.source_id}")
|
|
483
|
+
|
|
484
|
+
# Verify source exists
|
|
485
|
+
source = config.get_source(args.source_id)
|
|
486
|
+
if not source:
|
|
487
|
+
print(f"❌ Source not found: {args.source_id}")
|
|
488
|
+
print()
|
|
489
|
+
print("💡 List sources: claude-mpm skill-source list")
|
|
490
|
+
return 1
|
|
491
|
+
|
|
492
|
+
# Sync source
|
|
493
|
+
result = manager.sync_source(args.source_id, force=args.force)
|
|
494
|
+
|
|
495
|
+
if result.get("synced"):
|
|
496
|
+
print(f"✅ Successfully updated {args.source_id}")
|
|
497
|
+
skills_count = result.get("skills_discovered", 0)
|
|
498
|
+
print(f" Skills discovered: {skills_count}")
|
|
499
|
+
|
|
500
|
+
if skills_count > 0:
|
|
501
|
+
print()
|
|
502
|
+
print(
|
|
503
|
+
f"💡 View skills: claude-mpm skill-source show {args.source_id} --skills"
|
|
504
|
+
)
|
|
505
|
+
else:
|
|
506
|
+
print(f"❌ Failed to update {args.source_id}")
|
|
507
|
+
error_msg = result.get("error", "Unknown error")
|
|
508
|
+
print(f" Error: {error_msg}")
|
|
509
|
+
return 1
|
|
510
|
+
else:
|
|
511
|
+
# Update all sources
|
|
512
|
+
print("🔄 Updating all skill sources...")
|
|
513
|
+
results = manager.sync_all_sources(force=args.force)
|
|
514
|
+
|
|
515
|
+
success_count = results["synced_count"]
|
|
516
|
+
total_count = success_count + results["failed_count"]
|
|
517
|
+
|
|
518
|
+
print()
|
|
519
|
+
print(f"✅ Updated {success_count}/{total_count} sources")
|
|
520
|
+
print()
|
|
521
|
+
|
|
522
|
+
for source_id, result in results["sources"].items():
|
|
523
|
+
if result.get("synced"):
|
|
524
|
+
skills_count = result.get("skills_discovered", 0)
|
|
525
|
+
print(f" ✅ {source_id}: {skills_count} skills")
|
|
526
|
+
else:
|
|
527
|
+
error_msg = result.get("error", "Unknown error")
|
|
528
|
+
print(f" ❌ {source_id}: {error_msg}")
|
|
529
|
+
|
|
530
|
+
if success_count > 0:
|
|
531
|
+
print()
|
|
532
|
+
print("💡 List all skills: claude-mpm skills list")
|
|
533
|
+
|
|
534
|
+
return 0
|
|
535
|
+
|
|
536
|
+
except Exception as e:
|
|
537
|
+
logger.error(f"Failed to update skill sources: {e}", exc_info=True)
|
|
538
|
+
print(f"❌ Failed to update skill sources: {e}")
|
|
539
|
+
return 1
|
|
540
|
+
|
|
541
|
+
|
|
542
|
+
def handle_enable_skill_source(args) -> int:
|
|
543
|
+
"""Enable a skill source.
|
|
544
|
+
|
|
545
|
+
Args:
|
|
546
|
+
args: Parsed arguments with source_id
|
|
547
|
+
|
|
548
|
+
Returns:
|
|
549
|
+
Exit code
|
|
550
|
+
"""
|
|
551
|
+
try:
|
|
552
|
+
config = SkillSourceConfiguration()
|
|
553
|
+
source = config.get_source(args.source_id)
|
|
554
|
+
|
|
555
|
+
if not source:
|
|
556
|
+
print(f"❌ Source not found: {args.source_id}")
|
|
557
|
+
print()
|
|
558
|
+
print("💡 List sources: claude-mpm skill-source list")
|
|
559
|
+
return 1
|
|
560
|
+
|
|
561
|
+
if source.enabled:
|
|
562
|
+
print(f"⚠️ Source '{args.source_id}' is already enabled")
|
|
563
|
+
return 0
|
|
564
|
+
|
|
565
|
+
# Enable source
|
|
566
|
+
source.enabled = True
|
|
567
|
+
config.update_source(source)
|
|
568
|
+
|
|
569
|
+
print(f"✅ Enabled skill source: {args.source_id}")
|
|
570
|
+
print()
|
|
571
|
+
print(f"💡 Sync skills: claude-mpm skill-source update {args.source_id}")
|
|
572
|
+
|
|
573
|
+
return 0
|
|
574
|
+
|
|
575
|
+
except Exception as e:
|
|
576
|
+
logger.error(f"Failed to enable skill source: {e}", exc_info=True)
|
|
577
|
+
print(f"❌ Failed to enable skill source: {e}")
|
|
578
|
+
return 1
|
|
579
|
+
|
|
580
|
+
|
|
581
|
+
def handle_disable_skill_source(args) -> int:
|
|
582
|
+
"""Disable a skill source.
|
|
583
|
+
|
|
584
|
+
Args:
|
|
585
|
+
args: Parsed arguments with source_id
|
|
586
|
+
|
|
587
|
+
Returns:
|
|
588
|
+
Exit code
|
|
589
|
+
"""
|
|
590
|
+
try:
|
|
591
|
+
config = SkillSourceConfiguration()
|
|
592
|
+
source = config.get_source(args.source_id)
|
|
593
|
+
|
|
594
|
+
if not source:
|
|
595
|
+
print(f"❌ Source not found: {args.source_id}")
|
|
596
|
+
print()
|
|
597
|
+
print("💡 List sources: claude-mpm skill-source list")
|
|
598
|
+
return 1
|
|
599
|
+
|
|
600
|
+
if not source.enabled:
|
|
601
|
+
print(f"⚠️ Source '{args.source_id}' is already disabled")
|
|
602
|
+
return 0
|
|
603
|
+
|
|
604
|
+
# Disable source
|
|
605
|
+
source.enabled = False
|
|
606
|
+
config.update_source(source)
|
|
607
|
+
|
|
608
|
+
print(f"✅ Disabled skill source: {args.source_id}")
|
|
609
|
+
print(" Skills from this source will not be available")
|
|
610
|
+
print()
|
|
611
|
+
print(f"💡 Re-enable: claude-mpm skill-source enable {args.source_id}")
|
|
612
|
+
|
|
613
|
+
return 0
|
|
614
|
+
|
|
615
|
+
except Exception as e:
|
|
616
|
+
logger.error(f"Failed to disable skill source: {e}", exc_info=True)
|
|
617
|
+
print(f"❌ Failed to disable skill source: {e}")
|
|
618
|
+
return 1
|
|
619
|
+
|
|
620
|
+
|
|
621
|
+
def handle_show_skill_source(args) -> int:
|
|
622
|
+
"""Show detailed information about a skill source.
|
|
623
|
+
|
|
624
|
+
Args:
|
|
625
|
+
args: Parsed arguments with source_id, skills
|
|
626
|
+
|
|
627
|
+
Returns:
|
|
628
|
+
Exit code
|
|
629
|
+
"""
|
|
630
|
+
try:
|
|
631
|
+
config = SkillSourceConfiguration()
|
|
632
|
+
source = config.get_source(args.source_id)
|
|
633
|
+
|
|
634
|
+
if not source:
|
|
635
|
+
print(f"❌ Source not found: {args.source_id}")
|
|
636
|
+
print()
|
|
637
|
+
print("💡 List sources: claude-mpm skill-source list")
|
|
638
|
+
return 1
|
|
639
|
+
|
|
640
|
+
# Display source details
|
|
641
|
+
status_emoji = "✅" if source.enabled else "❌"
|
|
642
|
+
status_text = "Enabled" if source.enabled else "Disabled"
|
|
643
|
+
|
|
644
|
+
print()
|
|
645
|
+
print(f"📚 Skill Source: {source.id}")
|
|
646
|
+
print()
|
|
647
|
+
print(f" Status: {status_emoji} {status_text}")
|
|
648
|
+
print(f" URL: {source.url}")
|
|
649
|
+
print(f" Branch: {source.branch}")
|
|
650
|
+
print(f" Priority: {source.priority}")
|
|
651
|
+
print()
|
|
652
|
+
|
|
653
|
+
# Optionally list skills from this source
|
|
654
|
+
if args.skills:
|
|
655
|
+
try:
|
|
656
|
+
discovery = SkillDiscoveryService(config)
|
|
657
|
+
all_skills = discovery.discover_skills()
|
|
658
|
+
|
|
659
|
+
# Filter skills by source
|
|
660
|
+
source_skills = [
|
|
661
|
+
skill
|
|
662
|
+
for skill in all_skills
|
|
663
|
+
if skill.get("source_id") == args.source_id
|
|
664
|
+
]
|
|
665
|
+
|
|
666
|
+
if source_skills:
|
|
667
|
+
print(f" Skills ({len(source_skills)}):")
|
|
668
|
+
print()
|
|
669
|
+
for skill in source_skills:
|
|
670
|
+
print(f" - {skill['name']}")
|
|
671
|
+
if skill.get("description"):
|
|
672
|
+
desc = skill["description"]
|
|
673
|
+
# Truncate long descriptions
|
|
674
|
+
if len(desc) > 70:
|
|
675
|
+
desc = desc[:70] + "..."
|
|
676
|
+
print(f" {desc}")
|
|
677
|
+
print()
|
|
678
|
+
else:
|
|
679
|
+
print(" No skills found in this source")
|
|
680
|
+
print()
|
|
681
|
+
print(
|
|
682
|
+
f"💡 Sync source: claude-mpm skill-source update {args.source_id}"
|
|
683
|
+
)
|
|
684
|
+
except Exception as e:
|
|
685
|
+
logger.warning(f"Failed to load skills: {e}")
|
|
686
|
+
print(f" ⚠️ Could not load skills: {e}")
|
|
687
|
+
print()
|
|
688
|
+
|
|
689
|
+
return 0
|
|
690
|
+
|
|
691
|
+
except Exception as e:
|
|
692
|
+
logger.error(f"Failed to show skill source: {e}", exc_info=True)
|
|
693
|
+
print(f"❌ Failed to show skill source: {e}")
|
|
694
|
+
return 1
|