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
@@ -6,27 +6,106 @@ local agents with user-friendly prompts, intelligent defaults, and validation.
6
6
 
7
7
  import json
8
8
  import re
9
+ import shutil
9
10
  import sys
10
11
  from pathlib import Path
11
- from typing import Any, Dict, Optional, Tuple
12
+ from typing import Any, Dict, List, Optional, Tuple
13
+
14
+ import questionary
15
+ from questionary import Style
12
16
 
13
17
  from claude_mpm.core.logging_config import get_logger
14
18
  from claude_mpm.services.agents.local_template_manager import (
15
19
  LocalAgentTemplate,
16
20
  LocalAgentTemplateManager,
17
21
  )
22
+ from claude_mpm.utils.agent_filters import apply_all_filters
18
23
 
19
24
  logger = get_logger(__name__)
20
25
 
26
+ # Questionary style matching Rich cyan theme (consistent with configure.py)
27
+ QUESTIONARY_STYLE = Style(
28
+ [
29
+ ("selected", "fg:cyan bold"),
30
+ ("pointer", "fg:cyan bold"),
31
+ ("highlighted", "fg:cyan"),
32
+ ("question", "fg:cyan bold"),
33
+ ]
34
+ )
35
+
21
36
 
22
37
  class AgentWizard:
23
- """Interactive wizard for agent creation and management."""
38
+ """
39
+ Interactive wizard for agent creation and management.
40
+
41
+ DEPRECATED: This interface has been superseded by the unified
42
+ configuration interface. Please use 'claude-mpm config' instead.
43
+
44
+ This class is retained for backward compatibility but will be
45
+ removed in a future version.
46
+ """
24
47
 
25
48
  def __init__(self):
26
49
  """Initialize the agent wizard."""
27
50
  self.manager = LocalAgentTemplateManager()
28
51
  self.logger = logger
29
52
 
53
+ # Initialize remote discovery services
54
+ try:
55
+ from claude_mpm.services.agents.git_source_manager import GitSourceManager
56
+
57
+ self.source_manager = GitSourceManager()
58
+ self.discovery_enabled = True
59
+ self.logger.debug("Remote agent discovery enabled")
60
+ except Exception as e:
61
+ self.logger.warning(f"Failed to initialize remote discovery: {e}")
62
+ self.source_manager = None
63
+ self.discovery_enabled = False
64
+
65
+ @staticmethod
66
+ def _calculate_column_widths(
67
+ terminal_width: int, columns: Dict[str, int]
68
+ ) -> Dict[str, int]:
69
+ """Calculate dynamic column widths based on terminal size.
70
+
71
+ Args:
72
+ terminal_width: Current terminal width in characters
73
+ columns: Dict mapping column names to minimum widths
74
+
75
+ Returns:
76
+ Dict mapping column names to calculated widths
77
+
78
+ Design:
79
+ - Ensures minimum widths are respected
80
+ - Distributes extra space proportionally
81
+ - Handles narrow terminals gracefully (minimum 80 chars)
82
+ """
83
+ # Ensure minimum terminal width
84
+ min_terminal_width = 80
85
+ terminal_width = max(terminal_width, min_terminal_width)
86
+
87
+ # Calculate total minimum width needed
88
+ total_min_width = sum(columns.values())
89
+
90
+ # Account for spacing between columns
91
+ overhead = len(columns) + 1
92
+ available_width = terminal_width - overhead
93
+
94
+ # If we have extra space, distribute proportionally
95
+ if available_width > total_min_width:
96
+ extra_space = available_width - total_min_width
97
+ total_weight = sum(columns.values())
98
+
99
+ result = {}
100
+ for col_name, min_width in columns.items():
101
+ # Distribute extra space based on minimum width proportion
102
+ proportion = min_width / total_weight
103
+ extra = int(extra_space * proportion)
104
+ result[col_name] = min_width + extra
105
+ return result
106
+ # Terminal too narrow, use minimum widths
107
+ return columns.copy()
108
+
30
109
  def run_interactive_create(self) -> Tuple[bool, str]:
