claude-mpm 4.24.0__py3-none-any.whl → 5.4.41__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 (623) hide show
  1. claude_mpm/VERSION +1 -1
  2. claude_mpm/__init__.py +4 -0
  3. claude_mpm/agents/BASE_AGENT.md +164 -0
  4. claude_mpm/agents/{OUTPUT_STYLE.md → CLAUDE_MPM_OUTPUT_STYLE.md} +3 -48
  5. claude_mpm/agents/CLAUDE_MPM_TEACHER_OUTPUT_STYLE.md +2002 -0
  6. claude_mpm/agents/MEMORY.md +1 -1
  7. claude_mpm/agents/PM_INSTRUCTIONS.md +735 -925
  8. claude_mpm/agents/WORKFLOW.md +5 -254
  9. claude_mpm/agents/__init__.py +6 -0
  10. claude_mpm/agents/agent_loader.py +14 -48
  11. claude_mpm/agents/base_agent.json +7 -4
  12. claude_mpm/agents/frontmatter_validator.py +71 -3
  13. claude_mpm/agents/templates/circuit-breakers.md +1391 -0
  14. claude_mpm/agents/templates/context-management-examples.md +544 -0
  15. claude_mpm/agents/templates/{pm_red_flags.md → pm-red-flags.md} +48 -0
  16. claude_mpm/agents/templates/pr-workflow-examples.md +427 -0
  17. claude_mpm/agents/templates/research-gate-examples.md +669 -0
  18. claude_mpm/agents/templates/structured-questions-examples.md +615 -0
  19. claude_mpm/agents/templates/ticket-completeness-examples.md +139 -0
  20. claude_mpm/agents/templates/ticketing-examples.md +277 -0
  21. claude_mpm/cli/__init__.py +37 -2
  22. claude_mpm/cli/__main__.py +4 -0
  23. claude_mpm/cli/chrome_devtools_installer.py +175 -0
  24. claude_mpm/cli/commands/__init__.py +2 -0
  25. claude_mpm/cli/commands/agent_source.py +774 -0
  26. claude_mpm/cli/commands/agent_state_manager.py +180 -31
  27. claude_mpm/cli/commands/agents.py +1116 -55
  28. claude_mpm/cli/commands/agents_cleanup.py +210 -0
  29. claude_mpm/cli/commands/agents_discover.py +338 -0
  30. claude_mpm/cli/commands/aggregate.py +1 -1
  31. claude_mpm/cli/commands/analyze.py +3 -3
  32. claude_mpm/cli/commands/auto_configure.py +725 -242
  33. claude_mpm/cli/commands/config.py +95 -6
  34. claude_mpm/cli/commands/configure.py +1875 -46
  35. claude_mpm/cli/commands/configure_agent_display.py +29 -10
  36. claude_mpm/cli/commands/configure_navigation.py +63 -46
  37. claude_mpm/cli/commands/debug.py +12 -12
  38. claude_mpm/cli/commands/doctor.py +10 -2
  39. claude_mpm/cli/commands/hook_errors.py +277 -0
  40. claude_mpm/cli/commands/local_deploy.py +1 -4
  41. claude_mpm/cli/commands/mcp_install_commands.py +1 -1
  42. claude_mpm/cli/commands/mpm_init/core.py +229 -2
  43. claude_mpm/cli/commands/mpm_init/git_activity.py +10 -10
  44. claude_mpm/cli/commands/mpm_init/knowledge_extractor.py +481 -0
  45. claude_mpm/cli/commands/mpm_init/prompts.py +286 -6
  46. claude_mpm/cli/commands/postmortem.py +401 -0
  47. claude_mpm/cli/commands/profile.py +277 -0
  48. claude_mpm/cli/commands/run.py +123 -165
  49. claude_mpm/cli/commands/skill_source.py +694 -0
  50. claude_mpm/cli/commands/skills.py +782 -20
  51. claude_mpm/cli/commands/summarize.py +413 -0
  52. claude_mpm/cli/executor.py +96 -3
  53. claude_mpm/cli/interactive/agent_wizard.py +1030 -45
  54. claude_mpm/cli/parsers/agent_source_parser.py +171 -0
  55. claude_mpm/cli/parsers/agents_parser.py +307 -10
  56. claude_mpm/cli/parsers/auto_configure_parser.py +13 -138
  57. claude_mpm/cli/parsers/base_parser.py +65 -0
  58. claude_mpm/cli/parsers/config_parser.py +162 -39
  59. claude_mpm/cli/parsers/profile_parser.py +148 -0
  60. claude_mpm/cli/parsers/skill_source_parser.py +169 -0
  61. claude_mpm/cli/parsers/skills_parser.py +146 -0
  62. claude_mpm/cli/parsers/source_parser.py +138 -0
  63. claude_mpm/cli/startup.py +1280 -118
  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-config.md +21 -134
  68. claude_mpm/commands/mpm-doctor.md +16 -20
  69. claude_mpm/commands/mpm-help.md +13 -283
  70. claude_mpm/commands/mpm-init.md +88 -489
  71. claude_mpm/commands/mpm-monitor.md +23 -401
  72. claude_mpm/commands/mpm-organize.md +72 -247
  73. claude_mpm/commands/mpm-postmortem.md +21 -0
  74. claude_mpm/commands/mpm-session-resume.md +30 -0
  75. claude_mpm/commands/mpm-status.md +13 -68
  76. claude_mpm/commands/mpm-ticket-view.md +109 -0
  77. claude_mpm/commands/mpm-version.md +13 -106
  78. claude_mpm/commands/mpm.md +10 -0
  79. claude_mpm/config/agent_presets.py +488 -0
  80. claude_mpm/config/agent_sources.py +352 -0
  81. claude_mpm/config/skill_presets.py +392 -0
  82. claude_mpm/config/skill_sources.py +590 -0
  83. claude_mpm/constants.py +13 -0
  84. claude_mpm/core/claude_runner.py +5 -34
  85. claude_mpm/core/config.py +15 -1
  86. claude_mpm/core/constants.py +1 -1
  87. claude_mpm/core/framework/__init__.py +3 -16
  88. claude_mpm/core/framework/formatters/content_formatter.py +3 -13
  89. claude_mpm/core/framework/loaders/agent_loader.py +8 -5
  90. claude_mpm/core/framework/loaders/file_loader.py +54 -101
  91. claude_mpm/core/framework/loaders/instruction_loader.py +66 -5
  92. claude_mpm/core/framework_loader.py +4 -2
  93. claude_mpm/core/hook_error_memory.py +381 -0
  94. claude_mpm/core/hook_manager.py +41 -2
  95. claude_mpm/core/interactive_session.py +91 -10
  96. claude_mpm/core/logger.py +16 -1
  97. claude_mpm/core/oneshot_session.py +71 -8
  98. claude_mpm/core/optimized_startup.py +59 -0
  99. claude_mpm/core/output_style_manager.py +173 -43
  100. claude_mpm/core/protocols/__init__.py +23 -0
  101. claude_mpm/core/protocols/runner_protocol.py +103 -0
  102. claude_mpm/core/protocols/session_protocol.py +131 -0
  103. claude_mpm/core/shared/config_loader.py +1 -1
  104. claude_mpm/core/shared/singleton_manager.py +11 -4
  105. claude_mpm/core/socketio_pool.py +3 -3
  106. claude_mpm/core/system_context.py +38 -0
  107. claude_mpm/core/unified_agent_registry.py +134 -16
  108. claude_mpm/core/unified_config.py +22 -0
  109. claude_mpm/dashboard/static/svelte-build/_app/env.js +1 -0
  110. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/0.B_FtCwCQ.css +1 -0
  111. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/2.Cl_eSA4x.css +1 -0
  112. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BgChzWQ1.js +1 -0
  113. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CIXEwuWe.js +1 -0
  114. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CWc5urbQ.js +1 -0
  115. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DMkZpdF2.js +2 -0
  116. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DjhvlsAc.js +1 -0
  117. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/N4qtv3Hx.js +2 -0
  118. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/uj46x2Wr.js +1 -0
  119. claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/app.DTL5mJO-.js +2 -0
  120. claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/start.DzuEhzqh.js +1 -0
  121. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/0.CAGBuiOw.js +1 -0
  122. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/1.DFLC8jdE.js +1 -0
  123. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/2.DPvEihJJ.js +10 -0
  124. claude_mpm/dashboard/static/svelte-build/_app/version.json +1 -0
  125. claude_mpm/dashboard/static/svelte-build/favicon.svg +7 -0
  126. claude_mpm/dashboard/static/svelte-build/index.html +36 -0
  127. claude_mpm/experimental/cli_enhancements.py +1 -5
  128. claude_mpm/hooks/claude_hooks/__pycache__/__init__.cpython-311.pyc +0 -0
  129. claude_mpm/hooks/claude_hooks/__pycache__/correlation_manager.cpython-311.pyc +0 -0
  130. claude_mpm/hooks/claude_hooks/__pycache__/event_handlers.cpython-311.pyc +0 -0
  131. claude_mpm/hooks/claude_hooks/__pycache__/hook_handler.cpython-311.pyc +0 -0
  132. claude_mpm/hooks/claude_hooks/__pycache__/installer.cpython-311.pyc +0 -0
  133. claude_mpm/hooks/claude_hooks/__pycache__/memory_integration.cpython-311.pyc +0 -0
  134. claude_mpm/hooks/claude_hooks/__pycache__/response_tracking.cpython-311.pyc +0 -0
  135. claude_mpm/hooks/claude_hooks/__pycache__/tool_analysis.cpython-311.pyc +0 -0
  136. claude_mpm/hooks/claude_hooks/correlation_manager.py +60 -0
  137. claude_mpm/hooks/claude_hooks/event_handlers.py +214 -79
  138. claude_mpm/hooks/claude_hooks/hook_handler.py +155 -1
  139. claude_mpm/hooks/claude_hooks/installer.py +33 -10
  140. claude_mpm/hooks/claude_hooks/memory_integration.py +28 -0
  141. claude_mpm/hooks/claude_hooks/response_tracking.py +2 -3
  142. claude_mpm/hooks/claude_hooks/services/__pycache__/__init__.cpython-311.pyc +0 -0
  143. claude_mpm/hooks/claude_hooks/services/__pycache__/connection_manager.cpython-311.pyc +0 -0
  144. claude_mpm/hooks/claude_hooks/services/__pycache__/connection_manager_http.cpython-311.pyc +0 -0
  145. claude_mpm/hooks/claude_hooks/services/__pycache__/duplicate_detector.cpython-311.pyc +0 -0
  146. claude_mpm/hooks/claude_hooks/services/__pycache__/state_manager.cpython-311.pyc +0 -0
  147. claude_mpm/hooks/claude_hooks/services/__pycache__/subagent_processor.cpython-311.pyc +0 -0
  148. claude_mpm/hooks/claude_hooks/services/connection_manager.py +30 -6
  149. claude_mpm/hooks/failure_learning/__init__.py +2 -8
  150. claude_mpm/hooks/failure_learning/failure_detection_hook.py +1 -6
  151. claude_mpm/hooks/failure_learning/fix_detection_hook.py +1 -6
  152. claude_mpm/hooks/failure_learning/learning_extraction_hook.py +1 -6
  153. claude_mpm/hooks/kuzu_response_hook.py +1 -5
  154. claude_mpm/hooks/memory_integration_hook.py +46 -1
  155. claude_mpm/init.py +63 -19
  156. claude_mpm/models/agent_definition.py +7 -0
  157. claude_mpm/models/git_repository.py +198 -0
  158. claude_mpm/scripts/claude-hook-handler.sh +60 -20
  159. claude_mpm/scripts/launch_monitor.py +93 -13
  160. claude_mpm/scripts/start_activity_logging.py +3 -1
  161. claude_mpm/services/agents/agent_builder.py +48 -12
  162. claude_mpm/services/agents/agent_preset_service.py +238 -0
  163. claude_mpm/services/agents/agent_recommendation_service.py +278 -0
  164. claude_mpm/services/agents/agent_review_service.py +280 -0
  165. claude_mpm/services/agents/agent_selection_service.py +484 -0
  166. claude_mpm/services/agents/auto_deploy_index_parser.py +569 -0
  167. claude_mpm/services/agents/cache_git_manager.py +621 -0
  168. claude_mpm/services/agents/deployment/agent_deployment.py +148 -2
  169. claude_mpm/services/agents/deployment/agent_discovery_service.py +104 -73
  170. claude_mpm/services/agents/deployment/agent_format_converter.py +1 -1
  171. claude_mpm/services/agents/deployment/agent_lifecycle_manager.py +1 -5
  172. claude_mpm/services/agents/deployment/agent_metrics_collector.py +3 -3
  173. claude_mpm/services/agents/deployment/agent_restore_handler.py +1 -4
  174. claude_mpm/services/agents/deployment/agent_template_builder.py +238 -15
  175. claude_mpm/services/agents/deployment/agents_directory_resolver.py +101 -15
  176. claude_mpm/services/agents/deployment/async_agent_deployment.py +2 -1
  177. claude_mpm/services/agents/deployment/facade/deployment_facade.py +3 -3
  178. claude_mpm/services/agents/deployment/multi_source_deployment_service.py +422 -31
  179. claude_mpm/services/agents/deployment/pipeline/pipeline_executor.py +2 -2
  180. claude_mpm/services/agents/deployment/refactored_agent_deployment_service.py +1 -4
  181. claude_mpm/services/agents/deployment/remote_agent_discovery_service.py +841 -0
  182. claude_mpm/services/agents/deployment/single_agent_deployer.py +2 -2
  183. claude_mpm/services/agents/deployment/system_instructions_deployer.py +168 -46
  184. claude_mpm/services/agents/deployment/validation/deployment_validator.py +2 -2
  185. claude_mpm/services/agents/git_source_manager.py +663 -0
  186. claude_mpm/services/agents/loading/base_agent_manager.py +1 -13
  187. claude_mpm/services/agents/loading/framework_agent_loader.py +9 -12
  188. claude_mpm/services/agents/local_template_manager.py +50 -10
  189. claude_mpm/services/agents/recommender.py +5 -3
  190. claude_mpm/services/agents/single_tier_deployment_service.py +696 -0
  191. claude_mpm/services/agents/sources/__init__.py +13 -0
  192. claude_mpm/services/agents/sources/agent_sync_state.py +516 -0
  193. claude_mpm/services/agents/sources/git_source_sync_service.py +1094 -0
  194. claude_mpm/services/agents/startup_sync.py +259 -0
  195. claude_mpm/services/agents/toolchain_detector.py +478 -0
  196. claude_mpm/services/analysis/__init__.py +35 -0
  197. claude_mpm/services/analysis/clone_detector.py +1030 -0
  198. claude_mpm/services/analysis/postmortem_reporter.py +474 -0
  199. claude_mpm/services/analysis/postmortem_service.py +765 -0
  200. claude_mpm/services/cli/session_pause_manager.py +1 -1
  201. claude_mpm/services/command_deployment_service.py +271 -6
  202. claude_mpm/services/core/base.py +7 -2
  203. claude_mpm/services/core/interfaces/__init__.py +1 -3
  204. claude_mpm/services/core/interfaces/health.py +1 -4
  205. claude_mpm/services/core/models/__init__.py +2 -11
  206. claude_mpm/services/diagnostics/checks/__init__.py +4 -0
  207. claude_mpm/services/diagnostics/checks/agent_check.py +2 -4
  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/event_bus/config.py +3 -1
  217. claude_mpm/services/event_bus/direct_relay.py +3 -3
  218. claude_mpm/services/events/consumers/logging.py +1 -2
  219. claude_mpm/services/git/__init__.py +21 -0
  220. claude_mpm/services/git/git_operations_service.py +579 -0
  221. claude_mpm/services/github/__init__.py +21 -0
  222. claude_mpm/services/github/github_cli_service.py +397 -0
  223. claude_mpm/services/infrastructure/monitoring/__init__.py +1 -5
  224. claude_mpm/services/infrastructure/monitoring/aggregator.py +1 -6
  225. claude_mpm/services/instructions/__init__.py +9 -0
  226. claude_mpm/services/instructions/instruction_cache_service.py +374 -0
  227. claude_mpm/services/local_ops/__init__.py +3 -13
  228. claude_mpm/services/local_ops/health_checks/__init__.py +1 -3
  229. claude_mpm/services/local_ops/health_manager.py +1 -4
  230. claude_mpm/services/local_ops/resource_monitor.py +1 -1
  231. claude_mpm/services/mcp_config_manager.py +75 -145
  232. claude_mpm/services/mcp_service_verifier.py +6 -3
  233. claude_mpm/services/model/model_router.py +1 -2
  234. claude_mpm/services/monitor/daemon.py +38 -11
  235. claude_mpm/services/monitor/daemon_manager.py +134 -21
  236. claude_mpm/services/monitor/management/lifecycle.py +8 -1
  237. claude_mpm/services/monitor/server.py +700 -24
  238. claude_mpm/services/pm_skills_deployer.py +676 -0
  239. claude_mpm/services/port_manager.py +1 -1
  240. claude_mpm/services/pr/__init__.py +14 -0
  241. claude_mpm/services/pr/pr_template_service.py +329 -0
  242. claude_mpm/services/profile_manager.py +331 -0
  243. claude_mpm/services/project/documentation_manager.py +2 -1
  244. claude_mpm/services/project/project_organizer.py +4 -0
  245. claude_mpm/services/project/toolchain_analyzer.py +3 -1
  246. claude_mpm/services/runner_configuration_service.py +16 -3
  247. claude_mpm/services/self_upgrade_service.py +120 -12
  248. claude_mpm/services/session_management_service.py +16 -4
  249. claude_mpm/services/skills/__init__.py +21 -0
  250. claude_mpm/services/skills/git_skill_source_manager.py +1297 -0
  251. claude_mpm/services/skills/selective_skill_deployer.py +704 -0
  252. claude_mpm/services/skills/skill_discovery_service.py +568 -0
  253. claude_mpm/services/skills/skill_to_agent_mapper.py +406 -0
  254. claude_mpm/services/skills_config.py +547 -0
  255. claude_mpm/services/skills_deployer.py +1072 -0
  256. claude_mpm/services/socketio/dashboard_server.py +1 -0
  257. claude_mpm/services/socketio/event_normalizer.py +51 -6
  258. claude_mpm/services/socketio/handlers/connection.py +1 -1
  259. claude_mpm/services/socketio/handlers/git.py +1 -1
  260. claude_mpm/services/socketio/server/core.py +387 -112
  261. claude_mpm/services/socketio/server/main.py +1 -3
  262. claude_mpm/services/system_instructions_service.py +1 -3
  263. claude_mpm/services/unified/analyzer_strategies/performance_analyzer.py +0 -3
  264. claude_mpm/services/unified/analyzer_strategies/security_analyzer.py +0 -1
  265. claude_mpm/services/unified/deployment_strategies/vercel.py +1 -5
  266. claude_mpm/services/unified/unified_deployment.py +1 -5
  267. claude_mpm/services/version_control/conflict_resolution.py +6 -4
  268. claude_mpm/services/version_control/git_operations.py +103 -0
  269. claude_mpm/services/visualization/__init__.py +1 -5
  270. claude_mpm/services/visualization/mermaid_generator.py +2 -3
  271. claude_mpm/skills/bundled/testing/webapp-testing/scripts/with_server.py +2 -2
  272. claude_mpm/skills/skill_manager.py +92 -3
  273. claude_mpm/skills/skills_registry.py +0 -1
  274. claude_mpm/templates/questions/__init__.py +38 -0
  275. claude_mpm/templates/questions/base.py +193 -0
  276. claude_mpm/templates/questions/pr_strategy.py +311 -0
  277. claude_mpm/templates/questions/project_init.py +385 -0
  278. claude_mpm/templates/questions/ticket_mgmt.py +394 -0
  279. claude_mpm/tools/__main__.py +8 -8
  280. claude_mpm/utils/agent_dependency_loader.py +91 -12
  281. claude_mpm/utils/agent_filters.py +261 -0
  282. claude_mpm/utils/dependency_cache.py +3 -1
  283. claude_mpm/utils/gitignore.py +244 -0
  284. claude_mpm/utils/migration.py +372 -0
  285. claude_mpm/utils/progress.py +387 -0
  286. claude_mpm/utils/robust_installer.py +49 -7
  287. claude_mpm/utils/structured_questions.py +619 -0
  288. {claude_mpm-4.24.0.dist-info → claude_mpm-5.4.41.dist-info}/METADATA +445 -122
  289. {claude_mpm-4.24.0.dist-info → claude_mpm-5.4.41.dist-info}/RECORD +298 -503
  290. claude_mpm-5.4.41.dist-info/entry_points.txt +5 -0
  291. claude_mpm-5.4.41.dist-info/licenses/LICENSE +94 -0
  292. claude_mpm-5.4.41.dist-info/licenses/LICENSE-FAQ.md +153 -0
  293. claude_mpm/agents/BASE_AGENT_TEMPLATE.md +0 -292
  294. claude_mpm/agents/BASE_DOCUMENTATION.md +0 -53
  295. claude_mpm/agents/BASE_OPS.md +0 -219
  296. claude_mpm/agents/BASE_PM.md +0 -468
  297. claude_mpm/agents/BASE_PROMPT_ENGINEER.md +0 -787
  298. claude_mpm/agents/BASE_QA.md +0 -167
  299. claude_mpm/agents/BASE_RESEARCH.md +0 -53
  300. claude_mpm/agents/base_agent_loader.py +0 -626
  301. claude_mpm/agents/templates/.claude-mpm/memories/README.md +0 -17
  302. claude_mpm/agents/templates/.claude-mpm/memories/engineer_memories.md +0 -3
  303. claude_mpm/agents/templates/agent-manager.json +0 -273
  304. claude_mpm/agents/templates/agentic-coder-optimizer.json +0 -248
  305. claude_mpm/agents/templates/api_qa.json +0 -183
  306. claude_mpm/agents/templates/circuit_breakers.md +0 -638
  307. claude_mpm/agents/templates/clerk-ops.json +0 -235
  308. claude_mpm/agents/templates/code_analyzer.json +0 -101
  309. claude_mpm/agents/templates/content-agent.json +0 -358
  310. claude_mpm/agents/templates/dart_engineer.json +0 -307
  311. claude_mpm/agents/templates/data_engineer.json +0 -225
  312. claude_mpm/agents/templates/documentation.json +0 -238
  313. claude_mpm/agents/templates/engineer.json +0 -210
  314. claude_mpm/agents/templates/gcp_ops_agent.json +0 -253
  315. claude_mpm/agents/templates/golang_engineer.json +0 -270
  316. claude_mpm/agents/templates/imagemagick.json +0 -264
  317. claude_mpm/agents/templates/java_engineer.json +0 -346
  318. claude_mpm/agents/templates/javascript_engineer_agent.json +0 -380
  319. claude_mpm/agents/templates/local_ops_agent.json +0 -1840
  320. claude_mpm/agents/templates/logs/prompts/agent_engineer_20250826_014258_728.md +0 -39
  321. claude_mpm/agents/templates/logs/prompts/agent_engineer_20250901_010124_142.md +0 -400
  322. claude_mpm/agents/templates/memory_manager.json +0 -158
  323. claude_mpm/agents/templates/nextjs_engineer.json +0 -285
  324. claude_mpm/agents/templates/ops.json +0 -185
  325. claude_mpm/agents/templates/php-engineer.json +0 -287
  326. claude_mpm/agents/templates/product_owner.json +0 -338
  327. claude_mpm/agents/templates/project_organizer.json +0 -144
  328. claude_mpm/agents/templates/prompt-engineer.json +0 -737
  329. claude_mpm/agents/templates/python_engineer.json +0 -387
  330. claude_mpm/agents/templates/qa.json +0 -243
  331. claude_mpm/agents/templates/react_engineer.json +0 -239
  332. claude_mpm/agents/templates/refactoring_engineer.json +0 -276
  333. claude_mpm/agents/templates/research.json +0 -188
  334. claude_mpm/agents/templates/ruby-engineer.json +0 -280
  335. claude_mpm/agents/templates/rust_engineer.json +0 -275
  336. claude_mpm/agents/templates/security.json +0 -202
  337. claude_mpm/agents/templates/svelte-engineer.json +0 -225
  338. claude_mpm/agents/templates/tauri_engineer.json +0 -274
  339. claude_mpm/agents/templates/ticketing.json +0 -178
  340. claude_mpm/agents/templates/typescript_engineer.json +0 -285
  341. claude_mpm/agents/templates/vercel_ops_agent.json +0 -412
  342. claude_mpm/agents/templates/version_control.json +0 -159
  343. claude_mpm/agents/templates/web_qa.json +0 -400
  344. claude_mpm/agents/templates/web_ui.json +0 -189
  345. claude_mpm/cli/commands/agents_detect.py +0 -380
  346. claude_mpm/cli/commands/agents_recommend.py +0 -309
  347. claude_mpm/cli/ticket_cli.py +0 -35
  348. claude_mpm/commands/mpm-agents-detect.md +0 -168
  349. claude_mpm/commands/mpm-agents-recommend.md +0 -214
  350. claude_mpm/commands/mpm-agents.md +0 -122
  351. claude_mpm/commands/mpm-auto-configure.md +0 -269
  352. claude_mpm/commands/mpm-resume.md +0 -372
  353. claude_mpm/commands/mpm-tickets.md +0 -151
  354. claude_mpm/dashboard/.claude-mpm/socketio-instances.json +0 -1
  355. claude_mpm/dashboard/analysis_runner.py +0 -455
  356. claude_mpm/dashboard/index.html +0 -13
  357. claude_mpm/dashboard/open_dashboard.py +0 -66
  358. claude_mpm/dashboard/react/components/DataInspector/DataInspector.module.css +0 -188
  359. claude_mpm/dashboard/react/components/EventViewer/EventViewer.module.css +0 -156
  360. claude_mpm/dashboard/react/components/shared/ConnectionStatus.module.css +0 -38
  361. claude_mpm/dashboard/react/components/shared/FilterBar.module.css +0 -92
  362. claude_mpm/dashboard/static/archive/activity_dashboard_fixed.html +0 -248
  363. claude_mpm/dashboard/static/archive/activity_dashboard_test.html +0 -61
  364. claude_mpm/dashboard/static/archive/test_activity_connection.html +0 -179
  365. claude_mpm/dashboard/static/archive/test_claude_tree_tab.html +0 -68
  366. claude_mpm/dashboard/static/archive/test_dashboard.html +0 -409
  367. claude_mpm/dashboard/static/archive/test_dashboard_fixed.html +0 -519
  368. claude_mpm/dashboard/static/archive/test_dashboard_verification.html +0 -181
  369. claude_mpm/dashboard/static/archive/test_file_data.html +0 -315
  370. claude_mpm/dashboard/static/archive/test_file_tree_empty_state.html +0 -243
  371. claude_mpm/dashboard/static/archive/test_file_tree_fix.html +0 -234
  372. claude_mpm/dashboard/static/archive/test_file_tree_rename.html +0 -117
  373. claude_mpm/dashboard/static/archive/test_file_tree_tab.html +0 -115
  374. claude_mpm/dashboard/static/archive/test_file_viewer.html +0 -224
  375. claude_mpm/dashboard/static/archive/test_final_activity.html +0 -220
  376. claude_mpm/dashboard/static/archive/test_tab_fix.html +0 -139
  377. claude_mpm/dashboard/static/built/assets/events.DjpNxWNo.css +0 -1
  378. claude_mpm/dashboard/static/built/components/activity-tree.js +0 -2
  379. claude_mpm/dashboard/static/built/components/agent-hierarchy.js +0 -777
  380. claude_mpm/dashboard/static/built/components/agent-inference.js +0 -2
  381. claude_mpm/dashboard/static/built/components/build-tracker.js +0 -333
  382. claude_mpm/dashboard/static/built/components/code-simple.js +0 -857
  383. claude_mpm/dashboard/static/built/components/code-tree/tree-breadcrumb.js +0 -353
  384. claude_mpm/dashboard/static/built/components/code-tree/tree-constants.js +0 -235
  385. claude_mpm/dashboard/static/built/components/code-tree/tree-search.js +0 -409
  386. claude_mpm/dashboard/static/built/components/code-tree/tree-utils.js +0 -435
  387. claude_mpm/dashboard/static/built/components/code-tree.js +0 -2
  388. claude_mpm/dashboard/static/built/components/code-viewer.js +0 -2
  389. claude_mpm/dashboard/static/built/components/connection-debug.js +0 -654
  390. claude_mpm/dashboard/static/built/components/diff-viewer.js +0 -891
  391. claude_mpm/dashboard/static/built/components/event-processor.js +0 -2
  392. claude_mpm/dashboard/static/built/components/event-viewer.js +0 -2
  393. claude_mpm/dashboard/static/built/components/export-manager.js +0 -2
  394. claude_mpm/dashboard/static/built/components/file-change-tracker.js +0 -443
  395. claude_mpm/dashboard/static/built/components/file-change-viewer.js +0 -690
  396. claude_mpm/dashboard/static/built/components/file-tool-tracker.js +0 -2
  397. claude_mpm/dashboard/static/built/components/file-viewer.js +0 -2
  398. claude_mpm/dashboard/static/built/components/hud-library-loader.js +0 -2
  399. claude_mpm/dashboard/static/built/components/hud-manager.js +0 -2
  400. claude_mpm/dashboard/static/built/components/hud-visualizer.js +0 -2
  401. claude_mpm/dashboard/static/built/components/module-viewer.js +0 -2
  402. claude_mpm/dashboard/static/built/components/nav-bar.js +0 -145
  403. claude_mpm/dashboard/static/built/components/page-structure.js +0 -429
  404. claude_mpm/dashboard/static/built/components/session-manager.js +0 -2
  405. claude_mpm/dashboard/static/built/components/socket-manager.js +0 -2
  406. claude_mpm/dashboard/static/built/components/ui-state-manager.js +0 -2
  407. claude_mpm/dashboard/static/built/components/unified-data-viewer.js +0 -2
  408. claude_mpm/dashboard/static/built/components/working-directory.js +0 -2
  409. claude_mpm/dashboard/static/built/connection-manager.js +0 -536
  410. claude_mpm/dashboard/static/built/dashboard.js +0 -2
  411. claude_mpm/dashboard/static/built/extension-error-handler.js +0 -164
  412. claude_mpm/dashboard/static/built/react/events.js +0 -30
  413. claude_mpm/dashboard/static/built/shared/dom-helpers.js +0 -396
  414. claude_mpm/dashboard/static/built/shared/event-bus.js +0 -330
  415. claude_mpm/dashboard/static/built/shared/event-filter-service.js +0 -540
  416. claude_mpm/dashboard/static/built/shared/logger.js +0 -385
  417. claude_mpm/dashboard/static/built/shared/page-structure.js +0 -249
  418. claude_mpm/dashboard/static/built/shared/tooltip-service.js +0 -253
  419. claude_mpm/dashboard/static/built/socket-client.js +0 -2
  420. claude_mpm/dashboard/static/built/tab-isolation-fix.js +0 -185
  421. claude_mpm/dashboard/static/css/activity.css +0 -1958
  422. claude_mpm/dashboard/static/css/connection-status.css +0 -370
  423. claude_mpm/dashboard/static/css/dashboard.css +0 -4701
  424. claude_mpm/dashboard/static/dist/assets/events.DjpNxWNo.css +0 -1
  425. claude_mpm/dashboard/static/dist/components/activity-tree.js +0 -2
  426. claude_mpm/dashboard/static/dist/components/agent-inference.js +0 -2
  427. claude_mpm/dashboard/static/dist/components/code-tree.js +0 -2
  428. claude_mpm/dashboard/static/dist/components/code-viewer.js +0 -2
  429. claude_mpm/dashboard/static/dist/components/event-processor.js +0 -2
  430. claude_mpm/dashboard/static/dist/components/event-viewer.js +0 -2
  431. claude_mpm/dashboard/static/dist/components/export-manager.js +0 -2
  432. claude_mpm/dashboard/static/dist/components/file-tool-tracker.js +0 -2
  433. claude_mpm/dashboard/static/dist/components/file-viewer.js +0 -2
  434. claude_mpm/dashboard/static/dist/components/hud-library-loader.js +0 -2
  435. claude_mpm/dashboard/static/dist/components/hud-manager.js +0 -2
  436. claude_mpm/dashboard/static/dist/components/hud-visualizer.js +0 -2
  437. claude_mpm/dashboard/static/dist/components/module-viewer.js +0 -2
  438. claude_mpm/dashboard/static/dist/components/session-manager.js +0 -2
  439. claude_mpm/dashboard/static/dist/components/socket-manager.js +0 -2
  440. claude_mpm/dashboard/static/dist/components/ui-state-manager.js +0 -2
  441. claude_mpm/dashboard/static/dist/components/unified-data-viewer.js +0 -2
  442. claude_mpm/dashboard/static/dist/components/working-directory.js +0 -2
  443. claude_mpm/dashboard/static/dist/dashboard.js +0 -2
  444. claude_mpm/dashboard/static/dist/react/events.js +0 -30
  445. claude_mpm/dashboard/static/dist/socket-client.js +0 -2
  446. claude_mpm/dashboard/static/events.html +0 -607
  447. claude_mpm/dashboard/static/index.html +0 -635
  448. claude_mpm/dashboard/static/js/components/activity-tree.js +0 -1871
  449. claude_mpm/dashboard/static/js/components/agent-hierarchy.js +0 -777
  450. claude_mpm/dashboard/static/js/components/agent-inference.js +0 -956
  451. claude_mpm/dashboard/static/js/components/build-tracker.js +0 -333
  452. claude_mpm/dashboard/static/js/components/code-simple.js +0 -857
  453. claude_mpm/dashboard/static/js/components/connection-debug.js +0 -654
  454. claude_mpm/dashboard/static/js/components/diff-viewer.js +0 -891
  455. claude_mpm/dashboard/static/js/components/event-processor.js +0 -542
  456. claude_mpm/dashboard/static/js/components/event-viewer.js +0 -1155
  457. claude_mpm/dashboard/static/js/components/export-manager.js +0 -368
  458. claude_mpm/dashboard/static/js/components/file-change-tracker.js +0 -443
  459. claude_mpm/dashboard/static/js/components/file-change-viewer.js +0 -690
  460. claude_mpm/dashboard/static/js/components/file-tool-tracker.js +0 -724
  461. claude_mpm/dashboard/static/js/components/file-viewer.js +0 -580
  462. claude_mpm/dashboard/static/js/components/hud-library-loader.js +0 -211
  463. claude_mpm/dashboard/static/js/components/hud-manager.js +0 -671
  464. claude_mpm/dashboard/static/js/components/hud-visualizer.js +0 -1718
  465. claude_mpm/dashboard/static/js/components/module-viewer.js +0 -2764
  466. claude_mpm/dashboard/static/js/components/session-manager.js +0 -579
  467. claude_mpm/dashboard/static/js/components/socket-manager.js +0 -368
  468. claude_mpm/dashboard/static/js/components/ui-state-manager.js +0 -749
  469. claude_mpm/dashboard/static/js/components/unified-data-viewer.js +0 -1824
  470. claude_mpm/dashboard/static/js/components/working-directory.js +0 -920
  471. claude_mpm/dashboard/static/js/connection-manager.js +0 -536
  472. claude_mpm/dashboard/static/js/dashboard.js +0 -1896
  473. claude_mpm/dashboard/static/js/extension-error-handler.js +0 -164
  474. claude_mpm/dashboard/static/js/shared/dom-helpers.js +0 -396
  475. claude_mpm/dashboard/static/js/shared/event-bus.js +0 -330
  476. claude_mpm/dashboard/static/js/shared/logger.js +0 -385
  477. claude_mpm/dashboard/static/js/shared/tooltip-service.js +0 -253
  478. claude_mpm/dashboard/static/js/socket-client.js +0 -1457
  479. claude_mpm/dashboard/static/js/stores/dashboard-store.js +0 -562
  480. claude_mpm/dashboard/static/js/tab-isolation-fix.js +0 -185
  481. claude_mpm/dashboard/static/legacy/activity.html +0 -736
  482. claude_mpm/dashboard/static/legacy/agents.html +0 -786
  483. claude_mpm/dashboard/static/legacy/files.html +0 -747
  484. claude_mpm/dashboard/static/legacy/tools.html +0 -831
  485. claude_mpm/dashboard/static/monitors.html +0 -431
  486. claude_mpm/dashboard/static/production/events.html +0 -659
  487. claude_mpm/dashboard/static/production/main.html +0 -698
  488. claude_mpm/dashboard/static/production/monitors.html +0 -483
  489. claude_mpm/dashboard/static/socket.io.min.js +0 -7
  490. claude_mpm/dashboard/static/socket.io.v4.8.1.backup.js +0 -7
  491. claude_mpm/dashboard/static/test-archive/dashboard.html +0 -635
  492. claude_mpm/dashboard/static/test-archive/debug-events.html +0 -147
  493. claude_mpm/dashboard/static/test-archive/test-navigation.html +0 -256
  494. claude_mpm/dashboard/static/test-archive/test-react-exports.html +0 -180
  495. claude_mpm/dashboard/static/test-archive/test_debug.html +0 -25
  496. claude_mpm/dashboard/templates/code_simple.html +0 -153
  497. claude_mpm/dashboard/templates/index.html +0 -606
  498. claude_mpm/dashboard/test_dashboard.html +0 -372
  499. claude_mpm/scripts/mcp_server.py +0 -75
  500. claude_mpm/scripts/mcp_wrapper.py +0 -39
  501. claude_mpm/services/mcp_gateway/__init__.py +0 -159
  502. claude_mpm/services/mcp_gateway/auto_configure.py +0 -369
  503. claude_mpm/services/mcp_gateway/config/__init__.py +0 -17
  504. claude_mpm/services/mcp_gateway/config/config_loader.py +0 -296
  505. claude_mpm/services/mcp_gateway/config/config_schema.py +0 -243
  506. claude_mpm/services/mcp_gateway/config/configuration.py +0 -429
  507. claude_mpm/services/mcp_gateway/core/__init__.py +0 -43
  508. claude_mpm/services/mcp_gateway/core/base.py +0 -312
  509. claude_mpm/services/mcp_gateway/core/exceptions.py +0 -253
  510. claude_mpm/services/mcp_gateway/core/interfaces.py +0 -443
  511. claude_mpm/services/mcp_gateway/core/process_pool.py +0 -971
  512. claude_mpm/services/mcp_gateway/core/singleton_manager.py +0 -315
  513. claude_mpm/services/mcp_gateway/core/startup_verification.py +0 -316
  514. claude_mpm/services/mcp_gateway/main.py +0 -589
  515. claude_mpm/services/mcp_gateway/registry/__init__.py +0 -12
  516. claude_mpm/services/mcp_gateway/registry/service_registry.py +0 -412
  517. claude_mpm/services/mcp_gateway/registry/tool_registry.py +0 -489
  518. claude_mpm/services/mcp_gateway/server/__init__.py +0 -15
  519. claude_mpm/services/mcp_gateway/server/mcp_gateway.py +0 -419
  520. claude_mpm/services/mcp_gateway/server/stdio_handler.py +0 -372
  521. claude_mpm/services/mcp_gateway/server/stdio_server.py +0 -714
  522. claude_mpm/services/mcp_gateway/tools/__init__.py +0 -36
  523. claude_mpm/services/mcp_gateway/tools/base_adapter.py +0 -485
  524. claude_mpm/services/mcp_gateway/tools/document_summarizer.py +0 -789
  525. claude_mpm/services/mcp_gateway/tools/external_mcp_services.py +0 -654
  526. claude_mpm/services/mcp_gateway/tools/health_check_tool.py +0 -456
  527. claude_mpm/services/mcp_gateway/tools/hello_world.py +0 -551
  528. claude_mpm/services/mcp_gateway/tools/kuzu_memory_service.py +0 -551
  529. claude_mpm/services/mcp_gateway/utils/__init__.py +0 -14
  530. claude_mpm/services/mcp_gateway/utils/package_version_checker.py +0 -160
  531. claude_mpm/services/mcp_gateway/utils/update_preferences.py +0 -170
  532. claude_mpm/skills/bundled/collaboration/brainstorming/SKILL.md +0 -79
  533. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/SKILL.md +0 -178
  534. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/agent-prompts.md +0 -577
  535. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/coordination-patterns.md +0 -467
  536. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/examples.md +0 -537
  537. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/troubleshooting.md +0 -730
  538. claude_mpm/skills/bundled/collaboration/requesting-code-review/SKILL.md +0 -112
  539. claude_mpm/skills/bundled/collaboration/requesting-code-review/references/code-reviewer-template.md +0 -146
  540. claude_mpm/skills/bundled/collaboration/requesting-code-review/references/review-examples.md +0 -412
  541. claude_mpm/skills/bundled/collaboration/writing-plans/SKILL.md +0 -81
  542. claude_mpm/skills/bundled/collaboration/writing-plans/references/best-practices.md +0 -362
  543. claude_mpm/skills/bundled/collaboration/writing-plans/references/plan-structure-templates.md +0 -312
  544. claude_mpm/skills/bundled/debugging/root-cause-tracing/SKILL.md +0 -152
  545. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/advanced-techniques.md +0 -668
  546. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/examples.md +0 -587
  547. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/integration.md +0 -438
  548. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/tracing-techniques.md +0 -391
  549. claude_mpm/skills/bundled/debugging/systematic-debugging/CREATION-LOG.md +0 -119
  550. claude_mpm/skills/bundled/debugging/systematic-debugging/SKILL.md +0 -148
  551. claude_mpm/skills/bundled/debugging/systematic-debugging/references/anti-patterns.md +0 -483
  552. claude_mpm/skills/bundled/debugging/systematic-debugging/references/examples.md +0 -452
  553. claude_mpm/skills/bundled/debugging/systematic-debugging/references/troubleshooting.md +0 -449
  554. claude_mpm/skills/bundled/debugging/systematic-debugging/references/workflow.md +0 -411
  555. claude_mpm/skills/bundled/debugging/systematic-debugging/test-academic.md +0 -14
  556. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-1.md +0 -58
  557. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-2.md +0 -68
  558. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-3.md +0 -69
  559. claude_mpm/skills/bundled/debugging/verification-before-completion/SKILL.md +0 -131
  560. claude_mpm/skills/bundled/debugging/verification-before-completion/references/gate-function.md +0 -325
  561. claude_mpm/skills/bundled/debugging/verification-before-completion/references/integration-and-workflows.md +0 -490
  562. claude_mpm/skills/bundled/debugging/verification-before-completion/references/red-flags-and-failures.md +0 -425
  563. claude_mpm/skills/bundled/debugging/verification-before-completion/references/verification-patterns.md +0 -499
  564. claude_mpm/skills/bundled/main/artifacts-builder/SKILL.md +0 -86
  565. claude_mpm/skills/bundled/main/internal-comms/SKILL.md +0 -43
  566. claude_mpm/skills/bundled/main/internal-comms/examples/3p-updates.md +0 -47
  567. claude_mpm/skills/bundled/main/internal-comms/examples/company-newsletter.md +0 -65
  568. claude_mpm/skills/bundled/main/internal-comms/examples/faq-answers.md +0 -30
  569. claude_mpm/skills/bundled/main/internal-comms/examples/general-comms.md +0 -16
  570. claude_mpm/skills/bundled/main/mcp-builder/SKILL.md +0 -160
  571. claude_mpm/skills/bundled/main/mcp-builder/reference/design_principles.md +0 -412
  572. claude_mpm/skills/bundled/main/mcp-builder/reference/evaluation.md +0 -602
  573. claude_mpm/skills/bundled/main/mcp-builder/reference/mcp_best_practices.md +0 -915
  574. claude_mpm/skills/bundled/main/mcp-builder/reference/node_mcp_server.md +0 -916
  575. claude_mpm/skills/bundled/main/mcp-builder/reference/python_mcp_server.md +0 -752
  576. claude_mpm/skills/bundled/main/mcp-builder/reference/workflow.md +0 -1237
  577. claude_mpm/skills/bundled/main/skill-creator/SKILL.md +0 -189
  578. claude_mpm/skills/bundled/main/skill-creator/references/best-practices.md +0 -500
  579. claude_mpm/skills/bundled/main/skill-creator/references/creation-workflow.md +0 -464
  580. claude_mpm/skills/bundled/main/skill-creator/references/examples.md +0 -619
  581. claude_mpm/skills/bundled/main/skill-creator/references/progressive-disclosure.md +0 -437
  582. claude_mpm/skills/bundled/main/skill-creator/references/skill-structure.md +0 -231
  583. claude_mpm/skills/bundled/php/espocrm-development/SKILL.md +0 -170
  584. claude_mpm/skills/bundled/php/espocrm-development/references/architecture.md +0 -602
  585. claude_mpm/skills/bundled/php/espocrm-development/references/common-tasks.md +0 -821
  586. claude_mpm/skills/bundled/php/espocrm-development/references/development-workflow.md +0 -742
  587. claude_mpm/skills/bundled/php/espocrm-development/references/frontend-customization.md +0 -726
  588. claude_mpm/skills/bundled/php/espocrm-development/references/hooks-and-services.md +0 -764
  589. claude_mpm/skills/bundled/php/espocrm-development/references/testing-debugging.md +0 -831
  590. claude_mpm/skills/bundled/rust/desktop-applications/SKILL.md +0 -226
  591. claude_mpm/skills/bundled/rust/desktop-applications/references/architecture-patterns.md +0 -901
  592. claude_mpm/skills/bundled/rust/desktop-applications/references/native-gui-frameworks.md +0 -901
  593. claude_mpm/skills/bundled/rust/desktop-applications/references/platform-integration.md +0 -775
  594. claude_mpm/skills/bundled/rust/desktop-applications/references/state-management.md +0 -937
  595. claude_mpm/skills/bundled/rust/desktop-applications/references/tauri-framework.md +0 -770
  596. claude_mpm/skills/bundled/rust/desktop-applications/references/testing-deployment.md +0 -961
  597. claude_mpm/skills/bundled/testing/condition-based-waiting/SKILL.md +0 -119
  598. claude_mpm/skills/bundled/testing/condition-based-waiting/references/patterns-and-implementation.md +0 -253
  599. claude_mpm/skills/bundled/testing/test-driven-development/SKILL.md +0 -145
  600. claude_mpm/skills/bundled/testing/test-driven-development/references/anti-patterns.md +0 -543
  601. claude_mpm/skills/bundled/testing/test-driven-development/references/examples.md +0 -741
  602. claude_mpm/skills/bundled/testing/test-driven-development/references/integration.md +0 -470
  603. claude_mpm/skills/bundled/testing/test-driven-development/references/philosophy.md +0 -458
  604. claude_mpm/skills/bundled/testing/test-driven-development/references/workflow.md +0 -639
  605. claude_mpm/skills/bundled/testing/testing-anti-patterns/SKILL.md +0 -140
  606. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/completeness-anti-patterns.md +0 -572
  607. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/core-anti-patterns.md +0 -411
  608. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/detection-guide.md +0 -569
  609. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/tdd-connection.md +0 -695
  610. claude_mpm/skills/bundled/testing/webapp-testing/SKILL.md +0 -184
  611. claude_mpm/skills/bundled/testing/webapp-testing/decision-tree.md +0 -459
  612. claude_mpm/skills/bundled/testing/webapp-testing/playwright-patterns.md +0 -479
  613. claude_mpm/skills/bundled/testing/webapp-testing/reconnaissance-pattern.md +0 -687
  614. claude_mpm/skills/bundled/testing/webapp-testing/server-management.md +0 -758
  615. claude_mpm/skills/bundled/testing/webapp-testing/troubleshooting.md +0 -868
  616. claude_mpm-4.24.0.dist-info/entry_points.txt +0 -10
  617. claude_mpm-4.24.0.dist-info/licenses/LICENSE +0 -21
  618. /claude_mpm/agents/templates/{git_file_tracking.md → git-file-tracking.md} +0 -0
  619. /claude_mpm/agents/templates/{pm_examples.md → pm-examples.md} +0 -0
  620. /claude_mpm/agents/templates/{response_format.md → response-format.md} +0 -0
  621. /claude_mpm/agents/templates/{validation_templates.md → validation-templates.md} +0 -0
  622. {claude_mpm-4.24.0.dist-info → claude_mpm-5.4.41.dist-info}/WHEEL +0 -0
  623. {claude_mpm-4.24.0.dist-info → claude_mpm-5.4.41.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,1072 @@
1
+ """Skills Deployer Service - Deploy Claude Code skills from GitHub.
2
+
3
+ WHY: Claude Code loads skills at STARTUP ONLY from ~/.claude/skills/ directory.
4
+ This service manages downloading skills from external GitHub repositories and
5
+ deploying them to Claude Code's skills directory with automatic restart warnings.
6
+
7
+ DESIGN DECISIONS:
8
+ - Downloads from https://github.com/bobmatnyc/claude-mpm-skills by default
9
+ - Deploys to ~/.claude/skills/ (Claude Code's directory), NOT project directory
10
+ - Integrates with ToolchainAnalyzer for automatic language detection
11
+ - Handles Claude Code restart requirement (skills only load at startup)
12
+ - Provides filtering by toolchain and categories
13
+ - Graceful error handling with actionable messages
14
+
15
+ ARCHITECTURE:
16
+ 1. GitHub Download: Fetch ZIP archive from repository
17
+ 2. Manifest Parsing: Read skill metadata from manifest.json
18
+ 3. Filtering: Apply toolchain and category filters
19
+ 4. Deployment: Copy skills to ~/.claude/skills/
20
+ 5. Restart Detection: Warn if Claude Code is running
21
+ 6. Cleanup: Remove temporary files
22
+
23
+ References:
24
+ - Research: docs/research/skills-research.md
25
+ - GitHub Repo: https://github.com/bobmatnyc/claude-mpm-skills
26
+ """
27
+
28
+ import json
29
+ import platform
30
+ import shutil
31
+ import subprocess
32
+ from pathlib import Path
33
+ from typing import Any, Dict, List, Optional
34
+
35
+ from claude_mpm.core.mixins import LoggerMixin
36
+ from claude_mpm.services.skills_config import SkillsConfig
37
+
38
+
39
+ class SkillsDeployerService(LoggerMixin):
40
+ """Deploy Claude Code skills from external GitHub repositories.
41
+
42
+ This service:
43
+ - Downloads skills from GitHub repositories
44
+ - Deploys to ~/.claude/skills/ directory
45
+ - Filters by toolchain (python, javascript, rust, etc.)
46
+ - Filters by categories (testing, debugging, web, etc.)
47
+ - Detects Claude Code process and warns about restart requirement
48
+ - Provides deployment summaries and error handling
49
+
50
+ Example:
51
+ >>> deployer = SkillsDeployerService()
52
+ >>> result = deployer.deploy_skills(toolchain=['python'], categories=['testing'])
53
+ >>> print(f"Deployed {result['deployed_count']} skills")
54
+ >>> print(f"Restart Claude Code: {result['restart_required']}")
55
+ """
56
+
57
+ DEFAULT_REPO_URL = "https://github.com/bobmatnyc/claude-mpm-skills"
58
+ CLAUDE_SKILLS_DIR = Path.home() / ".claude" / "skills"
59
+
60
+ def __init__(
61
+ self,
62
+ repo_url: Optional[str] = None,
63
+ toolchain_analyzer: Optional[any] = None,
64
+ ):
65
+ """Initialize Skills Deployer Service.
66
+
67
+ Args:
68
+ repo_url: GitHub repository URL (default: bobmatnyc/claude-mpm-skills)
69
+ toolchain_analyzer: Optional ToolchainAnalyzer for auto-detection
70
+ """
71
+ super().__init__()
72
+ self.repo_url = repo_url or self.DEFAULT_REPO_URL
73
+ self.toolchain_analyzer = toolchain_analyzer
74
+ self.skills_config = SkillsConfig()
75
+
76
+ # Ensure Claude skills directory exists
77
+ self.CLAUDE_SKILLS_DIR.mkdir(parents=True, exist_ok=True)
78
+
79
+ def deploy_skills(
80
+ self,
81
+ collection: Optional[str] = None,
82
+ toolchain: Optional[List[str]] = None,
83
+ categories: Optional[List[str]] = None,
84
+ force: bool = False,
85
+ selective: bool = True,
86
+ project_root: Optional[Path] = None,
87
+ ) -> Dict:
88
+ """Deploy skills from GitHub repository.
89
+
90
+ This is the main entry point for skill deployment. It:
91
+ 1. Downloads skills from GitHub collection
92
+ 2. Parses manifest for metadata
93
+ 3. Filters by toolchain and categories
94
+ 4. (If selective=True) Filters to only agent-referenced skills
95
+ 5. Deploys to ~/.claude/skills/
96
+ 6. Warns about Claude Code restart
97
+
98
+ Args:
99
+ collection: Collection name to deploy from (default: uses default collection)
100
+ toolchain: Filter by toolchain (e.g., ['python', 'javascript'])
101
+ categories: Filter by categories (e.g., ['testing', 'debugging'])
102
+ force: Overwrite existing skills
103
+ selective: If True, only deploy skills referenced by agents (default)
104
+ project_root: Project root directory (for finding agents, auto-detected if None)
105
+
106
+ Returns:
107
+ Dict containing:
108
+ - deployed_count: Number of skills deployed
109
+ - skipped_count: Number of skills skipped
110
+ - errors: List of error messages
111
+ - deployed_skills: List of deployed skill names
112
+ - restart_required: True if Claude Code needs restart
113
+ - restart_instructions: Message about restarting
114
+ - collection: Collection name used for deployment
115
+ - selective_mode: True if selective deployment was used
116
+ - total_available: Total skills available before filtering
117
+
118
+ Example:
119
+ >>> result = deployer.deploy_skills(collection="obra-superpowers")
120
+ >>> result = deployer.deploy_skills(toolchain=['python']) # Uses default
121
+ >>> # Deploy all skills (not just agent-referenced)
122
+ >>> result = deployer.deploy_skills(selective=False)
123
+ >>> if result['restart_required']:
124
+ >>> print(result['restart_instructions'])
125
+ """
126
+ # Determine which collection to use
127
+ collection_name = collection or self.skills_config.get_default_collection()
128
+
129
+ self.logger.info(f"Deploying skills from collection '{collection_name}'")
130
+
131
+ # Step 1: Download skills from GitHub collection
132
+ try:
133
+ skills_data = self._download_from_github(collection_name)
134
+ except Exception as e:
135
+ self.logger.error(f"Failed to download skills: {e}")
136
+ return {
137
+ "deployed_count": 0,
138
+ "skipped_count": 0,
139
+ "errors": [f"Download failed: {e}"],
140
+ "deployed_skills": [],
141
+ "restart_required": False,
142
+ "restart_instructions": "",
143
+ "collection": collection_name,
144
+ }
145
+
146
+ # Step 2: Parse manifest and flatten skills
147
+ manifest = skills_data.get("manifest", {})
148
+ try:
149
+ skills = self._flatten_manifest_skills(manifest)
150
+ except ValueError as e:
151
+ self.logger.error(f"Invalid manifest structure: {e}")
152
+ return {
153
+ "deployed_count": 0,
154
+ "skipped_count": 0,
155
+ "errors": [f"Invalid manifest: {e}"],
156
+ "deployed_skills": [],
157
+ "restart_required": False,
158
+ "restart_instructions": "",
159
+ "collection": collection_name,
160
+ }
161
+
162
+ self.logger.info(f"Found {len(skills)} skills in repository")
163
+
164
+ # Step 3: Filter skills by toolchain and categories
165
+ filtered_skills = self._filter_skills(skills, toolchain, categories)
166
+
167
+ self.logger.info(
168
+ f"After filtering: {len(filtered_skills)} skills to deploy"
169
+ f" (toolchain={toolchain}, categories={categories})"
170
+ )
171
+
172
+ # Step 3.5: Apply selective filtering (only agent-referenced skills)
173
+ total_available = len(filtered_skills)
174
+ if selective:
175
+ # Auto-detect project root if not provided
176
+ if project_root is None:
177
+ # Try to find project root by looking for .claude directory
178
+ # Start from current directory and walk up
179
+ current = Path.cwd()
180
+ while current != current.parent:
181
+ if (current / ".claude").exists():
182
+ project_root = current
183
+ break
184
+ current = current.parent
185
+
186
+ if project_root:
187
+ agents_dir = Path(project_root) / ".claude" / "agents"
188
+ else:
189
+ # Fallback to current directory's .claude/agents
190
+ agents_dir = Path.cwd() / ".claude" / "agents"
191
+
192
+ from claude_mpm.services.skills.selective_skill_deployer import (
193
+ get_required_skills_from_agents,
194
+ )
195
+
196
+ required_skill_names = get_required_skills_from_agents(agents_dir)
197
+
198
+ if required_skill_names:
199
+ # Filter to only required skills
200
+ # Match on either 'name' or 'skill_id' field
201
+ filtered_skills = [
202
+ s
203
+ for s in filtered_skills
204
+ if s.get("name") in required_skill_names
205
+ or s.get("skill_id") in required_skill_names
206
+ ]
207
+
208
+ self.logger.info(
209
+ f"Selective deployment: {len(filtered_skills)}/{total_available} skills "
210
+ f"(agent-referenced only)"
211
+ )
212
+ else:
213
+ self.logger.warning(
214
+ f"No skills found in agent frontmatter at {agents_dir}. "
215
+ f"Deploying all {total_available} skills."
216
+ )
217
+ else:
218
+ self.logger.info(
219
+ f"Selective mode disabled: deploying all {total_available} skills"
220
+ )
221
+
222
+ # Step 4: Deploy skills
223
+ deployed = []
224
+ skipped = []
225
+ errors = []
226
+
227
+ # Extract skill names for cleanup (needed regardless of deployment outcome)
228
+ filtered_skills_names = [
229
+ skill["name"]
230
+ for skill in filtered_skills
231
+ if isinstance(skill, dict) and "name" in skill
232
+ ]
233
+
234
+ for skill in filtered_skills:
235
+ try:
236
+ # Validate skill is a dictionary
237
+ if not isinstance(skill, dict):
238
+ self.logger.error(f"Invalid skill format: {skill}")
239
+ errors.append(f"Invalid skill format: {skill}")
240
+ continue
241
+
242
+ result = self._deploy_skill(
243
+ skill, skills_data["temp_dir"], collection_name, force=force
244
+ )
245
+ if result["deployed"]:
246
+ deployed.append(skill["name"])
247
+ elif result["skipped"]:
248
+ skipped.append(skill["name"])
249
+ if result["error"]:
250
+ errors.append(result["error"])
251
+ except Exception as e:
252
+ skill_name = (
253
+ skill.get("name", "unknown")
254
+ if isinstance(skill, dict)
255
+ else "unknown"
256
+ )
257
+ self.logger.error(f"Failed to deploy {skill_name}: {e}")
258
+ errors.append(f"{skill_name}: {e}")
259
+
260
+ # Step 5: Cleanup orphaned skills (if selective mode enabled)
261
+ cleanup_result = {"removed_count": 0, "removed_skills": []}
262
+ if selective and len(deployed) > 0:
263
+ # Get the set of skills that should remain deployed
264
+ # This is the union of what we just deployed and what was already there
265
+ try:
266
+ from claude_mpm.services.skills.selective_skill_deployer import (
267
+ cleanup_orphan_skills,
268
+ )
269
+
270
+ # Only cleanup if we're in selective mode
271
+ cleanup_result = cleanup_orphan_skills(
272
+ self.CLAUDE_SKILLS_DIR, set(filtered_skills_names)
273
+ )
274
+
275
+ if cleanup_result["removed_count"] > 0:
276
+ self.logger.info(
277
+ f"Removed {cleanup_result['removed_count']} orphaned skills: "
278
+ f"{', '.join(cleanup_result['removed_skills'])}"
279
+ )
280
+ except Exception as e:
281
+ self.logger.warning(f"Failed to cleanup orphaned skills: {e}")
282
+
283
+ # Step 6: Cleanup temp directory
284
+ self._cleanup(skills_data["temp_dir"])
285
+
286
+ # Step 7: Check if Claude Code restart needed
287
+ restart_required = len(deployed) > 0
288
+ restart_instructions = ""
289
+
290
+ if restart_required:
291
+ claude_running = self._is_claude_code_running()
292
+ if claude_running:
293
+ restart_instructions = (
294
+ "⚠️ Claude Code is currently running.\n"
295
+ "Skills are only loaded at STARTUP.\n"
296
+ "Please restart Claude Code for new skills to be available.\n\n"
297
+ "How to restart Claude Code:\n"
298
+ "1. Close all Claude Code windows\n"
299
+ "2. Quit Claude Code completely (Cmd+Q on Mac, Alt+F4 on Windows)\n"
300
+ "3. Re-launch Claude Code\n"
301
+ )
302
+ else:
303
+ restart_instructions = (
304
+ "✓ Claude Code is not currently running.\n"
305
+ "Skills will be available when you launch Claude Code.\n"
306
+ )
307
+
308
+ self.logger.info(
309
+ f"Deployment complete: {len(deployed)} deployed, "
310
+ f"{len(skipped)} skipped, {len(errors)} errors, "
311
+ f"{cleanup_result['removed_count']} orphaned skills removed"
312
+ )
313
+
314
+ return {
315
+ "deployed_count": len(deployed),
316
+ "skipped_count": len(skipped),
317
+ "errors": errors,
318
+ "deployed_skills": deployed,
319
+ "skipped_skills": skipped,
320
+ "restart_required": restart_required,
321
+ "restart_instructions": restart_instructions,
322
+ "collection": collection_name,
323
+ "selective_mode": selective,
324
+ "total_available": total_available,
325
+ "cleanup": cleanup_result,
326
+ }
327
+
328
+ def list_available_skills(self, collection: Optional[str] = None) -> Dict:
329
+ """List all available skills from GitHub repository.
330
+
331
+ Downloads manifest and returns skill metadata grouped by category
332
+ and toolchain.
333
+
334
+ Args:
335
+ collection: Collection name to list from (default: uses default collection)
336
+
337
+ Returns:
338
+ Dict containing:
339
+ - total_skills: Total number of available skills
340
+ - by_category: Skills grouped by category
341
+ - by_toolchain: Skills grouped by toolchain
342
+ - skills: Full list of skill metadata
343
+ - collection: Collection name used
344
+
345
+ Example:
346
+ >>> result = deployer.list_available_skills(collection="obra-superpowers")
347
+ >>> result = deployer.list_available_skills() # Uses default
348
+ >>> print(f"Available: {result['total_skills']} skills")
349
+ >>> for category, skills in result['by_category'].items():
350
+ >>> print(f"{category}: {len(skills)} skills")
351
+ """
352
+ collection_name = collection or self.skills_config.get_default_collection()
353
+
354
+ try:
355
+ skills_data = self._download_from_github(collection_name)
356
+ manifest = skills_data.get("manifest", {})
357
+
358
+ # Flatten skills from manifest (supports both legacy and new structure)
359
+ try:
360
+ skills = self._flatten_manifest_skills(manifest)
361
+ except ValueError as e:
362
+ self.logger.error(f"Failed to parse manifest: {e}")
363
+ return {
364
+ "total_skills": 0,
365
+ "by_category": {},
366
+ "by_toolchain": {},
367
+ "skills": [],
368
+ "error": str(e),
369
+ }
370
+
371
+ # Group by category
372
+ by_category = {}
373
+ for skill in skills:
374
+ if not isinstance(skill, dict):
375
+ continue
376
+ category = skill.get("category", "uncategorized")
377
+ if category not in by_category:
378
+ by_category[category] = []
379
+ by_category[category].append(skill)
380
+
381
+ # Group by toolchain
382
+ by_toolchain = {}
383
+ for skill in skills:
384
+ if not isinstance(skill, dict):
385
+ continue
386
+ toolchains = skill.get("toolchain", [])
387
+ if isinstance(toolchains, str):
388
+ toolchains = [toolchains]
389
+ elif not isinstance(toolchains, list):
390
+ toolchains = []
391
+
392
+ for toolchain in toolchains:
393
+ if toolchain not in by_toolchain:
394
+ by_toolchain[toolchain] = []
395
+ by_toolchain[toolchain].append(skill)
396
+
397
+ # Cleanup
398
+ self._cleanup(skills_data["temp_dir"])
399
+
400
+ return {
401
+ "total_skills": len(skills),
402
+ "by_category": by_category,
403
+ "by_toolchain": by_toolchain,
404
+ "skills": skills,
405
+ "collection": collection_name,
406
+ }
407
+
408
+ except Exception as e:
409
+ self.logger.error(f"Failed to list available skills: {e}")
410
+ return {
411
+ "total_skills": 0,
412
+ "by_category": {},
413
+ "by_toolchain": {},
414
+ "skills": [],
415
+ "error": str(e),
416
+ "collection": collection_name,
417
+ }
418
+
419
+ def check_deployed_skills(self) -> Dict:
420
+ """Check which skills are currently deployed.
421
+
422
+ Scans ~/.claude/skills/ directory for deployed skills.
423
+
424
+ Returns:
425
+ Dict containing:
426
+ - deployed_count: Number of deployed skills
427
+ - skills: List of deployed skill names with paths
428
+ - claude_skills_dir: Path to Claude skills directory
429
+
430
+ Example:
431
+ >>> result = deployer.check_deployed_skills()
432
+ >>> print(f"Currently deployed: {result['deployed_count']} skills")
433
+ """
434
+ deployed_skills = []
435
+
436
+ if self.CLAUDE_SKILLS_DIR.exists():
437
+ for skill_dir in self.CLAUDE_SKILLS_DIR.iterdir():
438
+ if skill_dir.is_dir() and not skill_dir.name.startswith("."):
439
+ # Check for SKILL.md
440
+ skill_md = skill_dir / "SKILL.md"
441
+ if skill_md.exists():
442
+ deployed_skills.append(
443
+ {"name": skill_dir.name, "path": str(skill_dir)}
444
+ )
445
+
446
+ return {
447
+ "deployed_count": len(deployed_skills),
448
+ "skills": deployed_skills,
449
+ "claude_skills_dir": str(self.CLAUDE_SKILLS_DIR),
450
+ }
451
+
452
+ def remove_skills(self, skill_names: Optional[List[str]] = None) -> Dict:
453
+ """Remove deployed skills.
454
+
455
+ Args:
456
+ skill_names: List of skill names to remove, or None to remove all
457
+
458
+ Returns:
459
+ Dict containing:
460
+ - removed_count: Number of skills removed
461
+ - removed_skills: List of removed skill names
462
+ - errors: List of error messages
463
+
464
+ Example:
465
+ >>> # Remove specific skills
466
+ >>> result = deployer.remove_skills(['test-skill', 'debug-skill'])
467
+ >>> # Remove all skills
468
+ >>> result = deployer.remove_skills()
469
+ """
470
+ removed = []
471
+ errors = []
472
+
473
+ if not self.CLAUDE_SKILLS_DIR.exists():
474
+ return {
475
+ "removed_count": 0,
476
+ "removed_skills": [],
477
+ "errors": ["Claude skills directory does not exist"],
478
+ }
479
+
480
+ # Get all skills if no specific names provided
481
+ if skill_names is None:
482
+ skill_names = [
483
+ d.name
484
+ for d in self.CLAUDE_SKILLS_DIR.iterdir()
485
+ if d.is_dir() and not d.name.startswith(".")
486
+ ]
487
+
488
+ for skill_name in skill_names:
489
+ skill_dir = self.CLAUDE_SKILLS_DIR / skill_name
490
+
491
+ if not skill_dir.exists():
492
+ errors.append(f"Skill not found: {skill_name}")
493
+ continue
494
+
495
+ try:
496
+ # Security: Validate path is within CLAUDE_SKILLS_DIR
497
+ if not self._validate_safe_path(self.CLAUDE_SKILLS_DIR, skill_dir):
498
+ raise ValueError(f"Path traversal attempt detected: {skill_dir}")
499
+
500
+ # Remove skill directory
501
+ if skill_dir.is_symlink():
502
+ self.logger.warning(f"Removing symlink: {skill_dir}")
503
+ skill_dir.unlink()
504
+ else:
505
+ shutil.rmtree(skill_dir)
506
+
507
+ removed.append(skill_name)
508
+ self.logger.info(f"Removed skill: {skill_name}")
509
+
510
+ # Untrack skill from deployment index
511
+ from claude_mpm.services.skills.selective_skill_deployer import (
512
+ untrack_skill,
513
+ )
514
+
515
+ untrack_skill(self.CLAUDE_SKILLS_DIR, skill_name)
516
+
517
+ except Exception as e:
518
+ self.logger.error(f"Failed to remove {skill_name}: {e}")
519
+ errors.append(f"{skill_name}: {e}")
520
+
521
+ return {
522
+ "removed_count": len(removed),
523
+ "removed_skills": removed,
524
+ "errors": errors,
525
+ }
526
+
527
+ def _download_from_github(self, collection_name: str) -> Dict:
528
+ """Download skills repository from GitHub using git clone/pull.
529
+
530
+ Logic:
531
+ 1. Get collection config from SkillsConfig
532
+ 2. Determine target directory: ~/.claude/skills/{collection_name}/
533
+ 3. Check if directory exists:
534
+ - Exists + is git repo → git pull (update)
535
+ - Exists + not git repo → error (manual cleanup needed)
536
+ - Not exists → git clone (first install)
537
+ 4. Parse manifest.json from collection
538
+ 5. Update last_update timestamp in config
539
+ 6. Return skills data
540
+
541
+ Args:
542
+ collection_name: Name of collection to download
543
+
544
+ Returns:
545
+ Dict containing:
546
+ - temp_dir: Path to collection directory (not temp, but kept for compatibility)
547
+ - manifest: Parsed manifest.json
548
+ - repo_dir: Path to repository directory
549
+
550
+ Raises:
551
+ ValueError: If collection not found or disabled
552
+ Exception: If git operations fail
553
+ """
554
+ # Get collection configuration
555
+ collection_config = self.skills_config.get_collection(collection_name)
556
+ if not collection_config:
557
+ raise ValueError(
558
+ f"Collection '{collection_name}' not found. "
559
+ f"Use 'claude-mpm skills collection add' to add it."
560
+ )
561
+
562
+ if not collection_config.get("enabled", True):
563
+ raise ValueError(
564
+ f"Collection '{collection_name}' is disabled. "
565
+ f"Use 'claude-mpm skills collection enable {collection_name}' to enable it."
566
+ )
567
+
568
+ repo_url = collection_config["url"]
569
+ target_dir = self.CLAUDE_SKILLS_DIR / collection_name
570
+
571
+ self.logger.info(f"Processing collection '{collection_name}' from {repo_url}")
572
+
573
+ # Check if directory exists and handle accordingly
574
+ if target_dir.exists():
575
+ git_dir = target_dir / ".git"
576
+
577
+ if git_dir.exists():
578
+ # Update existing: git pull
579
+ self.logger.info(
580
+ f"Updating existing collection '{collection_name}' at {target_dir}"
581
+ )
582
+ try:
583
+ result = subprocess.run(
584
+ ["git", "pull"],
585
+ cwd=target_dir,
586
+ capture_output=True,
587
+ text=True,
588
+ check=True,
589
+ timeout=60,
590
+ )
591
+ self.logger.debug(f"Git pull output: {result.stdout}")
592
+
593
+ except subprocess.CalledProcessError as e:
594
+ raise Exception(
595
+ f"Failed to update collection '{collection_name}': {e.stderr}"
596
+ ) from e
597
+ except subprocess.TimeoutExpired as e:
598
+ raise Exception(
599
+ f"Git pull timed out for collection '{collection_name}'"
600
+ ) from e
601
+ else:
602
+ # Directory exists but not a git repo - error
603
+ raise ValueError(
604
+ f"Directory {target_dir} exists but is not a git repository. "
605
+ f"Please remove it manually and try again:\n"
606
+ f" rm -rf {target_dir}"
607
+ )
608
+ else:
609
+ # First install: git clone
610
+ self.logger.info(
611
+ f"Installing new collection '{collection_name}' to {target_dir}"
612
+ )
613
+ try:
614
+ result = subprocess.run(
615
+ ["git", "clone", repo_url, str(target_dir)],
616
+ capture_output=True,
617
+ text=True,
618
+ check=True,
619
+ timeout=120,
620
+ )
621
+ self.logger.debug(f"Git clone output: {result.stdout}")
622
+
623
+ except subprocess.CalledProcessError as e:
624
+ raise Exception(
625
+ f"Failed to clone collection '{collection_name}': {e.stderr}"
626
+ ) from e
627
+ except subprocess.TimeoutExpired as e:
628
+ raise Exception(
629
+ f"Git clone timed out for collection '{collection_name}'"
630
+ ) from e
631
+
632
+ # Update last_update timestamp
633
+ self.skills_config.update_collection_timestamp(collection_name)
634
+
635
+ # Parse manifest.json
636
+ manifest_path = target_dir / "manifest.json"
637
+ if not manifest_path.exists():
638
+ raise Exception(
639
+ f"manifest.json not found in collection '{collection_name}' at {target_dir}"
640
+ )
641
+
642
+ try:
643
+ with open(manifest_path, encoding="utf-8") as f:
644
+ manifest = json.load(f)
645
+ except json.JSONDecodeError as e:
646
+ raise Exception(
647
+ f"Invalid manifest.json in collection '{collection_name}': {e}"
648
+ ) from e
649
+
650
+ self.logger.info(
651
+ f"Successfully loaded collection '{collection_name}' from {target_dir}"
652
+ )
653
+
654
+ # Return data in same format as before for compatibility
655
+ # Note: temp_dir is now the persistent collection directory
656
+ return {"temp_dir": target_dir, "manifest": manifest, "repo_dir": target_dir}
657
+
658
+ def _flatten_manifest_skills(self, manifest: Dict) -> List[Dict]:
659
+ """Flatten skills from manifest, supporting both structures.
660
+
661
+ Supports both legacy flat list and new nested dict structures:
662
+ - Legacy: {"skills": [skill1, skill2, ...]}
663
+ - New: {"skills": {"universal": [...], "toolchains": {...}}}
664
+
665
+ Args:
666
+ manifest: The manifest dictionary
667
+
668
+ Returns:
669
+ List of flattened skill dictionaries
670
+
671
+ Raises:
672
+ ValueError: If manifest structure is invalid
673
+
674
+ Example:
675
+ >>> # Legacy flat structure
676
+ >>> manifest = {"skills": [{"name": "skill1"}, {"name": "skill2"}]}
677
+ >>> skills = deployer._flatten_manifest_skills(manifest)
678
+ >>> len(skills) # 2
679
+
680
+ >>> # New nested structure
681
+ >>> manifest = {
682
+ ... "skills": {
683
+ ... "universal": [{"name": "skill1"}],
684
+ ... "toolchains": {"python": [{"name": "skill2"}]}
685
+ ... }
686
+ ... }
687
+ >>> skills = deployer._flatten_manifest_skills(manifest)
688
+ >>> len(skills) # 2
689
+ """
690
+ skills_data = manifest.get("skills", {})
691
+
692
+ # Handle legacy flat list structure
693
+ if isinstance(skills_data, list):
694
+ self.logger.debug(
695
+ f"Using legacy flat manifest structure ({len(skills_data)} skills)"
696
+ )
697
+ return skills_data
698
+
699
+ # Handle new nested dict structure
700
+ if isinstance(skills_data, dict):
701
+ flat_skills = []
702
+
703
+ # Add universal skills
704
+ universal_skills = skills_data.get("universal", [])
705
+ if isinstance(universal_skills, list):
706
+ flat_skills.extend(universal_skills)
707
+ self.logger.debug(f"Added {len(universal_skills)} universal skills")
708
+
709
+ # Add toolchain-specific skills
710
+ toolchains = skills_data.get("toolchains", {})
711
+ if isinstance(toolchains, dict):
712
+ for toolchain_name, toolchain_skills in toolchains.items():
713
+ if isinstance(toolchain_skills, list):
714
+ flat_skills.extend(toolchain_skills)
715
+ self.logger.debug(
716
+ f"Added {len(toolchain_skills)} {toolchain_name} skills"
717
+ )
718
+
719
+ self.logger.info(
720
+ f"Flattened {len(flat_skills)} total skills from nested structure"
721
+ )
722
+ return flat_skills
723
+
724
+ # Invalid structure
725
+ raise ValueError(
726
+ f"Skills manifest must be a list or dict, got {type(skills_data).__name__}"
727
+ )
728
+
729
+ def _filter_skills(
730
+ self,
731
+ skills: List[Dict],
732
+ toolchain: Optional[List[str]] = None,
733
+ categories: Optional[List[str]] = None,
734
+ ) -> List[Dict]:
735
+ """Filter skills by toolchain and categories.
736
+
737
+ Args:
738
+ skills: List of skill metadata dicts
739
+ toolchain: List of toolchains to include (None = all)
740
+ categories: List of categories to include (None = all)
741
+
742
+ Returns:
743
+ Filtered list of skills
744
+ """
745
+ # Ensure skills is a list and contains dicts
746
+ if not isinstance(skills, list):
747
+ return []
748
+
749
+ # Filter out non-dict items
750
+ filtered = [s for s in skills if isinstance(s, dict)]
751
+
752
+ # Filter by toolchain
753
+ if toolchain:
754
+ toolchain_lower = [t.lower() for t in toolchain]
755
+ filtered = [
756
+ s
757
+ for s in filtered
758
+ if isinstance(s, dict)
759
+ and any(
760
+ t.lower() in toolchain_lower
761
+ for t in (
762
+ s.get("toolchain", [])
763
+ if isinstance(s.get("toolchain"), list)
764
+ else ([s.get("toolchain")] if s.get("toolchain") else [])
765
+ )
766
+ )
767
+ ]
768
+
769
+ # Filter by categories
770
+ if categories:
771
+ categories_lower = [c.lower() for c in categories]
772
+ filtered = [
773
+ s
774
+ for s in filtered
775
+ if isinstance(s, dict)
776
+ and s.get("category", "").lower() in categories_lower
777
+ ]
778
+
779
+ return filtered
780
+
781
+ def _deploy_skill(
782
+ self,
783
+ skill: Dict,
784
+ collection_dir: Path,
785
+ collection_name: str,
786
+ force: bool = False,
787
+ ) -> Dict:
788
+ """Deploy a single skill to ~/.claude/skills/ and track deployment.
789
+
790
+ NOTE: With multi-collection support, skills are now stored in collection
791
+ subdirectories. This method creates symlinks or copies to maintain the
792
+ flat structure that Claude Code expects in ~/.claude/skills/.
793
+
794
+ Additionally tracks deployed skills in .mpm-deployed-skills.json index
795
+ for orphan cleanup functionality.
796
+
797
+ Args:
798
+ skill: Skill metadata dict
799
+ collection_dir: Collection directory containing skills
800
+ collection_name: Name of collection (for tracking)
801
+ force: Overwrite if already exists
802
+
803
+ Returns:
804
+ Dict with deployed, skipped, error flags
805
+ """
806
+ skill_name = skill["name"]
807
+ target_dir = self.CLAUDE_SKILLS_DIR / skill_name
808
+
809
+ # Check if already deployed
810
+ if target_dir.exists() and not force:
811
+ self.logger.debug(f"Skipped {skill_name} (already deployed)")
812
+ return {"deployed": False, "skipped": True, "error": None}
813
+
814
+ # Find skill source in collection directory
815
+ # Updated structure: collection_dir / skills / category / skill-name
816
+ # OR: collection_dir / universal / skill-name
817
+ # OR: collection_dir / toolchains / toolchain-name / skill-name
818
+
819
+ skills_base = collection_dir / "skills"
820
+ category = skill.get("category", "")
821
+
822
+ # Try multiple possible locations
823
+ source_dir = None
824
+ search_paths = []
825
+
826
+ # Try category-based path
827
+ if category and skills_base.exists():
828
+ search_paths.append(skills_base / category / skill_name)
829
+
830
+ # Try universal/toolchains structure
831
+ if (collection_dir / "universal").exists():
832
+ search_paths.append(collection_dir / "universal" / skill_name)
833
+
834
+ if (collection_dir / "toolchains").exists():
835
+ toolchain_dir = collection_dir / "toolchains"
836
+ for tc in toolchain_dir.iterdir():
837
+ if tc.is_dir():
838
+ search_paths.append(tc / skill_name)
839
+
840
+ # Search in all possible locations
841
+ for path in search_paths:
842
+ if path.exists():
843
+ source_dir = path
844
+ break
845
+
846
+ # Fallback: search recursively for skill in skills directory
847
+ if not source_dir and skills_base.exists():
848
+ for cat_dir in skills_base.iterdir():
849
+ if not cat_dir.is_dir():
850
+ continue
851
+ potential = cat_dir / skill_name
852
+ if potential.exists():
853
+ source_dir = potential
854
+ break
855
+
856
+ if not source_dir or not source_dir.exists():
857
+ return {
858
+ "deployed": False,
859
+ "skipped": False,
860
+ "error": f"Skill source not found: {skill_name} (searched in {collection_dir})",
861
+ }
862
+
863
+ # Security: Validate paths
864
+ if not self._validate_safe_path(collection_dir, source_dir):
865
+ return {
866
+ "deployed": False,
867
+ "skipped": False,
868
+ "error": f"Invalid source path: {source_dir}",
869
+ }
870
+
871
+ if not self._validate_safe_path(self.CLAUDE_SKILLS_DIR, target_dir):
872
+ return {
873
+ "deployed": False,
874
+ "skipped": False,
875
+ "error": f"Invalid target path: {target_dir}",
876
+ }
877
+
878
+ try:
879
+ # Remove existing if force
880
+ if target_dir.exists():
881
+ if target_dir.is_symlink():
882
+ target_dir.unlink()
883
+ else:
884
+ shutil.rmtree(target_dir)
885
+
886
+ # Copy skill to Claude skills directory
887
+ # NOTE: We use copy instead of symlink to maintain Claude Code compatibility
888
+ shutil.copytree(source_dir, target_dir)
889
+
890
+ # Track deployment in index
891
+ from claude_mpm.services.skills.selective_skill_deployer import (
892
+ track_deployed_skill,
893
+ )
894
+
895
+ track_deployed_skill(self.CLAUDE_SKILLS_DIR, skill_name, collection_name)
896
+
897
+ self.logger.debug(
898
+ f"Deployed {skill_name} from {source_dir} to {target_dir}"
899
+ )
900
+ return {"deployed": True, "skipped": False, "error": None}
901
+
902
+ except Exception as e:
903
+ return {"deployed": False, "skipped": False, "error": str(e)}
904
+
905
+ def _validate_safe_path(self, base: Path, target: Path) -> bool:
906
+ """Ensure target path is within base directory (security).
907
+
908
+ Args:
909
+ base: Base directory
910
+ target: Target path to validate
911
+
912
+ Returns:
913
+ True if path is safe, False otherwise
914
+ """
915
+ try:
916
+ target.resolve().relative_to(base.resolve())
917
+ return True
918
+ except ValueError:
919
+ return False
920
+
921
+ def _is_claude_code_running(self) -> bool:
922
+ """Check if Claude Code process is running.
923
+
924
+ Returns:
925
+ True if Claude Code is running, False otherwise
926
+ """
927
+ try:
928
+ if platform.system() == "Windows":
929
+ result = subprocess.run(
930
+ ["tasklist"], check=False, capture_output=True, text=True, timeout=5
931
+ )
932
+ return "claude" in result.stdout.lower()
933
+ # macOS and Linux
934
+ result = subprocess.run(
935
+ ["ps", "aux"], check=False, capture_output=True, text=True, timeout=5
936
+ )
937
+ # Look for "Claude Code" or "claude-code" process
938
+ return (
939
+ "claude code" in result.stdout.lower()
940
+ or "claude-code" in result.stdout.lower()
941
+ )
942
+
943
+ except Exception as e:
944
+ self.logger.debug(f"Failed to check Claude Code process: {e}")
945
+ return False
946
+
947
+ def _cleanup(self, temp_dir: Path) -> None:
948
+ """Cleanup temporary directory.
949
+
950
+ NOTE: With multi-collection support, temp_dir is now the persistent
951
+ collection directory, so we DON'T delete it. This method is kept for
952
+ backward compatibility but is now a no-op.
953
+
954
+ Args:
955
+ temp_dir: Collection directory (not deleted)
956
+ """
957
+ # NO-OP: Collection directories are persistent, not temporary
958
+ # Skills are deployed from collection directories to Claude skills dir
959
+ self.logger.debug(f"Collection directory preserved at {temp_dir} (not deleted)")
960
+
961
+ # === Collection Management Methods ===
962
+
963
+ def list_collections(self) -> Dict[str, Any]:
964
+ """List all configured skill collections.
965
+
966
+ Returns:
967
+ Dict containing:
968
+ - collections: Dict of collection configurations
969
+ - default_collection: Name of default collection
970
+ - enabled_count: Number of enabled collections
971
+
972
+ Example:
973
+ >>> result = deployer.list_collections()
974
+ >>> for name, config in result['collections'].items():
975
+ >>> print(f"{name}: {config['url']} (priority: {config['priority']})")
976
+ """
977
+ collections = self.skills_config.get_collections()
978
+ default = self.skills_config.get_default_collection()
979
+ enabled = self.skills_config.get_enabled_collections()
980
+
981
+ return {
982
+ "collections": collections,
983
+ "default_collection": default,
984
+ "enabled_count": len(enabled),
985
+ "total_count": len(collections),
986
+ }
987
+
988
+ def add_collection(self, name: str, url: str, priority: int = 99) -> Dict[str, Any]:
989
+ """Add a new skill collection.
990
+
991
+ Args:
992
+ name: Collection name (must be unique)
993
+ url: GitHub repository URL
994
+ priority: Collection priority (lower = higher priority, default: 99)
995
+
996
+ Returns:
997
+ Dict with operation result
998
+
999
+ Example:
1000
+ >>> deployer.add_collection("obra-superpowers", "https://github.com/obra/superpowers")
1001
+ """
1002
+ return self.skills_config.add_collection(name, url, priority)
1003
+
1004
+ def remove_collection(self, name: str) -> Dict[str, Any]:
1005
+ """Remove a skill collection.
1006
+
1007
+ Args:
1008
+ name: Collection name to remove
1009
+
1010
+ Returns:
1011
+ Dict with operation result
1012
+
1013
+ Example:
1014
+ >>> deployer.remove_collection("obra-superpowers")
1015
+ """
1016
+ result = self.skills_config.remove_collection(name)
1017
+
1018
+ # Also remove the collection directory
1019
+ collection_dir = self.CLAUDE_SKILLS_DIR / name
1020
+ if collection_dir.exists():
1021
+ try:
1022
+ shutil.rmtree(collection_dir)
1023
+ self.logger.info(f"Removed collection directory: {collection_dir}")
1024
+ result["directory_removed"] = True
1025
+ except Exception as e:
1026
+ self.logger.warning(f"Failed to remove directory {collection_dir}: {e}")
1027
+ result["directory_removed"] = False
1028
+ result["directory_error"] = str(e)
1029
+
1030
+ return result
1031
+
1032
+ def enable_collection(self, name: str) -> Dict[str, Any]:
1033
+ """Enable a disabled collection.
1034
+
1035
+ Args:
1036
+ name: Collection name
1037
+
1038
+ Returns:
1039
+ Dict with operation result
1040
+
1041
+ Example:
1042
+ >>> deployer.enable_collection("obra-superpowers")
1043
+ """
1044
+ return self.skills_config.enable_collection(name)
1045
+
1046
+ def disable_collection(self, name: str) -> Dict[str, Any]:
1047
+ """Disable a collection without removing it.
1048
+
1049
+ Args:
1050
+ name: Collection name
1051
+
1052
+ Returns:
1053
+ Dict with operation result
1054
+
1055
+ Example:
1056
+ >>> deployer.disable_collection("obra-superpowers")
1057
+ """
1058
+ return self.skills_config.disable_collection(name)
1059
+
1060
+ def set_default_collection(self, name: str) -> Dict[str, Any]:
1061
+ """Set the default collection for deployments.
1062
+
1063
+ Args:
1064
+ name: Collection name to set as default
1065
+
1066
+ Returns:
1067
+ Dict with operation result
1068
+
1069
+ Example:
1070
+ >>> deployer.set_default_collection("obra-superpowers")
1071
+ """
1072
+ return self.skills_config.set_default_collection(name)