claude-mpm 4.20.3__py3-none-any.whl → 5.1.8__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.

Potentially problematic release.


This version of claude-mpm might be problematic. Click here for more details.

Files changed (520) hide show
  1. claude_mpm/VERSION +1 -1
  2. claude_mpm/agents/BASE_PM.md +35 -6
  3. claude_mpm/agents/OUTPUT_STYLE.md +3 -48
  4. claude_mpm/agents/PM_INSTRUCTIONS.md +1241 -667
  5. claude_mpm/agents/PM_INSTRUCTIONS_TEACH.md +1322 -0
  6. claude_mpm/agents/WORKFLOW.md +75 -2
  7. claude_mpm/agents/__init__.py +6 -0
  8. claude_mpm/agents/agent_loader.py +1 -4
  9. claude_mpm/agents/base_agent.json +6 -3
  10. claude_mpm/agents/base_agent_loader.py +10 -35
  11. claude_mpm/agents/frontmatter_validator.py +1 -1
  12. claude_mpm/agents/templates/circuit-breakers.md +1254 -0
  13. claude_mpm/agents/templates/context-management-examples.md +544 -0
  14. claude_mpm/agents/templates/{pm_red_flags.md → pm-red-flags.md} +89 -19
  15. claude_mpm/agents/templates/pr-workflow-examples.md +427 -0
  16. claude_mpm/agents/templates/research-gate-examples.md +669 -0
  17. claude_mpm/agents/templates/structured-questions-examples.md +615 -0
  18. claude_mpm/agents/templates/ticket-completeness-examples.md +139 -0
  19. claude_mpm/agents/templates/ticketing-examples.md +277 -0
  20. claude_mpm/cli/__init__.py +37 -2
  21. claude_mpm/cli/commands/__init__.py +2 -0
  22. claude_mpm/cli/commands/agent_source.py +774 -0
  23. claude_mpm/cli/commands/agent_state_manager.py +188 -30
  24. claude_mpm/cli/commands/agents.py +959 -36
  25. claude_mpm/cli/commands/agents_cleanup.py +210 -0
  26. claude_mpm/cli/commands/agents_discover.py +338 -0
  27. claude_mpm/cli/commands/aggregate.py +1 -1
  28. claude_mpm/cli/commands/analyze.py +3 -3
  29. claude_mpm/cli/commands/auto_configure.py +537 -239
  30. claude_mpm/cli/commands/cleanup.py +1 -1
  31. claude_mpm/cli/commands/config.py +7 -4
  32. claude_mpm/cli/commands/configure.py +924 -45
  33. claude_mpm/cli/commands/configure_agent_display.py +4 -4
  34. claude_mpm/cli/commands/configure_navigation.py +63 -46
  35. claude_mpm/cli/commands/debug.py +12 -12
  36. claude_mpm/cli/commands/doctor.py +10 -2
  37. claude_mpm/cli/commands/hook_errors.py +277 -0
  38. claude_mpm/cli/commands/local_deploy.py +1 -4
  39. claude_mpm/cli/commands/mcp_install_commands.py +1 -1
  40. claude_mpm/cli/commands/mpm_init/__init__.py +73 -0
  41. claude_mpm/cli/commands/mpm_init/core.py +573 -0
  42. claude_mpm/cli/commands/mpm_init/display.py +341 -0
  43. claude_mpm/cli/commands/mpm_init/git_activity.py +427 -0
  44. claude_mpm/cli/commands/mpm_init/modes.py +397 -0
  45. claude_mpm/cli/commands/mpm_init/prompts.py +442 -0
  46. claude_mpm/cli/commands/mpm_init_cli.py +396 -0
  47. claude_mpm/cli/commands/mpm_init_handler.py +67 -1
  48. claude_mpm/cli/commands/postmortem.py +401 -0
  49. claude_mpm/cli/commands/run.py +125 -167
  50. claude_mpm/cli/commands/skill_source.py +694 -0
  51. claude_mpm/cli/commands/skills.py +835 -44
  52. claude_mpm/cli/executor.py +78 -3
  53. claude_mpm/cli/interactive/agent_wizard.py +1032 -47
  54. claude_mpm/cli/parsers/agent_source_parser.py +171 -0
  55. claude_mpm/cli/parsers/agents_parser.py +256 -4
  56. claude_mpm/cli/parsers/auto_configure_parser.py +13 -0
  57. claude_mpm/cli/parsers/base_parser.py +53 -0
  58. claude_mpm/cli/parsers/config_parser.py +96 -43
  59. claude_mpm/cli/parsers/mpm_init_parser.py +42 -0
  60. claude_mpm/cli/parsers/skill_source_parser.py +169 -0
  61. claude_mpm/cli/parsers/skills_parser.py +145 -0
  62. claude_mpm/cli/parsers/source_parser.py +138 -0
  63. claude_mpm/cli/startup.py +564 -108
  64. claude_mpm/cli/startup_display.py +480 -0
  65. claude_mpm/cli/utils.py +1 -1
  66. claude_mpm/cli_module/commands.py +1 -1
  67. claude_mpm/commands/{mpm-auto-configure.md → mpm-agents-auto-configure.md} +9 -0
  68. claude_mpm/commands/mpm-agents-detect.md +9 -0
  69. claude_mpm/commands/{mpm-agents.md → mpm-agents-list.md} +9 -0
  70. claude_mpm/commands/mpm-agents-recommend.md +9 -0
  71. claude_mpm/commands/{mpm-config.md → mpm-config-view.md} +9 -0
  72. claude_mpm/commands/mpm-doctor.md +9 -0
  73. claude_mpm/commands/mpm-help.md +17 -2
  74. claude_mpm/commands/mpm-init.md +28 -3
  75. claude_mpm/commands/mpm-monitor.md +9 -0
  76. claude_mpm/commands/mpm-postmortem.md +123 -0
  77. claude_mpm/commands/mpm-session-resume.md +381 -0
  78. claude_mpm/commands/mpm-status.md +9 -0
  79. claude_mpm/commands/{mpm-organize.md → mpm-ticket-organize.md} +9 -0
  80. claude_mpm/commands/mpm-ticket-view.md +552 -0
  81. claude_mpm/commands/mpm-version.md +9 -0
  82. claude_mpm/commands/mpm.md +11 -0
  83. claude_mpm/config/agent_presets.py +488 -0
  84. claude_mpm/config/agent_sources.py +325 -0
  85. claude_mpm/config/skill_presets.py +392 -0
  86. claude_mpm/config/skill_sources.py +590 -0
  87. claude_mpm/constants.py +13 -0
  88. claude_mpm/core/api_validator.py +1 -1
  89. claude_mpm/core/claude_runner.py +19 -35
  90. claude_mpm/core/config.py +24 -0
  91. claude_mpm/core/constants.py +1 -1
  92. claude_mpm/core/framework/__init__.py +3 -16
  93. claude_mpm/core/framework/loaders/file_loader.py +54 -101
  94. claude_mpm/core/framework/loaders/instruction_loader.py +25 -5
  95. claude_mpm/core/framework/processors/metadata_processor.py +1 -1
  96. claude_mpm/core/hook_error_memory.py +381 -0
  97. claude_mpm/core/hook_manager.py +41 -2
  98. claude_mpm/core/interactive_session.py +131 -10
  99. claude_mpm/core/interfaces.py +56 -1
  100. claude_mpm/core/logger.py +3 -1
  101. claude_mpm/core/oneshot_session.py +110 -8
  102. claude_mpm/core/protocols/__init__.py +23 -0
  103. claude_mpm/core/protocols/runner_protocol.py +103 -0
  104. claude_mpm/core/protocols/session_protocol.py +131 -0
  105. claude_mpm/core/shared/singleton_manager.py +11 -4
  106. claude_mpm/core/system_context.py +38 -0
  107. claude_mpm/core/unified_config.py +22 -0
  108. claude_mpm/dashboard/static/css/activity.css +69 -69
  109. claude_mpm/dashboard/static/css/connection-status.css +10 -10
  110. claude_mpm/dashboard/static/css/dashboard.css +15 -15
  111. claude_mpm/dashboard/static/js/components/activity-tree.js +178 -178
  112. claude_mpm/dashboard/static/js/components/agent-hierarchy.js +101 -101
  113. claude_mpm/dashboard/static/js/components/agent-inference.js +31 -31
  114. claude_mpm/dashboard/static/js/components/build-tracker.js +59 -59
  115. claude_mpm/dashboard/static/js/components/code-simple.js +107 -107
  116. claude_mpm/dashboard/static/js/components/connection-debug.js +101 -101
  117. claude_mpm/dashboard/static/js/components/diff-viewer.js +113 -113
  118. claude_mpm/dashboard/static/js/components/event-viewer.js +12 -12
  119. claude_mpm/dashboard/static/js/components/file-change-tracker.js +57 -57
  120. claude_mpm/dashboard/static/js/components/file-change-viewer.js +74 -74
  121. claude_mpm/dashboard/static/js/components/file-tool-tracker.js +6 -6
  122. claude_mpm/dashboard/static/js/components/file-viewer.js +42 -42
  123. claude_mpm/dashboard/static/js/components/module-viewer.js +27 -27
  124. claude_mpm/dashboard/static/js/components/session-manager.js +14 -14
  125. claude_mpm/dashboard/static/js/components/socket-manager.js +1 -1
  126. claude_mpm/dashboard/static/js/components/ui-state-manager.js +14 -14
  127. claude_mpm/dashboard/static/js/components/unified-data-viewer.js +110 -110
  128. claude_mpm/dashboard/static/js/components/working-directory.js +8 -8
  129. claude_mpm/dashboard/static/js/connection-manager.js +76 -76
  130. claude_mpm/dashboard/static/js/dashboard.js +76 -58
  131. claude_mpm/dashboard/static/js/extension-error-handler.js +22 -22
  132. claude_mpm/dashboard/static/js/socket-client.js +138 -121
  133. claude_mpm/dashboard/templates/code_simple.html +23 -23
  134. claude_mpm/dashboard/templates/index.html +18 -18
  135. claude_mpm/experimental/cli_enhancements.py +1 -5
  136. claude_mpm/hooks/claude_hooks/__pycache__/__init__.cpython-313.pyc +0 -0
  137. claude_mpm/hooks/claude_hooks/__pycache__/event_handlers.cpython-313.pyc +0 -0
  138. claude_mpm/hooks/claude_hooks/__pycache__/hook_handler.cpython-313.pyc +0 -0
  139. claude_mpm/hooks/claude_hooks/__pycache__/memory_integration.cpython-313.pyc +0 -0
  140. claude_mpm/hooks/claude_hooks/__pycache__/response_tracking.cpython-313.pyc +0 -0
  141. claude_mpm/hooks/claude_hooks/__pycache__/tool_analysis.cpython-313.pyc +0 -0
  142. claude_mpm/hooks/claude_hooks/event_handlers.py +3 -1
  143. claude_mpm/hooks/claude_hooks/hook_handler.py +24 -7
  144. claude_mpm/hooks/claude_hooks/installer.py +45 -0
  145. claude_mpm/hooks/claude_hooks/memory_integration.py +12 -1
  146. claude_mpm/hooks/claude_hooks/services/__pycache__/__init__.cpython-313.pyc +0 -0
  147. claude_mpm/hooks/claude_hooks/services/__pycache__/connection_manager_http.cpython-313.pyc +0 -0
  148. claude_mpm/hooks/claude_hooks/services/__pycache__/duplicate_detector.cpython-313.pyc +0 -0
  149. claude_mpm/hooks/claude_hooks/services/__pycache__/state_manager.cpython-313.pyc +0 -0
  150. claude_mpm/hooks/claude_hooks/services/__pycache__/subagent_processor.cpython-313.pyc +0 -0
  151. claude_mpm/hooks/failure_learning/__init__.py +2 -8
  152. claude_mpm/hooks/failure_learning/failure_detection_hook.py +1 -6
  153. claude_mpm/hooks/failure_learning/fix_detection_hook.py +1 -6
  154. claude_mpm/hooks/failure_learning/learning_extraction_hook.py +1 -6
  155. claude_mpm/hooks/kuzu_response_hook.py +1 -5
  156. claude_mpm/hooks/templates/pre_tool_use_simple.py +78 -0
  157. claude_mpm/hooks/templates/pre_tool_use_template.py +323 -0
  158. claude_mpm/models/git_repository.py +198 -0
  159. claude_mpm/scripts/claude-hook-handler.sh +3 -3
  160. claude_mpm/scripts/start_activity_logging.py +3 -1
  161. claude_mpm/services/agents/agent_builder.py +45 -9
  162. claude_mpm/services/agents/agent_preset_service.py +238 -0
  163. claude_mpm/services/agents/agent_selection_service.py +484 -0
  164. claude_mpm/services/agents/auto_deploy_index_parser.py +569 -0
  165. claude_mpm/services/agents/cache_git_manager.py +621 -0
  166. claude_mpm/services/agents/deployment/agent_deployment.py +126 -2
  167. claude_mpm/services/agents/deployment/agent_discovery_service.py +105 -73
  168. claude_mpm/services/agents/deployment/agent_format_converter.py +1 -1
  169. claude_mpm/services/agents/deployment/agent_lifecycle_manager.py +1 -5
  170. claude_mpm/services/agents/deployment/agent_metrics_collector.py +3 -3
  171. claude_mpm/services/agents/deployment/agent_restore_handler.py +1 -4
  172. claude_mpm/services/agents/deployment/agent_template_builder.py +236 -15
  173. claude_mpm/services/agents/deployment/agents_directory_resolver.py +101 -15
  174. claude_mpm/services/agents/deployment/async_agent_deployment.py +2 -1
  175. claude_mpm/services/agents/deployment/facade/deployment_facade.py +3 -3
  176. claude_mpm/services/agents/deployment/multi_source_deployment_service.py +115 -15
  177. claude_mpm/services/agents/deployment/pipeline/pipeline_executor.py +2 -2
  178. claude_mpm/services/agents/deployment/refactored_agent_deployment_service.py +1 -4
  179. claude_mpm/services/agents/deployment/remote_agent_discovery_service.py +363 -0
  180. claude_mpm/services/agents/deployment/single_agent_deployer.py +2 -2
  181. claude_mpm/services/agents/deployment/system_instructions_deployer.py +168 -46
  182. claude_mpm/services/agents/deployment/validation/deployment_validator.py +2 -2
  183. claude_mpm/services/agents/git_source_manager.py +629 -0
  184. claude_mpm/services/agents/loading/framework_agent_loader.py +9 -12
  185. claude_mpm/services/agents/local_template_manager.py +50 -10
  186. claude_mpm/services/agents/single_tier_deployment_service.py +696 -0
  187. claude_mpm/services/agents/sources/__init__.py +13 -0
  188. claude_mpm/services/agents/sources/agent_sync_state.py +516 -0
  189. claude_mpm/services/agents/sources/git_source_sync_service.py +1087 -0
  190. claude_mpm/services/agents/startup_sync.py +239 -0
  191. claude_mpm/services/agents/toolchain_detector.py +474 -0
  192. claude_mpm/services/analysis/__init__.py +25 -0
  193. claude_mpm/services/analysis/postmortem_reporter.py +474 -0
  194. claude_mpm/services/analysis/postmortem_service.py +765 -0
  195. claude_mpm/services/cli/session_pause_manager.py +504 -0
  196. claude_mpm/services/cli/session_resume_helper.py +36 -16
  197. claude_mpm/services/cli/unified_dashboard_manager.py +1 -1
  198. claude_mpm/services/command_deployment_service.py +200 -6
  199. claude_mpm/services/core/base.py +31 -11
  200. claude_mpm/services/core/interfaces/__init__.py +1 -3
  201. claude_mpm/services/core/interfaces/health.py +1 -4
  202. claude_mpm/services/core/interfaces.py +56 -1
  203. claude_mpm/services/core/models/__init__.py +2 -11
  204. claude_mpm/services/core/models/agent_config.py +3 -0
  205. claude_mpm/services/core/models/process.py +4 -0
  206. claude_mpm/services/diagnostics/checks/__init__.py +4 -0
  207. claude_mpm/services/diagnostics/checks/agent_check.py +0 -2
  208. claude_mpm/services/diagnostics/checks/agent_sources_check.py +577 -0
  209. claude_mpm/services/diagnostics/checks/instructions_check.py +1 -2
  210. claude_mpm/services/diagnostics/checks/mcp_check.py +0 -1
  211. claude_mpm/services/diagnostics/checks/mcp_services_check.py +7 -15
  212. claude_mpm/services/diagnostics/checks/monitor_check.py +0 -1
  213. claude_mpm/services/diagnostics/checks/skill_sources_check.py +587 -0
  214. claude_mpm/services/diagnostics/diagnostic_runner.py +9 -0
  215. claude_mpm/services/diagnostics/doctor_reporter.py +40 -10
  216. claude_mpm/services/diagnostics/models.py +21 -0
  217. claude_mpm/services/event_bus/direct_relay.py +3 -3
  218. claude_mpm/services/event_bus/event_bus.py +36 -3
  219. claude_mpm/services/event_bus/relay.py +23 -7
  220. claude_mpm/services/events/consumers/logging.py +1 -2
  221. claude_mpm/services/git/__init__.py +21 -0
  222. claude_mpm/services/git/git_operations_service.py +494 -0
  223. claude_mpm/services/github/__init__.py +21 -0
  224. claude_mpm/services/github/github_cli_service.py +397 -0
  225. claude_mpm/services/infrastructure/monitoring/__init__.py +1 -5
  226. claude_mpm/services/infrastructure/monitoring/aggregator.py +1 -6
  227. claude_mpm/services/infrastructure/monitoring/resources.py +1 -1
  228. claude_mpm/services/instructions/__init__.py +9 -0
  229. claude_mpm/services/instructions/instruction_cache_service.py +374 -0
  230. claude_mpm/services/local_ops/__init__.py +5 -13
  231. claude_mpm/services/local_ops/health_checks/__init__.py +1 -3
  232. claude_mpm/services/local_ops/health_manager.py +1 -4
  233. claude_mpm/services/local_ops/process_manager.py +1 -1
  234. claude_mpm/services/local_ops/resource_monitor.py +2 -2
  235. claude_mpm/services/mcp_config_manager.py +75 -145
  236. claude_mpm/services/mcp_gateway/auto_configure.py +31 -25
  237. claude_mpm/services/mcp_gateway/config/configuration.py +1 -1
  238. claude_mpm/services/mcp_gateway/core/process_pool.py +41 -26
  239. claude_mpm/services/mcp_gateway/server/mcp_gateway.py +1 -6
  240. claude_mpm/services/mcp_gateway/server/stdio_server.py +0 -2
  241. claude_mpm/services/mcp_gateway/tools/document_summarizer.py +1 -1
  242. claude_mpm/services/mcp_gateway/tools/external_mcp_services.py +26 -21
  243. claude_mpm/services/mcp_gateway/tools/kuzu_memory_service.py +6 -2
  244. claude_mpm/services/mcp_service_verifier.py +6 -3
  245. claude_mpm/services/memory/failure_tracker.py +19 -4
  246. claude_mpm/services/memory/optimizer.py +1 -1
  247. claude_mpm/services/model/model_router.py +8 -9
  248. claude_mpm/services/monitor/daemon.py +29 -9
  249. claude_mpm/services/monitor/daemon_manager.py +96 -19
  250. claude_mpm/services/monitor/server.py +2 -2
  251. claude_mpm/services/native_agent_converter.py +356 -0
  252. claude_mpm/services/port_manager.py +1 -1
  253. claude_mpm/services/pr/__init__.py +14 -0
  254. claude_mpm/services/pr/pr_template_service.py +329 -0
  255. claude_mpm/services/project/documentation_manager.py +2 -1
  256. claude_mpm/services/project/project_organizer.py +4 -0
  257. claude_mpm/services/project/toolchain_analyzer.py +3 -1
  258. claude_mpm/services/runner_configuration_service.py +17 -3
  259. claude_mpm/services/self_upgrade_service.py +165 -7
  260. claude_mpm/services/session_management_service.py +16 -4
  261. claude_mpm/services/skills/__init__.py +18 -0
  262. claude_mpm/services/skills/git_skill_source_manager.py +1169 -0
  263. claude_mpm/services/skills/skill_discovery_service.py +568 -0
  264. claude_mpm/services/skills_config.py +547 -0
  265. claude_mpm/services/skills_deployer.py +955 -0
  266. claude_mpm/services/socketio/handlers/connection.py +1 -1
  267. claude_mpm/services/socketio/handlers/git.py +2 -2
  268. claude_mpm/services/socketio/server/core.py +1 -4
  269. claude_mpm/services/socketio/server/main.py +1 -3
  270. claude_mpm/services/system_instructions_service.py +1 -3
  271. claude_mpm/services/unified/analyzer_strategies/performance_analyzer.py +0 -3
  272. claude_mpm/services/unified/analyzer_strategies/security_analyzer.py +0 -1
  273. claude_mpm/services/unified/deployment_strategies/cloud_strategies.py +1 -1
  274. claude_mpm/services/unified/deployment_strategies/vercel.py +1 -5
  275. claude_mpm/services/unified/unified_deployment.py +1 -5
  276. claude_mpm/services/version_control/conflict_resolution.py +6 -4
  277. claude_mpm/services/visualization/__init__.py +1 -5
  278. claude_mpm/services/visualization/mermaid_generator.py +2 -3
  279. claude_mpm/skills/__init__.py +3 -3
  280. claude_mpm/skills/agent_skills_injector.py +42 -49
  281. claude_mpm/skills/bundled/infrastructure/env-manager/scripts/validate_env.py +576 -0
  282. claude_mpm/skills/bundled/main/mcp-builder/scripts/connections.py +17 -10
  283. claude_mpm/skills/bundled/main/mcp-builder/scripts/evaluation.py +92 -39
  284. claude_mpm/skills/bundled/main/skill-creator/scripts/init_skill.py +13 -12
  285. claude_mpm/skills/bundled/main/skill-creator/scripts/package_skill.py +5 -3
  286. claude_mpm/skills/bundled/main/skill-creator/scripts/quick_validate.py +19 -12
  287. claude_mpm/skills/bundled/performance-profiling.md +6 -0
  288. claude_mpm/skills/bundled/testing/webapp-testing/examples/console_logging.py +6 -6
  289. claude_mpm/skills/bundled/testing/webapp-testing/examples/element_discovery.py +13 -9
  290. claude_mpm/skills/bundled/testing/webapp-testing/examples/static_html_automation.py +8 -8
  291. claude_mpm/skills/bundled/testing/webapp-testing/scripts/with_server.py +37 -15
  292. claude_mpm/skills/skills_registry.py +44 -48
  293. claude_mpm/skills/skills_service.py +117 -108
  294. claude_mpm/templates/questions/__init__.py +38 -0
  295. claude_mpm/templates/questions/base.py +193 -0
  296. claude_mpm/templates/questions/pr_strategy.py +311 -0
  297. claude_mpm/templates/questions/project_init.py +385 -0
  298. claude_mpm/templates/questions/ticket_mgmt.py +394 -0
  299. claude_mpm/tools/__main__.py +8 -8
  300. claude_mpm/tools/code_tree_analyzer/__init__.py +45 -0
  301. claude_mpm/tools/code_tree_analyzer/analysis.py +299 -0
  302. claude_mpm/tools/code_tree_analyzer/cache.py +131 -0
  303. claude_mpm/tools/code_tree_analyzer/core.py +380 -0
  304. claude_mpm/tools/code_tree_analyzer/discovery.py +403 -0
  305. claude_mpm/tools/code_tree_analyzer/events.py +168 -0
  306. claude_mpm/tools/code_tree_analyzer/gitignore.py +308 -0
  307. claude_mpm/tools/code_tree_analyzer/models.py +39 -0
  308. claude_mpm/tools/code_tree_analyzer/multilang_analyzer.py +224 -0
  309. claude_mpm/tools/code_tree_analyzer/python_analyzer.py +284 -0
  310. claude_mpm/utils/agent_dependency_loader.py +80 -13
  311. claude_mpm/utils/agent_filters.py +288 -0
  312. claude_mpm/utils/dependency_cache.py +3 -1
  313. claude_mpm/utils/gitignore.py +244 -0
  314. claude_mpm/utils/log_cleanup.py +3 -3
  315. claude_mpm/utils/migration.py +372 -0
  316. claude_mpm/utils/progress.py +387 -0
  317. claude_mpm/utils/robust_installer.py +3 -5
  318. claude_mpm/utils/structured_questions.py +619 -0
  319. {claude_mpm-4.20.3.dist-info → claude_mpm-5.1.8.dist-info}/METADATA +496 -65
  320. {claude_mpm-4.20.3.dist-info → claude_mpm-5.1.8.dist-info}/RECORD +328 -416
  321. claude_mpm/agents/templates/.claude-mpm/memories/README.md +0 -17
  322. claude_mpm/agents/templates/.claude-mpm/memories/engineer_memories.md +0 -3
  323. claude_mpm/agents/templates/agent-manager.json +0 -273
  324. claude_mpm/agents/templates/agentic-coder-optimizer.json +0 -248
  325. claude_mpm/agents/templates/api_qa.json +0 -180
  326. claude_mpm/agents/templates/circuit_breakers.md +0 -638
  327. claude_mpm/agents/templates/clerk-ops.json +0 -235
  328. claude_mpm/agents/templates/code_analyzer.json +0 -101
  329. claude_mpm/agents/templates/content-agent.json +0 -358
  330. claude_mpm/agents/templates/dart_engineer.json +0 -307
  331. claude_mpm/agents/templates/data_engineer.json +0 -225
  332. claude_mpm/agents/templates/documentation.json +0 -211
  333. claude_mpm/agents/templates/engineer.json +0 -210
  334. claude_mpm/agents/templates/gcp_ops_agent.json +0 -253
  335. claude_mpm/agents/templates/golang_engineer.json +0 -270
  336. claude_mpm/agents/templates/imagemagick.json +0 -264
  337. claude_mpm/agents/templates/java_engineer.json +0 -346
  338. claude_mpm/agents/templates/local_ops_agent.json +0 -1840
  339. claude_mpm/agents/templates/logs/prompts/agent_engineer_20250826_014258_728.md +0 -39
  340. claude_mpm/agents/templates/logs/prompts/agent_engineer_20250901_010124_142.md +0 -400
  341. claude_mpm/agents/templates/memory_manager.json +0 -158
  342. claude_mpm/agents/templates/nextjs_engineer.json +0 -285
  343. claude_mpm/agents/templates/ops.json +0 -185
  344. claude_mpm/agents/templates/php-engineer.json +0 -281
  345. claude_mpm/agents/templates/product_owner.json +0 -338
  346. claude_mpm/agents/templates/project_organizer.json +0 -140
  347. claude_mpm/agents/templates/prompt-engineer.json +0 -737
  348. claude_mpm/agents/templates/python_engineer.json +0 -387
  349. claude_mpm/agents/templates/qa.json +0 -242
  350. claude_mpm/agents/templates/react_engineer.json +0 -238
  351. claude_mpm/agents/templates/refactoring_engineer.json +0 -276
  352. claude_mpm/agents/templates/research.json +0 -188
  353. claude_mpm/agents/templates/ruby-engineer.json +0 -280
  354. claude_mpm/agents/templates/rust_engineer.json +0 -275
  355. claude_mpm/agents/templates/security.json +0 -202
  356. claude_mpm/agents/templates/svelte-engineer.json +0 -225
  357. claude_mpm/agents/templates/ticketing.json +0 -177
  358. claude_mpm/agents/templates/typescript_engineer.json +0 -285
  359. claude_mpm/agents/templates/vercel_ops_agent.json +0 -412
  360. claude_mpm/agents/templates/version_control.json +0 -157
  361. claude_mpm/agents/templates/web_qa.json +0 -399
  362. claude_mpm/agents/templates/web_ui.json +0 -189
  363. claude_mpm/cli/commands/mpm_init.py +0 -2093
  364. claude_mpm/commands/mpm-tickets.md +0 -102
  365. claude_mpm/dashboard/.claude-mpm/socketio-instances.json +0 -1
  366. claude_mpm/dashboard/react/components/DataInspector/DataInspector.module.css +0 -188
  367. claude_mpm/dashboard/react/components/EventViewer/EventViewer.module.css +0 -156
  368. claude_mpm/dashboard/react/components/shared/ConnectionStatus.module.css +0 -38
  369. claude_mpm/dashboard/react/components/shared/FilterBar.module.css +0 -92
  370. claude_mpm/dashboard/static/archive/activity_dashboard_fixed.html +0 -248
  371. claude_mpm/dashboard/static/archive/activity_dashboard_test.html +0 -61
  372. claude_mpm/dashboard/static/archive/test_activity_connection.html +0 -179
  373. claude_mpm/dashboard/static/archive/test_claude_tree_tab.html +0 -68
  374. claude_mpm/dashboard/static/archive/test_dashboard.html +0 -409
  375. claude_mpm/dashboard/static/archive/test_dashboard_fixed.html +0 -519
  376. claude_mpm/dashboard/static/archive/test_dashboard_verification.html +0 -181
  377. claude_mpm/dashboard/static/archive/test_file_data.html +0 -315
  378. claude_mpm/dashboard/static/archive/test_file_tree_empty_state.html +0 -243
  379. claude_mpm/dashboard/static/archive/test_file_tree_fix.html +0 -234
  380. claude_mpm/dashboard/static/archive/test_file_tree_rename.html +0 -117
  381. claude_mpm/dashboard/static/archive/test_file_tree_tab.html +0 -115
  382. claude_mpm/dashboard/static/archive/test_file_viewer.html +0 -224
  383. claude_mpm/dashboard/static/archive/test_final_activity.html +0 -220
  384. claude_mpm/dashboard/static/archive/test_tab_fix.html +0 -139
  385. claude_mpm/dashboard/static/built/assets/events.DjpNxWNo.css +0 -1
  386. claude_mpm/dashboard/static/built/components/activity-tree.js +0 -2
  387. claude_mpm/dashboard/static/built/components/agent-hierarchy.js +0 -777
  388. claude_mpm/dashboard/static/built/components/agent-inference.js +0 -2
  389. claude_mpm/dashboard/static/built/components/build-tracker.js +0 -333
  390. claude_mpm/dashboard/static/built/components/code-simple.js +0 -857
  391. claude_mpm/dashboard/static/built/components/code-tree/tree-breadcrumb.js +0 -353
  392. claude_mpm/dashboard/static/built/components/code-tree/tree-constants.js +0 -235
  393. claude_mpm/dashboard/static/built/components/code-tree/tree-search.js +0 -409
  394. claude_mpm/dashboard/static/built/components/code-tree/tree-utils.js +0 -435
  395. claude_mpm/dashboard/static/built/components/code-tree.js +0 -2
  396. claude_mpm/dashboard/static/built/components/code-viewer.js +0 -2
  397. claude_mpm/dashboard/static/built/components/connection-debug.js +0 -654
  398. claude_mpm/dashboard/static/built/components/diff-viewer.js +0 -891
  399. claude_mpm/dashboard/static/built/components/event-processor.js +0 -2
  400. claude_mpm/dashboard/static/built/components/event-viewer.js +0 -2
  401. claude_mpm/dashboard/static/built/components/export-manager.js +0 -2
  402. claude_mpm/dashboard/static/built/components/file-change-tracker.js +0 -443
  403. claude_mpm/dashboard/static/built/components/file-change-viewer.js +0 -690
  404. claude_mpm/dashboard/static/built/components/file-tool-tracker.js +0 -2
  405. claude_mpm/dashboard/static/built/components/file-viewer.js +0 -2
  406. claude_mpm/dashboard/static/built/components/hud-library-loader.js +0 -2
  407. claude_mpm/dashboard/static/built/components/hud-manager.js +0 -2
  408. claude_mpm/dashboard/static/built/components/hud-visualizer.js +0 -2
  409. claude_mpm/dashboard/static/built/components/module-viewer.js +0 -2
  410. claude_mpm/dashboard/static/built/components/nav-bar.js +0 -145
  411. claude_mpm/dashboard/static/built/components/page-structure.js +0 -429
  412. claude_mpm/dashboard/static/built/components/session-manager.js +0 -2
  413. claude_mpm/dashboard/static/built/components/socket-manager.js +0 -2
  414. claude_mpm/dashboard/static/built/components/ui-state-manager.js +0 -2
  415. claude_mpm/dashboard/static/built/components/unified-data-viewer.js +0 -2
  416. claude_mpm/dashboard/static/built/components/working-directory.js +0 -2
  417. claude_mpm/dashboard/static/built/connection-manager.js +0 -536
  418. claude_mpm/dashboard/static/built/dashboard.js +0 -2
  419. claude_mpm/dashboard/static/built/extension-error-handler.js +0 -164
  420. claude_mpm/dashboard/static/built/react/events.js +0 -30
  421. claude_mpm/dashboard/static/built/shared/dom-helpers.js +0 -396
  422. claude_mpm/dashboard/static/built/shared/event-bus.js +0 -330
  423. claude_mpm/dashboard/static/built/shared/event-filter-service.js +0 -540
  424. claude_mpm/dashboard/static/built/shared/logger.js +0 -385
  425. claude_mpm/dashboard/static/built/shared/page-structure.js +0 -249
  426. claude_mpm/dashboard/static/built/shared/tooltip-service.js +0 -253
  427. claude_mpm/dashboard/static/built/socket-client.js +0 -2
  428. claude_mpm/dashboard/static/built/tab-isolation-fix.js +0 -185
  429. claude_mpm/dashboard/static/dist/assets/events.DjpNxWNo.css +0 -1
  430. claude_mpm/dashboard/static/dist/components/activity-tree.js +0 -2
  431. claude_mpm/dashboard/static/dist/components/agent-inference.js +0 -2
  432. claude_mpm/dashboard/static/dist/components/code-tree.js +0 -2
  433. claude_mpm/dashboard/static/dist/components/code-viewer.js +0 -2
  434. claude_mpm/dashboard/static/dist/components/event-processor.js +0 -2
  435. claude_mpm/dashboard/static/dist/components/event-viewer.js +0 -2
  436. claude_mpm/dashboard/static/dist/components/export-manager.js +0 -2
  437. claude_mpm/dashboard/static/dist/components/file-tool-tracker.js +0 -2
  438. claude_mpm/dashboard/static/dist/components/file-viewer.js +0 -2
  439. claude_mpm/dashboard/static/dist/components/hud-library-loader.js +0 -2
  440. claude_mpm/dashboard/static/dist/components/hud-manager.js +0 -2
  441. claude_mpm/dashboard/static/dist/components/hud-visualizer.js +0 -2
  442. claude_mpm/dashboard/static/dist/components/module-viewer.js +0 -2
  443. claude_mpm/dashboard/static/dist/components/session-manager.js +0 -2
  444. claude_mpm/dashboard/static/dist/components/socket-manager.js +0 -2
  445. claude_mpm/dashboard/static/dist/components/ui-state-manager.js +0 -2
  446. claude_mpm/dashboard/static/dist/components/unified-data-viewer.js +0 -2
  447. claude_mpm/dashboard/static/dist/components/working-directory.js +0 -2
  448. claude_mpm/dashboard/static/dist/dashboard.js +0 -2
  449. claude_mpm/dashboard/static/dist/react/events.js +0 -30
  450. claude_mpm/dashboard/static/dist/socket-client.js +0 -2
  451. claude_mpm/dashboard/static/events.html +0 -607
  452. claude_mpm/dashboard/static/index.html +0 -635
  453. claude_mpm/dashboard/static/js/shared/dom-helpers.js +0 -396
  454. claude_mpm/dashboard/static/js/shared/event-bus.js +0 -330
  455. claude_mpm/dashboard/static/js/shared/logger.js +0 -385
  456. claude_mpm/dashboard/static/js/shared/tooltip-service.js +0 -253
  457. claude_mpm/dashboard/static/js/stores/dashboard-store.js +0 -562
  458. claude_mpm/dashboard/static/legacy/activity.html +0 -736
  459. claude_mpm/dashboard/static/legacy/agents.html +0 -786
  460. claude_mpm/dashboard/static/legacy/files.html +0 -747
  461. claude_mpm/dashboard/static/legacy/tools.html +0 -831
  462. claude_mpm/dashboard/static/monitors.html +0 -431
  463. claude_mpm/dashboard/static/production/events.html +0 -659
  464. claude_mpm/dashboard/static/production/main.html +0 -698
  465. claude_mpm/dashboard/static/production/monitors.html +0 -483
  466. claude_mpm/dashboard/static/test-archive/dashboard.html +0 -635
  467. claude_mpm/dashboard/static/test-archive/debug-events.html +0 -147
  468. claude_mpm/dashboard/static/test-archive/test-navigation.html +0 -256
  469. claude_mpm/dashboard/static/test-archive/test-react-exports.html +0 -180
  470. claude_mpm/dashboard/static/test-archive/test_debug.html +0 -25
  471. claude_mpm/skills/bundled/collaboration/brainstorming/SKILL.md +0 -75
  472. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/SKILL.md +0 -184
  473. claude_mpm/skills/bundled/collaboration/requesting-code-review/SKILL.md +0 -107
  474. claude_mpm/skills/bundled/collaboration/requesting-code-review/code-reviewer.md +0 -146
  475. claude_mpm/skills/bundled/collaboration/writing-plans/SKILL.md +0 -118
  476. claude_mpm/skills/bundled/debugging/root-cause-tracing/SKILL.md +0 -177
  477. claude_mpm/skills/bundled/debugging/systematic-debugging/CREATION-LOG.md +0 -119
  478. claude_mpm/skills/bundled/debugging/systematic-debugging/SKILL.md +0 -148
  479. claude_mpm/skills/bundled/debugging/systematic-debugging/references/anti-patterns.md +0 -483
  480. claude_mpm/skills/bundled/debugging/systematic-debugging/references/examples.md +0 -452
  481. claude_mpm/skills/bundled/debugging/systematic-debugging/references/troubleshooting.md +0 -449
  482. claude_mpm/skills/bundled/debugging/systematic-debugging/references/workflow.md +0 -411
  483. claude_mpm/skills/bundled/debugging/systematic-debugging/test-academic.md +0 -14
  484. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-1.md +0 -58
  485. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-2.md +0 -68
  486. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-3.md +0 -69
  487. claude_mpm/skills/bundled/debugging/verification-before-completion/SKILL.md +0 -175
  488. claude_mpm/skills/bundled/debugging/verification-before-completion/references/common-failures.md +0 -213
  489. claude_mpm/skills/bundled/debugging/verification-before-completion/references/gate-function.md +0 -314
  490. claude_mpm/skills/bundled/debugging/verification-before-completion/references/verification-patterns.md +0 -227
  491. claude_mpm/skills/bundled/main/artifacts-builder/SKILL.md +0 -74
  492. claude_mpm/skills/bundled/main/internal-comms/SKILL.md +0 -32
  493. claude_mpm/skills/bundled/main/internal-comms/examples/3p-updates.md +0 -47
  494. claude_mpm/skills/bundled/main/internal-comms/examples/company-newsletter.md +0 -65
  495. claude_mpm/skills/bundled/main/internal-comms/examples/faq-answers.md +0 -30
  496. claude_mpm/skills/bundled/main/internal-comms/examples/general-comms.md +0 -16
  497. claude_mpm/skills/bundled/main/mcp-builder/SKILL.md +0 -328
  498. claude_mpm/skills/bundled/main/mcp-builder/reference/evaluation.md +0 -602
  499. claude_mpm/skills/bundled/main/mcp-builder/reference/mcp_best_practices.md +0 -915
  500. claude_mpm/skills/bundled/main/mcp-builder/reference/node_mcp_server.md +0 -916
  501. claude_mpm/skills/bundled/main/mcp-builder/reference/python_mcp_server.md +0 -752
  502. claude_mpm/skills/bundled/main/skill-creator/SKILL.md +0 -209
  503. claude_mpm/skills/bundled/testing/condition-based-waiting/SKILL.md +0 -123
  504. claude_mpm/skills/bundled/testing/test-driven-development/SKILL.md +0 -145
  505. claude_mpm/skills/bundled/testing/test-driven-development/references/anti-patterns.md +0 -543
  506. claude_mpm/skills/bundled/testing/test-driven-development/references/examples.md +0 -741
  507. claude_mpm/skills/bundled/testing/test-driven-development/references/integration.md +0 -470
  508. claude_mpm/skills/bundled/testing/test-driven-development/references/philosophy.md +0 -458
  509. claude_mpm/skills/bundled/testing/test-driven-development/references/workflow.md +0 -639
  510. claude_mpm/skills/bundled/testing/testing-anti-patterns/SKILL.md +0 -304
  511. claude_mpm/skills/bundled/testing/webapp-testing/SKILL.md +0 -96
  512. claude_mpm/tools/code_tree_analyzer.py +0 -1825
  513. /claude_mpm/agents/templates/{git_file_tracking.md → git-file-tracking.md} +0 -0
  514. /claude_mpm/agents/templates/{pm_examples.md → pm-examples.md} +0 -0
  515. /claude_mpm/agents/templates/{response_format.md → response-format.md} +0 -0
  516. /claude_mpm/agents/templates/{validation_templates.md → validation-templates.md} +0 -0
  517. {claude_mpm-4.20.3.dist-info → claude_mpm-5.1.8.dist-info}/WHEEL +0 -0
  518. {claude_mpm-4.20.3.dist-info → claude_mpm-5.1.8.dist-info}/entry_points.txt +0 -0
  519. {claude_mpm-4.20.3.dist-info → claude_mpm-5.1.8.dist-info}/licenses/LICENSE +0 -0
  520. {claude_mpm-4.20.3.dist-info → claude_mpm-5.1.8.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