31
110
  """Run interactive agent creation wizard.
32
111
 
@@ -55,7 +134,7 @@ class AgentWizard:
55
134
  model = self._get_model_choice()
56
135
 
57
136
  # Step 5: Inheritance Option
58
- parent_agent, base_template = self._get_inheritance_option()
137
+ parent_agent, _base_template = self._get_inheritance_option()
59
138
 
60
139
  # Step 6: Capabilities Configuration
61
140
  capabilities = self._get_capabilities_configuration()
@@ -117,6 +196,80 @@ class AgentWizard:
117
196
  self.logger.error(error_msg, exc_info=True)
118
197
  return False, error_msg
119
198
 
199
+ def _merge_agent_sources(self) -> List[Dict[str, Any]]:
200
+ """
201
+ Merge agents from all sources with precedence: local > discovered.
202
+
203
+ Returns list of agents with metadata:
204
+ {
205
+ "agent_id": "engineer/backend/python-engineer",
206
+ "name": "Python Engineer",
207
+ "description": "...",
208
+ "source_type": "system" | "project",
209
+ "source_identifier": "bobmatnyc/claude-mpm-agents",
210
+ "category": "engineer/backend",
211
+ "deployed": True | False,
212
+ "path": "/path/to/agent.md"
213
+ }
214
+ """
215
+ agents = {}
216
+
217
+ # Get discovered agents (system/user sources)
218
+ if self.discovery_enabled and self.source_manager:
219
+ try:
220
+ discovered = self.source_manager.list_cached_agents()
221
+ self.logger.debug(f"Discovered {len(discovered)} remote agents")
222
+
223
+ for agent in discovered:
224
+ agent_id = agent.get("agent_id", "")
225
+ if not agent_id:
226
+ continue
227
+
228
+ # Extract metadata
229
+ metadata = agent.get("metadata", {})
230
+ agents[agent_id] = {
231
+ "agent_id": agent_id,
232
+ "name": metadata.get("name", agent_id),
233
+ "description": metadata.get("description", ""),
234
+ "source_type": "system",
235
+ "source_identifier": agent.get("source", "unknown"),
236
+ "category": agent.get("category", ""),
237
+ "deployed": False, # Will be updated below
238
+ "path": agent.get("path", agent.get("source_file", "")),
239
+ }
240
+ except Exception as e:
241
+ self.logger.warning(f"Failed to discover remote agents: {e}")
242
+
243
+ # Get local agents (project-level, highest precedence)
244
+ local_templates = self.manager.list_local_templates()
245
+ for template in local_templates:
246
+ agent_id = template.agent_id
247
+ agents[agent_id] = {
248
+ "agent_id": agent_id,
249
+ "name": template.metadata.get("name", agent_id),
250
+ "description": template.metadata.get("description", ""),
251
+ "source_type": "project",
252
+ "source_identifier": "local",
253
+ "category": template.metadata.get("category", ""),
254
+ "deployed": True, # Local templates are deployed
255
+ "path": str(self._get_template_path(template)),
256
+ }
257
+
258
+ # Check deployment status for discovered agents
259
+ deployed_dir = Path.cwd() / ".claude" / "agents"
260
+ if deployed_dir.exists():
261
+ for agent_id, agent_data in agents.items():
262
+ deployed_file = deployed_dir / f"{agent_id.replace('/', '-')}.md"
263
+ # Also check hierarchical path
264
+ deployed_file_alt = deployed_dir / f"{agent_id.split('/')[-1]}.md"
265
+ if deployed_file.exists() or deployed_file_alt.exists():
266
+ agent_data["deployed"] = True
267
+
268
+ # Filter BASE_AGENT from all agent lists (1M-502 Phase 1)
269
+ # BASE_AGENT is a build tool, not a deployable agent
270
+ agent_list = list(agents.values())
271
+ return apply_all_filters(agent_list, filter_base=True, filter_deployed=False)
272
+
120
273
  def run_interactive_manage(self) -> Tuple[bool, str]:
121
274
  """Run interactive agent management menu.
122
275
 
