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