@@ -125,15 +278,17 @@ class AgentWizard:
125
278
  """
126
279
  try:
127
280
  while True:
128
- # List current local agents
129
- templates = self.manager.list_local_templates()
281
+ # Get merged agents from all sources
282
+ all_agents = self._merge_agent_sources()
130
283
 
131
284
  print("\n" + "=" * 60)
132
285
  print("🔧 Agent Management Menu")
133
286
  print("=" * 60)
134
287
 
135
- if not templates:
136
- print("\n📭 No local agents found.")
288
+ if not all_agents:
289
+ print(
290
+ "\n📭 No agents found. Configure sources with 'claude-mpm agents discover'"
291
+ )
137
292
  print("\n1. Create new agent")
138
293
  print("2. Import agents")
139
294
  print("3. Exit")
@@ -148,60 +303,134 @@ class AgentWizard:
148
303
  return True, "Management menu exited"
149
304
  print("❌ Invalid choice. Please try again.")
150
305
  continue
151
- # Show existing agents
152
- print(f"\n📋 Found {len(templates)} local agent(s):")
153
- for i, template in enumerate(templates, 1):
154
- tier_icon = "🏢" if template.tier == "project" else "👤"
306
+
307
+ # Show existing agents in a table with dynamic widths
308
+ print(f"\n📋 Found {len(all_agents)} agent(s):\n")
309
+
310
+ # Calculate dynamic column widths based on terminal size
311
+ terminal_width = shutil.get_terminal_size().columns
312
+ min_widths = {
313
+ "#": 4,
314
+ "Agent ID": 30,
315
+ "Name": 20,
316
+ "Source": 15,
317
+ "Status": 10,
318
+ }
319
+ widths = self._calculate_column_widths(terminal_width, min_widths)
320
+
321
+ # Print header with dynamic widths
322
+ print(
323
+ f"{'#':<{widths['#']}} "
324
+ f"{'Agent ID':<{widths['Agent ID']}} "
325
+ f"{'Name':<{widths['Name']}} "
326
+ f"{'Source':<{widths['Source']}} "
327
+ f"{'Status':<{widths['Status']}}"
328
+ )
329
+ separator_width = sum(widths.values()) + len(widths) - 1
330
+ print("-" * separator_width)
331
+
332
+ for i, agent in enumerate(all_agents, 1):
333
+ agent_id = agent["agent_id"]
334
+ # Truncate to fit dynamic width
335
+ if len(agent_id) > widths["Agent ID"]:
336
+ agent_id = agent_id[: widths["Agent ID"] - 1] + "…"
337
+
338
+ name = agent["name"]
339
+ if len(name) > widths["Name"]:
340
+ name = name[: widths["Name"] - 1] + "…"
341
+
342
+ source_label = (
343
+ f"[{agent['source_type']}] {agent['source_identifier']}"
344
+ )
345
+ if len(source_label) > widths["Source"]:
346
+ source_label = source_label[: widths["Source"] - 1] + "…"
347
+
348
+ status = "✓ Deployed" if agent["deployed"] else "Available"
349
+
155
350
  print(
156
- f" {i}. {tier_icon} {template.agent_id} - {template.metadata.get('name', template.agent_id)}"
351
+ f"{i:<{widths['#']}} "
352
+ f"{agent_id:<{widths['Agent ID']}} "
353
+ f"{name:<{widths['Name']}} "
354
+ f"{source_label:<{widths['Source']}} "
355
+ f"{status:<{widths['Status']}}"
157
356
  )
158
357
 
159
- print(f"\n{len(templates) + 1}. Create new agent")
160
- print(f"{len(templates) + 2}. Delete agent(s)")
161
- print(f"{len(templates) + 3}. Import agents")
162
- print(f"{len(templates) + 4}. Export all agents")
163
- print(f"{len(templates) + 5}. Exit")
358
+ # Build menu choices with arrow-key navigation
359
+ menu_choices = []
164
360
 
165
- max_choice = len(templates) + 5
166
- choice = input(f"\nSelect option [1-{max_choice}]: ").strip()
361
+ # Add agent viewing options (1-N)
362
+ for i, agent in enumerate(all_agents, 1):
363
+ menu_choices.append(f"{i}. View agent: {agent['agent_id']}")
167
364
 
168
- try:
169
- choice_num = int(choice)
170
- except ValueError:
171
- print(" Invalid choice. Please enter a number.")
172
- continue
365
+ # Add action options
366
+ menu_choices.append(f"{len(all_agents) + 1}. Deploy agent")
367
+ menu_choices.append(f"{len(all_agents) + 2}. Create new agent")
368
+ menu_choices.append(f"{len(all_agents) + 3}. Delete agent(s)")
369
+ menu_choices.append(f"{len(all_agents) + 4}. Import agents")
370
+ menu_choices.append(f"{len(all_agents) + 5}. Export all agents")
371
+
372
+ if self.discovery_enabled:
373
+ menu_choices.append(
374
+ f"{len(all_agents) + 6}. Browse & filter agents"
375
+ )
376
+ menu_choices.append(f"{len(all_agents) + 7}. Deploy preset")
377
+ menu_choices.append(f"{len(all_agents) + 8}. Manage agent sources")
378
+ menu_choices.append(f"{len(all_agents) + 9}. Exit")
379
+ exit_num = len(all_agents) + 9
380
+ else:
381
+ menu_choices.append(f"{len(all_agents) + 6}. Exit")
382
+ exit_num = len(all_agents) + 6
383
+
384
+ choice = questionary.select(
385
+ "Agent Management Menu:",
386
+ choices=menu_choices,
387
+ style=QUESTIONARY_STYLE,
388
+ ).ask()
389
+
390
+ if not choice: # User pressed Esc
391
+ return True, "Management menu exited"
173
392
 
174
- if 1 <= choice_num <= len(templates):
175
- # Manage specific agent
176
- selected_template = templates[choice_num - 1]
177
- result = self._manage_single_agent(selected_template)
178
- if not result[0]:
179
- print(f"❌ {result[1]}")
180
- elif choice_num == len(templates) + 1:
181
- success, message = self.run_interactive_create()
393
+ # Parse choice number from "N. Description" format
394
+ choice_num = int(choice.split(".")[0])
395
+
396
+ if 1 <= choice_num <= len(all_agents):
397
+ # View agent details
398
+ selected_agent = all_agents[choice_num - 1]
399
+ self._show_agent_details(selected_agent)
400
+ continue
401
+ if choice_num == len(all_agents) + 1:
402
+ self._deploy_agent_interactive(all_agents)
403
+ elif choice_num == len(all_agents) + 2:
404
+ _, message = self.run_interactive_create()
182
405
  if message:
183
- # Message already has emoji from the function
184
406
  print(f"\n{message}")
185
- continue # Return to main menu
186
- elif choice_num == len(templates) + 2:
187
- success, message = self._interactive_delete_menu(templates)
407
+ continue
408
+ elif choice_num == len(all_agents) + 3:
409
+ local_templates = self.manager.list_local_templates()
410
+ _, message = self._interactive_delete_menu(local_templates)
188
411
  if message:
189
- # Message already has emoji from the function
190
412
  print(f"\n{message}")
191
- continue # Return to main menu
192
- elif choice_num == len(templates) + 3:
193
- success, message = self._interactive_import()
413
+ continue
414
+ elif choice_num == len(all_agents) + 4:
415
+ _, message = self._interactive_import()
194
416
  if message:
195
- # Message already has emoji from the function
196
417
  print(f"\n{message}")
197
- continue # Return to main menu
198
- elif choice_num == len(templates) + 4:
199
- success, message = self._interactive_export()
418
+ continue
419
+ elif choice_num == len(all_agents) + 5:
420
+ _success, message = self._interactive_export()
200
421
  if message:
201
- # Message already has emoji from the function
202
422
  print(f"\n{message}")
203
- continue # Return to main menu
204
- elif choice_num == len(templates) + 5:
423
+ continue
424
+ elif choice_num == len(all_agents) + 6 and self.discovery_enabled:
425
+ self._browse_agents_interactive()
426
+ continue
427
+ elif choice_num == len(all_agents) + 7 and self.discovery_enabled:
428
+ self._deploy_preset_interactive()
429
+ continue
430
+ elif choice_num == len(all_agents) + 8 and self.discovery_enabled:
431
+ self._manage_sources_interactive()
432
+ continue
433
+ elif choice_num == exit_num:
205
434
  return True, "Management menu exited"
206
435
  else:
207
436
  print("❌ Invalid choice. Please try again.")
@@ -919,6 +1148,762 @@ class AgentWizard:
919
1148
 
920
1149
  return len(results["successful"]) > 0, message.strip()
921
1150
 
1151
+ def _show_agent_details(self, agent: Dict[str, Any]) -> None:
1152
+ """Show detailed information about an agent.
1153
+
1154
+ Args:
1155
+ agent: Agent metadata dictionary
1156
+ """
1157
+ print("\n" + "=" * 60)
1158
+ print(f"📄 Agent Details: {agent['agent_id']}")
1159
+ print("=" * 60)
1160
+ print(f"Name: {agent['name']}")
1161
+ print(f"Category: {agent['category'] or 'N/A'}")
1162
+ print(f"Source: [{agent['source_type']}] {agent['source_identifier']}")
1163
+ print(f"Status: {'✓ Deployed' if agent['deployed'] else 'Available'}")
1164
+ print(f"Path: {agent['path']}")
1165
+
1166
+ if agent["description"]:
1167
+ print("\nDescription:")
1168
+ print(
1169
+ f" {agent['description'][:200]}{'...' if len(agent['description']) > 200 else ''}"
1170
+ )
1171
+
1172
+ input("\nPress Enter to continue...")
1173
+
1174
+ def _deploy_agent_interactive(self, available_agents: List[Dict[str, Any]]):
1175
+ """Interactive agent deployment.
1176
+
1177
+ Args:
1178
+ available_agents: List of all available agents
1179
+ """
1180
+ # Filter to non-deployed agents using improved detection (1M-502 Phase 1)
1181
+ # This checks both .claude-mpm/agents/ and .claude/agents/
1182
+ deployable = apply_all_filters(
1183
+ available_agents, filter_base=True, filter_deployed=True
1184
+ )
1185
+
1186
+ if not deployable:
1187
+ print("\n✅ All agents are already deployed!")
1188
+ input("\nPress Enter to continue...")
1189
+ return
1190
+
1191
+ print("\n" + "=" * 60)
1192
+ print("📦 Deploy Agent")
1193
+ print("=" * 60)
1194
+ print(f"\n{len(deployable)} agent(s) available to deploy:\n")
1195
+
1196
+ # Build agent selection choices with arrow-key navigation
1197
+ agent_choices = [
1198
+ f"{i}. {agent['agent_id']} - {agent['description'][:60]}{'...' if len(agent['description']) > 60 else ''}"
1199
+ for i, agent in enumerate(deployable, 1)
1200
+ ]
1201
+
1202
+ choice = questionary.select(
1203
+ "Select agent to deploy:", choices=agent_choices, style=QUESTIONARY_STYLE
1204
+ ).ask()
1205
+
1206
+ if not choice: # User pressed Esc
1207
+ return
1208
+
1209
+ # Parse agent index from "N. agent_id - description" format
1210
+ idx = int(choice.split(".")[0]) - 1
1211
+ agent = deployable[idx]
1212
+
1213
+ # Deploy agent using deployment service
1214
+ print(f"\n🚀 Deploying {agent['agent_id']}...")
1215
+
1216
+ try:
1217
+ # Use SingleAgentDeployer for deployment
1218
+ from claude_mpm.services.agents.deployment.agent_template_builder import (
1219
+ AgentTemplateBuilder,
1220
+ )
1221
+ from claude_mpm.services.agents.deployment.agent_version_manager import (
1222
+ AgentVersionManager,
1223
+ )
1224
+ from claude_mpm.services.agents.deployment.deployment_results_manager import (
1225
+ DeploymentResultsManager,
1226
+ )
1227
+ from claude_mpm.services.agents.deployment.single_agent_deployer import (
1228
+ SingleAgentDeployer,
1229
+ )
1230
+
1231
+ # Initialize deployment services
1232
+ template_builder = AgentTemplateBuilder()
1233
+ version_manager = AgentVersionManager()
1234
+ results_manager = DeploymentResultsManager(self.logger)
1235
+ deployer = SingleAgentDeployer(
1236
+ template_builder=template_builder,
1237
+ version_manager=version_manager,
1238
+ results_manager=results_manager,
1239
+ logger=self.logger,
1240
+ )
1241
+
1242
+ # Prepare deployment parameters
1243
+ template_path = Path(agent["path"])
1244
+ target_dir = Path.cwd() / ".claude" / "agents"
1245
+
1246
+ # Find base_agent.json in multiple possible locations
1247
+ base_agent_candidates = [
1248
+ Path.home()
1249
+ / ".claude-mpm"
1250
+ / "agents"
1251
+ / "templates"
1252
+ / "base_agent.json",
1253
+ Path.home() / ".claude-mpm" / "cache" / "base_agent.json",
1254
+ Path(__file__).parent.parent.parent
1255
+ / "agents"
1256
+ / "templates"
1257
+ / "base_agent.json",
1258
+ ]
1259
+ base_agent_path = None
1260
+ for candidate in base_agent_candidates:
1261
+ if candidate.exists():
1262
+ base_agent_path = candidate
1263
+ break
1264
+
1265
+ if not base_agent_path:
1266
+ base_agent_path = base_agent_candidates[
1267
+ 0
1268
+ ] # Use default even if not exists
1269
+
1270
+ # Deploy the agent
1271
+ success = deployer.deploy_agent(
1272
+ agent_name=agent["agent_id"],
1273
+ templates_dir=template_path.parent,
1274
+ target_dir=target_dir,
1275
+ base_agent_path=base_agent_path,
1276
+ force_rebuild=True,
1277
+ working_directory=Path.cwd(),
1278
+ )
1279
+
1280
+ if success:
1281
+ print(f"\n✅ Successfully deployed {agent['agent_id']}")
1282
+ else:
1283
+ print(f"\n❌ Failed to deploy {agent['agent_id']}")
1284
+
1285
+ except Exception as e:
1286
+ self.logger.error(f"Deployment failed: {e}", exc_info=True)
1287
+ print(f"\n❌ Deployment error: {e}")
1288
+
1289
+ input("\nPress Enter to continue...")
1290
+
1291
+ def _browse_agents_interactive(self):
1292
+ """Interactive agent browsing with filters."""
1293
+ if not self.discovery_enabled or not self.source_manager:
1294
+ print("\n❌ Discovery service not available")
1295
+ input("\nPress Enter to continue...")
1296
+ return
1297
+
1298
+ while True:
1299
+ print("\n" + "=" * 60)
1300
+ print("🔍 Browse & Filter Agents")
1301
+ print("=" * 60)
1302
+
1303
+ # Show filter menu with arrow-key navigation
1304
+ print("\n[bold]Filter by:[/bold]")
1305
+
1306
+ filter_choices = [
1307
+ "1. Category (engineer/backend, qa, ops, etc.)",
1308
+ "2. Language (python, typescript, rust, etc.)",
1309
+ "3. Framework (react, nextjs, flask, etc.)",
1310
+ "4. Show all agents",
1311
+ "← Back to main menu",
1312
+ ]
1313
+
1314
+ choice = questionary.select(
1315
+ "Browse & Filter Agents:",
1316
+ choices=filter_choices,
1317
+ style=QUESTIONARY_STYLE,
1318
+ ).ask()
1319
+
1320
+ if not choice or "Back" in choice:
1321
+ break
1322
+
1323
+ # Parse choice number if it starts with a digit
1324
+ if choice[0].isdigit():
1325
+ choice_num = choice.split(".")[0]
1326
+ else:
1327
+ break
1328
+
1329
+ filtered_agents = []
1330
+ filter_description = ""
1331
+
1332
+ if choice_num == "1":
1333
+ # Category filtering with arrow-key navigation
1334
+ categories = [
1335
+ "engineer/backend",
1336
+ "engineer/frontend",
1337
+ "qa",
1338
+ "ops",
1339
+ "documentation",
1340
+ "universal",
1341
+ ]
1342
+
1343
+ cat_choices = [f"{idx}. {cat}" for idx, cat in enumerate(categories, 1)]
1344
+
1345
+ cat_choice = questionary.select(
1346
+ "Select category:", choices=cat_choices, style=QUESTIONARY_STYLE
1347
+ ).ask()
1348
+
1349
+ if not cat_choice: # User pressed Esc
1350
+ continue
1351
+
1352
+ # Parse category from "N. category" format
1353
+ cat_idx = int(cat_choice.split(".")[0]) - 1
1354
+ category = categories[cat_idx]
1355
+ all_agents = self._merge_agent_sources()
1356
+ filtered_agents = [
1357
+ a for a in all_agents if a.get("category", "").startswith(category)
1358
+ ]
1359
+ filter_description = f"Category: {category}"
1360
+
1361
+ elif choice_num == "2":
1362
+ # Language filtering (using AUTO-DEPLOY-INDEX if available)
1363
+ language = input(
1364
+ "\nEnter language (python, typescript, rust, go, etc.): "
1365
+ ).strip()
1366
+
1367
+ try:
1368
+ # Find AUTO-DEPLOY-INDEX.md in agent repository
1369
+ from claude_mpm.services.agents.auto_deploy_index_parser import (
1370
+ AutoDeployIndexParser,
1371
+ )
1372
+
1373
+ index_path = (
1374
+ Path.home()
1375
+ / ".claude-mpm"
1376
+ / "cache"
1377
+ / "remote-agents"
1378
+ / "bobmatnyc"
1379
+ / "claude-mpm-agents"
1380
+ / "AUTO-DEPLOY-INDEX.md"
1381
+ )
1382
+ if not index_path.exists():
1383
+ print(
1384
+ f"[yellow]Could not find AUTO-DEPLOY-INDEX.md at: {index_path}[/yellow]"
1385
+ )
1386
+ input("\nPress Enter to continue...")
1387
+ continue
1388
+
1389
+ parser = AutoDeployIndexParser(index_path)
1390
+ lang_agents = parser.get_agents_by_language(language.lower())
1391
+
1392
+ # Get full agent details from discovery
1393
+ all_agents = self._merge_agent_sources()
1394
+ agent_ids = lang_agents.get("core", []) + lang_agents.get(
1395
+ "optional", []
1396
+ )
1397
+ filtered_agents = [
1398
+ a for a in all_agents if a["agent_id"] in agent_ids
1399
+ ]
1400
+ filter_description = f"Language: {language}"
1401
+ except Exception as e:
1402
+ self.logger.error(f"Language filter error: {e}", exc_info=True)
1403
+ print(f"[yellow]Could not filter by language: {e}[/yellow]")
1404
+ input("\nPress Enter to continue...")
1405
+ continue
1406
+
1407
+ elif choice_num == "3":
1408
+ # Framework filtering
1409
+ framework = input(
1410
+ "\nEnter framework (react, nextjs, flask, django, etc.): "
1411
+ ).strip()
1412
+
1413
+ try:
1414
+ from claude_mpm.services.agents.auto_deploy_index_parser import (
1415
+ AutoDeployIndexParser,
1416
+ )
1417
+
1418
+ index_path = (
1419
+ Path.home()
1420
+ / ".claude-mpm"
1421
+ / "cache"
1422
+ / "remote-agents"
1423
+ / "bobmatnyc"
1424
+ / "claude-mpm-agents"
1425
+ / "AUTO-DEPLOY-INDEX.md"
1426
+ )
1427
+ if not index_path.exists():
1428
+ print(
1429
+ f"[yellow]Could not find AUTO-DEPLOY-INDEX.md at: {index_path}[/yellow]"
1430
+ )
1431
+ input("\nPress Enter to continue...")
1432
+ continue
1433
+
1434
+ parser = AutoDeployIndexParser(index_path)
1435
+ framework_agent_ids = parser.get_agents_by_framework(
1436
+ framework.lower()
1437
+ )
1438
+
1439
+ all_agents = self._merge_agent_sources()
1440
+ filtered_agents = [
1441
+ a for a in all_agents if a["agent_id"] in framework_agent_ids
1442
+ ]
1443
+ filter_description = f"Framework: {framework}"
1444
+ except Exception as e:
1445
+ self.logger.error(f"Framework filter error: {e}", exc_info=True)
1446
+ print(f"[yellow]Could not filter by framework: {e}[/yellow]")
1447
+ input("\nPress Enter to continue...")
1448
+ continue
1449
+
1450
+ elif choice_num == "4":
1451
+ # Show all agents
1452
+ filtered_agents = self._merge_agent_sources()
1453
+ filter_description = "All agents"
1454
+ else:
1455
+ print("❌ Invalid choice")
1456
+ input("\nPress Enter to continue...")
1457
+ continue
1458
+
1459
+ # Display filtered results
1460
+ print("\n" + "=" * 60)
1461
+ print(f"📋 {filter_description} ({len(filtered_agents)} agents)")
1462
+ print("=" * 60)
1463
+
1464
+ if not filtered_agents:
1465
+ print("\n[yellow]No agents found matching filter[/yellow]")
1466
+ else:
1467
+ print(f"\n{'#':<4} {'Agent ID':<40} {'Name':<25} {'Status':<12}")
1468
+ print("-" * 85)
1469
+
1470
+ for idx, agent in enumerate(filtered_agents, 1):
1471
+ agent_id = (
1472
+ agent["agent_id"][:39]
1473
+ if len(agent["agent_id"]) > 39
1474
+ else agent["agent_id"]
1475
+ )
1476
+ name = (
1477
+ agent["name"][:24] if len(agent["name"]) > 24 else agent["name"]
1478
+ )
1479
+ status = "✓ Deployed" if agent.get("deployed") else "Available"
1480
+ print(f"{idx:<4} {agent_id:<40} {name:<25} {status:<12}")
1481
+
1482
+ print("\n[bold]Actions:[/bold]")
1483
+ print(" [d] Deploy agent from this list")
1484
+ print(" [v] View agent details")
1485
+ print(" [n] New filter")
1486
+ print(" [b] Back to main menu")
1487
+
1488
+ action = input("\nSelect action: ").strip()
1489
+
1490
+ if action == "b":
1491
+ break
1492
+ if action == "n":
1493
+ continue
1494
+ if action == "d":
1495
+ self._deploy_from_filtered_list(filtered_agents)
1496
+ elif action == "v":
1497
+ self._view_from_filtered_list(filtered_agents)
1498
+ else:
1499
+ print("❌ Invalid choice")
1500
+ input("\nPress Enter to continue...")
1501
+
1502
+ def _deploy_from_filtered_list(self, agents: List[Dict[str, Any]]):
1503
+ """Deploy an agent from a filtered list.
1504
+
1505
+ Args:
1506
+ agents: List of agent dictionaries with metadata
1507
+ """
1508
+ if not agents:
1509
+ print("\n[yellow]No agents in list[/yellow]")
1510
+ input("\nPress Enter to continue...")
1511
+ return
1512
+
1513
+ deployable = [a for a in agents if not a.get("deployed")]
1514
+
1515
+ if not deployable:
1516
+ print("\n[yellow]All agents in this list are already deployed[/yellow]")
1517
+ input("\nPress Enter to continue...")
1518
+ return
1519
+
1520
+ # Build agent selection choices
1521
+ agent_choices = [
1522
+ f"{i}. {agent['agent_id']}" for i, agent in enumerate(agents, 1)
1523
+ ]
1524
+
1525
+ agent_choice = questionary.select(
1526
+ "Select agent to deploy:", choices=agent_choices, style=QUESTIONARY_STYLE
1527
+ ).ask()
1528
+
1529
+ if not agent_choice: # User pressed Esc
1530
+ return
1531
+
1532
+ # Parse agent index from "N. agent_id" format
1533
+ idx = int(agent_choice.split(".")[0]) - 1
1534
+ agent = agents[idx]
1535
+
1536
+ if agent.get("deployed"):
1537
+ print(f"\n[yellow]{agent['agent_id']} is already deployed[/yellow]")
1538
+ else:
1539
+ print(f"\n🚀 Deploying {agent['agent_id']}...")
1540
+
1541
+ try:
1542
+ from claude_mpm.services.agents.deployment.agent_template_builder import (
1543
+ AgentTemplateBuilder,
1544
+ )
1545
+ from claude_mpm.services.agents.deployment.agent_version_manager import (
1546
+ AgentVersionManager,
1547
+ )
1548
+ from claude_mpm.services.agents.deployment.deployment_results_manager import (
1549
+ DeploymentResultsManager,
1550
+ )
1551
+ from claude_mpm.services.agents.deployment.single_agent_deployer import (
1552
+ SingleAgentDeployer,
1553
+ )
1554
+
1555
+ # Initialize deployment services
1556
+ template_builder = AgentTemplateBuilder()
1557
+ version_manager = AgentVersionManager()
1558
+ results_manager = DeploymentResultsManager(self.logger)
1559
+ deployer = SingleAgentDeployer(
1560
+ template_builder=template_builder,
1561
+ version_manager=version_manager,
1562
+ results_manager=results_manager,
1563
+ logger=self.logger,
1564
+ )
1565
+
1566
+ # Prepare deployment parameters
1567
+ template_path = Path(agent["path"])
1568
+ target_dir = Path.cwd() / ".claude" / "agents"
1569
+
1570
+ # Find base_agent.json in multiple possible locations
1571
+ base_agent_candidates = [
1572
+ Path.home()
1573
+ / ".claude-mpm"
1574
+ / "agents"
1575
+ / "templates"
1576
+ / "base_agent.json",
1577
+ Path.home() / ".claude-mpm" / "cache" / "base_agent.json",
1578
+ Path(__file__).parent.parent.parent
1579
+ / "agents"
1580
+ / "templates"
1581
+ / "base_agent.json",
1582
+ ]
1583
+ base_agent_path = None
1584
+ for candidate in base_agent_candidates:
1585
+ if candidate.exists():
1586
+ base_agent_path = candidate
1587
+ break
1588
+
1589
+ if not base_agent_path:
1590
+ base_agent_path = base_agent_candidates[
1591
+ 0
1592
+ ] # Use default even if not exists
1593
+
1594
+ # Deploy the agent
1595
+ success = deployer.deploy_agent(
1596
+ agent_name=agent["agent_id"],
1597
+ templates_dir=template_path.parent,
1598
+ target_dir=target_dir,
1599
+ base_agent_path=base_agent_path,
1600
+ force_rebuild=True,
1601
+ working_directory=Path.cwd(),
1602
+ )
1603
+
1604
+ if success:
1605
+ print(f"[green]✓ Successfully deployed {agent['agent_id']}[/green]")
1606
+ else:
1607
+ print(f"[red]✗ Failed to deploy {agent['agent_id']}[/red]")
1608
+
1609
+ except Exception as e:
1610
+ self.logger.error(f"Deployment error: {e}", exc_info=True)
1611
+ print(f"❌ Deployment error: {e}")
1612
+
1613
+ input("\nPress Enter to continue...")
1614
+
1615
+ def _view_from_filtered_list(self, agents: List[Dict[str, Any]]):
1616
+ """View details of an agent from filtered list.
1617
+
1618
+ Args:
1619
+ agents: List of agent dictionaries with metadata
1620
+ """
1621
+ if not agents:
1622
+ print("\n[yellow]No agents in list[/yellow]")
1623
+ input("\nPress Enter to continue...")
1624
+ return
1625
+
1626
+ # Build agent selection choices
1627
+ agent_choices = [
1628
+ f"{i}. {agent['agent_id']}" for i, agent in enumerate(agents, 1)
1629
+ ]
1630
+
1631
+ agent_choice = questionary.select(
1632
+ "Select agent to view:", choices=agent_choices, style=QUESTIONARY_STYLE
1633
+ ).ask()
1634
+
1635
+ if not agent_choice: # User pressed Esc
1636
+ return
1637
+
1638
+ # Parse agent index from "N. agent_id" format
1639
+ idx = int(agent_choice.split(".")[0]) - 1
1640
+ agent = agents[idx]
1641
+ self._show_agent_details(agent)
1642
+
1643
+ def _deploy_preset_interactive(self):
1644
+ """Interactive preset deployment with preview and confirmation."""
1645
+ from claude_mpm.services.agents.agent_preset_service import AgentPresetService
1646
+
1647
+ if not self.source_manager:
1648
+ print("\n❌ Source manager not available")
1649
+ input("\nPress Enter to continue...")
1650
+ return
1651
+
1652
+ preset_service = AgentPresetService(self.source_manager)
1653
+
1654
+ while True:
1655
+ print("\n" + "=" * 60)
1656
+ print("📦 Deploy Agent Preset")
1657
+ print("=" * 60)
1658
+
1659
+ # List available presets
1660
+ presets = preset_service.list_presets()
1661
+
1662
+ print(f"\n{len(presets)} preset(s) available:\n")
1663
+ print(f"{'#':<4} {'Preset':<20} {'Agents':<10} {'Description':<50}")
1664
+ print("-" * 90)
1665
+
1666
+ for idx, preset in enumerate(presets, 1):
1667
+ description = (
1668
+ preset["description"][:48] + "..."
1669
+ if len(preset["description"]) > 50
1670
+ else preset["description"]
1671
+ )
1672
+ print(
1673
+ f"{idx:<4} {preset['name']:<20} {len(preset.get('agents', [])):<10} {description:<50}"
1674
+ )
1675
+
1676
+ print("\n[bold]Actions:[/bold]")
1677
+ print(" [1-11] Select preset number")
1678
+ print(" [b] Back to main menu")
1679
+
1680
+ choice = input("\nSelect preset number or action: ").strip()
1681
+
1682
+ if choice.lower() == "b":
1683
+ break
1684
+
1685
+ try:
1686
+ idx = int(choice) - 1
1687
+ if idx < 0 or idx >= len(presets):
1688
+ raise ValueError("Out of range")
1689
+
1690
+ preset_name = presets[idx]["name"]
1691
+
1692
+ # Show preset details
1693
+ print("\n" + "=" * 60)
1694
+ print(f"📦 Preset: {preset_name}")
1695
+ print("=" * 60)
1696
+ print(f"\n[bold]Description:[/bold] {presets[idx]['description']}\n")
1697
+
1698
+ # Resolve preset
1699
+ print("🔍 Resolving preset agents...")
1700
+ resolution = preset_service.resolve_agents(
1701
+ preset_name, validate_availability=True
1702
+ )
1703
+
1704
+ if resolution.get("missing_agents"):
1705
+ print(
1706
+ f"\n⚠️ [red]Missing agents ({len(resolution['missing_agents'])}):[/red]"
1707
+ )
1708
+ for agent_id in resolution["missing_agents"]:
1709
+ print(f" • {agent_id}")
1710
+ print("\n[yellow]Cannot deploy preset with missing agents[/yellow]")
1711
+ input("\nPress Enter to continue...")
1712
+ continue
1713
+
1714
+ # Show agents to deploy
1715
+ agents = resolution.get("agents", [])
1716
+ print(f"\n[bold]Agents to deploy ({len(agents)}):[/bold]\n")
1717
+
1718
+ print(f"{'Agent ID':<40} {'Name':<25} {'Source':<25}")
1719
+ print("-" * 95)
1720
+
1721
+ for agent in agents:
1722
+ # Get agent metadata
1723
+ agent_metadata = agent.get("metadata", {})
1724
+ agent_meta_data = agent_metadata.get("metadata", {})
1725
+
1726
+ agent_id = (
1727
+ agent.get("agent_id", "")[:39]
1728
+ if len(agent.get("agent_id", "")) > 39
1729
+ else agent.get("agent_id", "")
1730
+ )
1731
+ name = (
1732
+ agent_meta_data.get("name", "")[:24]
1733
+ if len(agent_meta_data.get("name", "")) > 24
1734
+ else agent_meta_data.get("name", "")
1735
+ )
1736
+ source = (
1737
+ agent.get("source", "unknown")[:24]
1738
+ if len(agent.get("source", "unknown")) > 24
1739
+ else agent.get("source", "unknown")
1740
+ )
1741
+
1742
+ print(f"{agent_id:<40} {name:<25} {source:<25}")
1743
+
1744
+ # Confirm deployment
1745
+ print("\n[bold]Options:[/bold]")
1746
+ print(" [y] Deploy all agents")
1747
+ print(" [n] Cancel")
1748
+
1749
+ confirm = input("\nProceed with deployment? ").strip()
1750
+
1751
+ if confirm.lower() == "y":
1752
+ print(f"\n🚀 Deploying preset '{preset_name}'...\n")
1753
+
1754
+ from claude_mpm.services.agents.deployment.agent_template_builder import (
1755
+ AgentTemplateBuilder,
1756
+ )
1757
+ from claude_mpm.services.agents.deployment.agent_version_manager import (
1758
+ AgentVersionManager,
1759
+ )
1760
+ from claude_mpm.services.agents.deployment.deployment_results_manager import (
1761
+ DeploymentResultsManager,
1762
+ )
1763
+ from claude_mpm.services.agents.deployment.single_agent_deployer import (
1764
+ SingleAgentDeployer,
1765
+ )
1766
+
1767
+ # Initialize deployment services once for all agents
1768
+ template_builder = AgentTemplateBuilder()
1769
+ version_manager = AgentVersionManager()
1770
+ results_manager = DeploymentResultsManager(self.logger)
1771
+ deployer = SingleAgentDeployer(
1772
+ template_builder=template_builder,
1773
+ version_manager=version_manager,
1774
+ results_manager=results_manager,
1775
+ logger=self.logger,
1776
+ )
1777
+
1778
+ target_dir = Path.cwd() / ".claude" / "agents"
1779
+
1780
+ # Find base_agent.json
1781
+ base_agent_candidates = [
1782
+ Path.home()
1783
+ / ".claude-mpm"
1784
+ / "agents"
1785
+ / "templates"
1786
+ / "base_agent.json",
1787
+ Path.home() / ".claude-mpm" / "cache" / "base_agent.json",
1788
+ Path(__file__).parent.parent.parent
1789
+ / "agents"
1790
+ / "templates"
1791
+ / "base_agent.json",
1792
+ ]
1793
+ base_agent_path = None
1794
+ for candidate in base_agent_candidates:
1795
+ if candidate.exists():
1796
+ base_agent_path = candidate
1797
+ break
1798
+
1799
+ if not base_agent_path:
1800
+ base_agent_path = base_agent_candidates[0]
1801
+
1802
+ deployed = 0
1803
+ failed = 0
1804
+
1805
+ for agent in agents:
1806
+ agent_id = agent["agent_id"]
1807
+ agent_metadata = agent.get("metadata", {})
1808
+ agent_path = agent_metadata.get(
1809
+ "path", agent_metadata.get("source_file", "")
1810
+ )
1811
+
1812
+ if not agent_path:
1813
+ print(f" Deploying {agent_id}... [red]✗ (no path)[/red]")
1814
+ failed += 1
1815
+ continue
1816
+
1817
+ print(f" Deploying {agent_id}...", end=" ", flush=True)
1818
+
1819
+ try:
1820
+ template_path = Path(agent_path)
1821
+ success = deployer.deploy_agent(
1822
+ agent_name=agent_id,
1823
+ templates_dir=template_path.parent,
1824
+ target_dir=target_dir,
1825
+ base_agent_path=base_agent_path,
1826
+ force_rebuild=True,
1827
+ working_directory=Path.cwd(),
1828
+ )
1829
+
1830
+ if success:
1831
+ print("[green]✓[/green]")
1832
+ deployed += 1
1833
+ else:
1834
+ print("[red]✗[/red]")
1835
+ failed += 1
1836
+ except Exception as e:
1837
+ print(f"[red]✗ ({e})[/red]")
1838
+ self.logger.error(
1839
+ f"Failed to deploy {agent_id}: {e}", exc_info=True
1840
+ )
1841
+ failed += 1
1842
+
1843
+ print("\n[bold]Summary:[/bold]")
1844
+ print(f" • Deployed: {deployed}")
1845
+ print(f" • Failed: {failed}")
1846
+ print(f" • Total: {len(agents)}")
1847
+
1848
+ if failed == 0:
1849
+ print(
1850
+ f"\n[green]✓ Preset '{preset_name}' deployed successfully![/green]"
1851
+ )
1852
+ else:
1853
+ print(
1854
+ f"\n[yellow]⚠ Preset deployed with {failed} failures[/yellow]"
1855
+ )
1856
+
1857
+ input("\nPress Enter to continue...")
1858
+ break
1859
+
1860
+ except (ValueError, IndexError):
1861
+ print("❌ Invalid preset selection")
1862
+ input("\nPress Enter to continue...")
1863
+ except Exception as e:
1864
+ self.logger.error(f"Preset deployment error: {e}", exc_info=True)
1865
+ print(f"❌ Error: {e}")
1866
+ input("\nPress Enter to continue...")
1867
+
1868
+ def _manage_sources_interactive(self):
1869
+ """Interactive source management."""
1870
+ if not self.discovery_enabled or not self.source_manager:
1871
+ print("\n❌ Source manager not available")
1872
+ input("\nPress Enter to continue...")
1873
+ return
1874
+
1875
+ print("\n" + "=" * 60)
1876
+ print("🔗 Manage Agent Sources")
1877
+ print("=" * 60)
1878
+
1879
+ try:
1880
+ from claude_mpm.config.agent_sources import AgentSourceConfiguration
1881
+
1882
+ config = AgentSourceConfiguration()
1883
+ sources = config.list_sources()
1884
+
1885
+ if not sources:
1886
+ print("\n📭 No sources configured.")
1887
+ else:
1888
+ print(f"\n{len(sources)} source(s) configured:\n")
1889
+ print(f"{'Source':<40} {'Priority':<10} {'Status':<10}")
1890
+ print("-" * 60)
1891
+
1892
+ for source in sources:
1893
+ identifier = source.get("identifier", "unknown")[:39]
1894
+ priority = str(source.get("priority", 100))
1895
+ status = "✓ Active" if source.get("enabled", True) else "Disabled"
1896
+ print(f"{identifier:<40} {priority:<10} {status:<10}")
1897
+
1898
+ print("\n💡 Use 'claude-mpm agent-source' command to add/remove sources")
1899
+ print("💡 Use 'claude-mpm agents discover' command to refresh agent cache")
1900
+
1901
+ except Exception as e:
1902
+ self.logger.error(f"Failed to list sources: {e}", exc_info=True)
1903
+ print(f"\n❌ Error: {e}")
1904
+
1905
+ input("\nPress Enter to continue...")
1906
+
922
1907
 
923
1908
  def run_interactive_agent_wizard() -> int:
924
1909
  """Entry point for interactive agent wizard.