claude-mpm 4.25.10__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 (507) hide show
  1. claude_mpm/VERSION +1 -1
  2. claude_mpm/agents/BASE_PM.md +12 -0
  3. claude_mpm/agents/PM_INSTRUCTIONS.md +1055 -2230
  4. claude_mpm/agents/PM_INSTRUCTIONS_TEACH.md +1322 -0
  5. claude_mpm/agents/WORKFLOW.md +4 -4
  6. claude_mpm/agents/__init__.py +6 -0
  7. claude_mpm/agents/agent_loader.py +1 -4
  8. claude_mpm/agents/base_agent_loader.py +10 -35
  9. claude_mpm/agents/templates/{circuit_breakers.md → circuit-breakers.md} +576 -66
  10. claude_mpm/agents/templates/context-management-examples.md +544 -0
  11. claude_mpm/agents/templates/pr-workflow-examples.md +427 -0
  12. claude_mpm/agents/templates/research-gate-examples.md +669 -0
  13. claude_mpm/agents/templates/structured-questions-examples.md +615 -0
  14. claude_mpm/agents/templates/ticket-completeness-examples.md +139 -0
  15. claude_mpm/agents/templates/ticketing-examples.md +277 -0
  16. claude_mpm/cli/__init__.py +28 -3
  17. claude_mpm/cli/commands/__init__.py +2 -0
  18. claude_mpm/cli/commands/agent_source.py +774 -0
  19. claude_mpm/cli/commands/agent_state_manager.py +188 -30
  20. claude_mpm/cli/commands/agents.py +959 -36
  21. claude_mpm/cli/commands/agents_cleanup.py +210 -0
  22. claude_mpm/cli/commands/agents_discover.py +338 -0
  23. claude_mpm/cli/commands/auto_configure.py +537 -239
  24. claude_mpm/cli/commands/config.py +7 -4
  25. claude_mpm/cli/commands/configure.py +924 -45
  26. claude_mpm/cli/commands/configure_navigation.py +63 -46
  27. claude_mpm/cli/commands/doctor.py +10 -2
  28. claude_mpm/cli/commands/local_deploy.py +1 -4
  29. claude_mpm/cli/commands/postmortem.py +401 -0
  30. claude_mpm/cli/commands/run.py +1 -39
  31. claude_mpm/cli/commands/skill_source.py +694 -0
  32. claude_mpm/cli/commands/skills.py +322 -19
  33. claude_mpm/cli/executor.py +22 -3
  34. claude_mpm/cli/interactive/agent_wizard.py +1028 -43
  35. claude_mpm/cli/parsers/agent_source_parser.py +171 -0
  36. claude_mpm/cli/parsers/agents_parser.py +256 -4
  37. claude_mpm/cli/parsers/auto_configure_parser.py +13 -0
  38. claude_mpm/cli/parsers/base_parser.py +25 -0
  39. claude_mpm/cli/parsers/config_parser.py +96 -43
  40. claude_mpm/cli/parsers/skill_source_parser.py +169 -0
  41. claude_mpm/cli/parsers/skills_parser.py +7 -0
  42. claude_mpm/cli/parsers/source_parser.py +138 -0
  43. claude_mpm/cli/startup.py +456 -103
  44. claude_mpm/cli/startup_display.py +4 -4
  45. claude_mpm/commands/{mpm-auto-configure.md → mpm-agents-auto-configure.md} +9 -0
  46. claude_mpm/commands/mpm-agents-detect.md +9 -0
  47. claude_mpm/commands/{mpm-agents.md → mpm-agents-list.md} +9 -0
  48. claude_mpm/commands/mpm-agents-recommend.md +9 -0
  49. claude_mpm/commands/{mpm-config.md → mpm-config-view.md} +9 -0
  50. claude_mpm/commands/mpm-doctor.md +9 -0
  51. claude_mpm/commands/mpm-help.md +14 -2
  52. claude_mpm/commands/mpm-init.md +9 -0
  53. claude_mpm/commands/mpm-monitor.md +9 -0
  54. claude_mpm/commands/mpm-postmortem.md +123 -0
  55. claude_mpm/commands/{mpm-resume.md → mpm-session-resume.md} +9 -0
  56. claude_mpm/commands/mpm-status.md +9 -0
  57. claude_mpm/commands/{mpm-organize.md → mpm-ticket-organize.md} +9 -0
  58. claude_mpm/commands/mpm-ticket-view.md +552 -0
  59. claude_mpm/commands/mpm-version.md +9 -0
  60. claude_mpm/commands/mpm.md +10 -0
  61. claude_mpm/config/agent_presets.py +488 -0
  62. claude_mpm/config/agent_sources.py +325 -0
  63. claude_mpm/config/skill_presets.py +392 -0
  64. claude_mpm/config/skill_sources.py +590 -0
  65. claude_mpm/constants.py +1 -0
  66. claude_mpm/core/claude_runner.py +5 -34
  67. claude_mpm/core/config.py +16 -0
  68. claude_mpm/core/framework/__init__.py +3 -16
  69. claude_mpm/core/framework/loaders/file_loader.py +54 -101
  70. claude_mpm/core/framework/loaders/instruction_loader.py +25 -5
  71. claude_mpm/core/interactive_session.py +83 -7
  72. claude_mpm/core/oneshot_session.py +71 -8
  73. claude_mpm/core/protocols/__init__.py +23 -0
  74. claude_mpm/core/protocols/runner_protocol.py +103 -0
  75. claude_mpm/core/protocols/session_protocol.py +131 -0
  76. claude_mpm/core/shared/singleton_manager.py +11 -4
  77. claude_mpm/core/system_context.py +38 -0
  78. claude_mpm/core/unified_config.py +22 -0
  79. claude_mpm/experimental/cli_enhancements.py +1 -5
  80. claude_mpm/hooks/claude_hooks/__pycache__/__init__.cpython-313.pyc +0 -0
  81. claude_mpm/hooks/claude_hooks/__pycache__/event_handlers.cpython-313.pyc +0 -0
  82. claude_mpm/hooks/claude_hooks/__pycache__/hook_handler.cpython-313.pyc +0 -0
  83. claude_mpm/hooks/claude_hooks/__pycache__/memory_integration.cpython-313.pyc +0 -0
  84. claude_mpm/hooks/claude_hooks/__pycache__/response_tracking.cpython-313.pyc +0 -0
  85. claude_mpm/hooks/claude_hooks/__pycache__/tool_analysis.cpython-313.pyc +0 -0
  86. claude_mpm/hooks/claude_hooks/memory_integration.py +12 -1
  87. claude_mpm/hooks/claude_hooks/services/__pycache__/__init__.cpython-313.pyc +0 -0
  88. claude_mpm/hooks/claude_hooks/services/__pycache__/connection_manager_http.cpython-313.pyc +0 -0
  89. claude_mpm/hooks/claude_hooks/services/__pycache__/duplicate_detector.cpython-313.pyc +0 -0
  90. claude_mpm/hooks/claude_hooks/services/__pycache__/state_manager.cpython-313.pyc +0 -0
  91. claude_mpm/hooks/claude_hooks/services/__pycache__/subagent_processor.cpython-313.pyc +0 -0
  92. claude_mpm/hooks/failure_learning/__init__.py +2 -8
  93. claude_mpm/hooks/failure_learning/failure_detection_hook.py +1 -6
  94. claude_mpm/hooks/failure_learning/fix_detection_hook.py +1 -6
  95. claude_mpm/hooks/failure_learning/learning_extraction_hook.py +1 -6
  96. claude_mpm/hooks/kuzu_response_hook.py +1 -5
  97. claude_mpm/models/git_repository.py +198 -0
  98. claude_mpm/services/agents/agent_builder.py +45 -9
  99. claude_mpm/services/agents/agent_preset_service.py +238 -0
  100. claude_mpm/services/agents/agent_selection_service.py +484 -0
  101. claude_mpm/services/agents/auto_deploy_index_parser.py +569 -0
  102. claude_mpm/services/agents/cache_git_manager.py +621 -0
  103. claude_mpm/services/agents/deployment/agent_deployment.py +126 -2
  104. claude_mpm/services/agents/deployment/agent_discovery_service.py +105 -73
  105. claude_mpm/services/agents/deployment/agent_lifecycle_manager.py +1 -5
  106. claude_mpm/services/agents/deployment/agent_restore_handler.py +1 -4
  107. claude_mpm/services/agents/deployment/agent_template_builder.py +236 -15
  108. claude_mpm/services/agents/deployment/agents_directory_resolver.py +101 -15
  109. claude_mpm/services/agents/deployment/async_agent_deployment.py +2 -1
  110. claude_mpm/services/agents/deployment/multi_source_deployment_service.py +115 -15
  111. claude_mpm/services/agents/deployment/refactored_agent_deployment_service.py +1 -4
  112. claude_mpm/services/agents/deployment/remote_agent_discovery_service.py +363 -0
  113. claude_mpm/services/agents/deployment/single_agent_deployer.py +2 -2
  114. claude_mpm/services/agents/deployment/system_instructions_deployer.py +168 -46
  115. claude_mpm/services/agents/deployment/validation/deployment_validator.py +2 -2
  116. claude_mpm/services/agents/git_source_manager.py +629 -0
  117. claude_mpm/services/agents/loading/framework_agent_loader.py +1 -4
  118. claude_mpm/services/agents/local_template_manager.py +47 -9
  119. claude_mpm/services/agents/single_tier_deployment_service.py +696 -0
  120. claude_mpm/services/agents/sources/__init__.py +13 -0
  121. claude_mpm/services/agents/sources/agent_sync_state.py +516 -0
  122. claude_mpm/services/agents/sources/git_source_sync_service.py +1087 -0
  123. claude_mpm/services/agents/startup_sync.py +239 -0
  124. claude_mpm/services/agents/toolchain_detector.py +474 -0
  125. claude_mpm/services/analysis/__init__.py +25 -0
  126. claude_mpm/services/analysis/postmortem_reporter.py +474 -0
  127. claude_mpm/services/analysis/postmortem_service.py +765 -0
  128. claude_mpm/services/command_deployment_service.py +200 -6
  129. claude_mpm/services/core/base.py +7 -2
  130. claude_mpm/services/core/interfaces/__init__.py +1 -3
  131. claude_mpm/services/core/interfaces/health.py +1 -4
  132. claude_mpm/services/core/models/__init__.py +2 -11
  133. claude_mpm/services/diagnostics/checks/__init__.py +4 -0
  134. claude_mpm/services/diagnostics/checks/agent_sources_check.py +577 -0
  135. claude_mpm/services/diagnostics/checks/mcp_services_check.py +7 -15
  136. claude_mpm/services/diagnostics/checks/skill_sources_check.py +587 -0
  137. claude_mpm/services/diagnostics/diagnostic_runner.py +9 -0
  138. claude_mpm/services/diagnostics/doctor_reporter.py +34 -6
  139. claude_mpm/services/git/__init__.py +21 -0
  140. claude_mpm/services/git/git_operations_service.py +494 -0
  141. claude_mpm/services/github/__init__.py +21 -0
  142. claude_mpm/services/github/github_cli_service.py +397 -0
  143. claude_mpm/services/infrastructure/monitoring/__init__.py +1 -5
  144. claude_mpm/services/infrastructure/monitoring/aggregator.py +1 -6
  145. claude_mpm/services/instructions/__init__.py +9 -0
  146. claude_mpm/services/instructions/instruction_cache_service.py +374 -0
  147. claude_mpm/services/local_ops/__init__.py +3 -13
  148. claude_mpm/services/local_ops/health_checks/__init__.py +1 -3
  149. claude_mpm/services/local_ops/health_manager.py +1 -4
  150. claude_mpm/services/mcp_config_manager.py +75 -145
  151. claude_mpm/services/mcp_gateway/core/process_pool.py +22 -16
  152. claude_mpm/services/mcp_gateway/server/mcp_gateway.py +1 -6
  153. claude_mpm/services/mcp_service_verifier.py +6 -3
  154. claude_mpm/services/monitor/daemon.py +28 -8
  155. claude_mpm/services/monitor/daemon_manager.py +96 -19
  156. claude_mpm/services/pr/__init__.py +14 -0
  157. claude_mpm/services/pr/pr_template_service.py +329 -0
  158. claude_mpm/services/project/project_organizer.py +4 -0
  159. claude_mpm/services/runner_configuration_service.py +16 -3
  160. claude_mpm/services/session_management_service.py +16 -4
  161. claude_mpm/services/skills/__init__.py +18 -0
  162. claude_mpm/services/skills/git_skill_source_manager.py +1169 -0
  163. claude_mpm/services/skills/skill_discovery_service.py +568 -0
  164. claude_mpm/services/socketio/server/core.py +1 -4
  165. claude_mpm/services/socketio/server/main.py +1 -3
  166. claude_mpm/services/unified/deployment_strategies/vercel.py +1 -5
  167. claude_mpm/services/unified/unified_deployment.py +1 -5
  168. claude_mpm/services/visualization/__init__.py +1 -5
  169. claude_mpm/templates/questions/__init__.py +2 -7
  170. claude_mpm/templates/questions/pr_strategy.py +1 -4
  171. claude_mpm/templates/questions/project_init.py +1 -4
  172. claude_mpm/templates/questions/ticket_mgmt.py +1 -4
  173. claude_mpm/utils/agent_dependency_loader.py +77 -10
  174. claude_mpm/utils/agent_filters.py +288 -0
  175. claude_mpm/utils/gitignore.py +3 -0
  176. claude_mpm/utils/migration.py +372 -0
  177. claude_mpm/utils/progress.py +387 -0
  178. {claude_mpm-4.25.10.dist-info → claude_mpm-5.1.8.dist-info}/METADATA +356 -112
  179. {claude_mpm-4.25.10.dist-info → claude_mpm-5.1.8.dist-info}/RECORD +188 -439
  180. claude_mpm/agents/templates/agent-manager.json +0 -273
  181. claude_mpm/agents/templates/agentic-coder-optimizer.json +0 -248
  182. claude_mpm/agents/templates/api_qa.json +0 -183
  183. claude_mpm/agents/templates/clerk-ops.json +0 -235
  184. claude_mpm/agents/templates/code_analyzer.json +0 -101
  185. claude_mpm/agents/templates/content-agent.json +0 -358
  186. claude_mpm/agents/templates/dart_engineer.json +0 -307
  187. claude_mpm/agents/templates/data_engineer.json +0 -225
  188. claude_mpm/agents/templates/documentation.json +0 -238
  189. claude_mpm/agents/templates/engineer.json +0 -210
  190. claude_mpm/agents/templates/gcp_ops_agent.json +0 -253
  191. claude_mpm/agents/templates/golang_engineer.json +0 -270
  192. claude_mpm/agents/templates/imagemagick.json +0 -264
  193. claude_mpm/agents/templates/java_engineer.json +0 -346
  194. claude_mpm/agents/templates/javascript_engineer_agent.json +0 -380
  195. claude_mpm/agents/templates/local_ops_agent.json +0 -1840
  196. claude_mpm/agents/templates/memory_manager.json +0 -158
  197. claude_mpm/agents/templates/nextjs_engineer.json +0 -285
  198. claude_mpm/agents/templates/ops.json +0 -185
  199. claude_mpm/agents/templates/php-engineer.json +0 -287
  200. claude_mpm/agents/templates/product_owner.json +0 -338
  201. claude_mpm/agents/templates/project_organizer.json +0 -144
  202. claude_mpm/agents/templates/prompt-engineer.json +0 -737
  203. claude_mpm/agents/templates/python_engineer.json +0 -387
  204. claude_mpm/agents/templates/qa.json +0 -243
  205. claude_mpm/agents/templates/react_engineer.json +0 -239
  206. claude_mpm/agents/templates/refactoring_engineer.json +0 -276
  207. claude_mpm/agents/templates/research.json +0 -258
  208. claude_mpm/agents/templates/ruby-engineer.json +0 -280
  209. claude_mpm/agents/templates/rust_engineer.json +0 -275
  210. claude_mpm/agents/templates/security.json +0 -202
  211. claude_mpm/agents/templates/svelte-engineer.json +0 -225
  212. claude_mpm/agents/templates/tauri_engineer.json +0 -274
  213. claude_mpm/agents/templates/ticketing.json +0 -181
  214. claude_mpm/agents/templates/typescript_engineer.json +0 -285
  215. claude_mpm/agents/templates/vercel_ops_agent.json +0 -412
  216. claude_mpm/agents/templates/version_control.json +0 -159
  217. claude_mpm/agents/templates/web_qa.json +0 -400
  218. claude_mpm/agents/templates/web_ui.json +0 -189
  219. claude_mpm/cli/README.md +0 -253
  220. claude_mpm/cli/commands/mcp_install_commands.py.backup +0 -284
  221. claude_mpm/cli/commands/mpm_init/README.md +0 -365
  222. claude_mpm/cli_module/refactoring_guide.md +0 -253
  223. claude_mpm/commands/mpm-tickets.md +0 -151
  224. claude_mpm/config/agent_capabilities.yaml +0 -658
  225. claude_mpm/config/async_logging_config.yaml +0 -145
  226. claude_mpm/core/.claude-mpm/logs/hooks_20250730.log +0 -34
  227. claude_mpm/d2/.gitignore +0 -22
  228. claude_mpm/d2/ARCHITECTURE_COMPARISON.md +0 -273
  229. claude_mpm/d2/FLASK_INTEGRATION.md +0 -156
  230. claude_mpm/d2/IMPLEMENTATION_SUMMARY.md +0 -452
  231. claude_mpm/d2/QUICKSTART.md +0 -186
  232. claude_mpm/d2/README.md +0 -232
  233. claude_mpm/d2/STORE_FIX_SUMMARY.md +0 -167
  234. claude_mpm/d2/SVELTE5_STORES_GUIDE.md +0 -180
  235. claude_mpm/d2/TESTING.md +0 -288
  236. claude_mpm/d2/index.html +0 -118
  237. claude_mpm/d2/package.json +0 -19
  238. claude_mpm/d2/src/App.svelte +0 -110
  239. claude_mpm/d2/src/components/Header.svelte +0 -153
  240. claude_mpm/d2/src/components/MainContent.svelte +0 -74
  241. claude_mpm/d2/src/components/Sidebar.svelte +0 -85
  242. claude_mpm/d2/src/components/tabs/EventsTab.svelte +0 -326
  243. claude_mpm/d2/src/lib/socketio.js +0 -144
  244. claude_mpm/d2/src/main.js +0 -7
  245. claude_mpm/d2/src/stores/events.js +0 -114
  246. claude_mpm/d2/src/stores/socket.js +0 -108
  247. claude_mpm/d2/src/stores/theme.js +0 -65
  248. claude_mpm/d2/svelte.config.js +0 -12
  249. claude_mpm/d2/vite.config.js +0 -15
  250. claude_mpm/dashboard/.claude-mpm/memories/README.md +0 -36
  251. claude_mpm/dashboard/BUILD_NUMBER +0 -1
  252. claude_mpm/dashboard/README.md +0 -121
  253. claude_mpm/dashboard/VERSION +0 -1
  254. claude_mpm/dashboard/react/components/DataInspector/DataInspector.module.css +0 -188
  255. claude_mpm/dashboard/react/components/DataInspector/DataInspector.tsx +0 -273
  256. claude_mpm/dashboard/react/components/ErrorBoundary.tsx +0 -75
  257. claude_mpm/dashboard/react/components/EventViewer/EventViewer.module.css +0 -156
  258. claude_mpm/dashboard/react/components/EventViewer/EventViewer.tsx +0 -141
  259. claude_mpm/dashboard/react/components/shared/ConnectionStatus.module.css +0 -38
  260. claude_mpm/dashboard/react/components/shared/ConnectionStatus.tsx +0 -36
  261. claude_mpm/dashboard/react/components/shared/FilterBar.module.css +0 -92
  262. claude_mpm/dashboard/react/components/shared/FilterBar.tsx +0 -89
  263. claude_mpm/dashboard/react/contexts/DashboardContext.tsx +0 -215
  264. claude_mpm/dashboard/react/entries/events.tsx +0 -165
  265. claude_mpm/dashboard/react/hooks/useEvents.ts +0 -191
  266. claude_mpm/dashboard/react/hooks/useSocket.ts +0 -225
  267. claude_mpm/dashboard/static/archive/activity_dashboard_fixed.html +0 -248
  268. claude_mpm/dashboard/static/built/REFACTORING_SUMMARY.md +0 -170
  269. claude_mpm/dashboard/static/built/assets/events.DjpNxWNo.css +0 -1
  270. claude_mpm/dashboard/static/built/components/activity-tree.js +0 -2
  271. claude_mpm/dashboard/static/built/components/activity-tree.js.map +0 -1
  272. claude_mpm/dashboard/static/built/components/agent-hierarchy.js +0 -777
  273. claude_mpm/dashboard/static/built/components/agent-inference.js +0 -2
  274. claude_mpm/dashboard/static/built/components/agent-inference.js.map +0 -1
  275. claude_mpm/dashboard/static/built/components/build-tracker.js +0 -333
  276. claude_mpm/dashboard/static/built/components/code-simple.js +0 -857
  277. claude_mpm/dashboard/static/built/components/code-tree/tree-breadcrumb.js +0 -353
  278. claude_mpm/dashboard/static/built/components/code-tree/tree-constants.js +0 -235
  279. claude_mpm/dashboard/static/built/components/code-tree/tree-search.js +0 -409
  280. claude_mpm/dashboard/static/built/components/code-tree/tree-utils.js +0 -435
  281. claude_mpm/dashboard/static/built/components/code-tree.js +0 -2
  282. claude_mpm/dashboard/static/built/components/code-tree.js.map +0 -1
  283. claude_mpm/dashboard/static/built/components/code-viewer.js +0 -2
  284. claude_mpm/dashboard/static/built/components/code-viewer.js.map +0 -1
  285. claude_mpm/dashboard/static/built/components/connection-debug.js +0 -654
  286. claude_mpm/dashboard/static/built/components/diff-viewer.js +0 -891
  287. claude_mpm/dashboard/static/built/components/event-processor.js +0 -2
  288. claude_mpm/dashboard/static/built/components/event-processor.js.map +0 -1
  289. claude_mpm/dashboard/static/built/components/event-viewer.js +0 -2
  290. claude_mpm/dashboard/static/built/components/event-viewer.js.map +0 -1
  291. claude_mpm/dashboard/static/built/components/export-manager.js +0 -2
  292. claude_mpm/dashboard/static/built/components/export-manager.js.map +0 -1
  293. claude_mpm/dashboard/static/built/components/file-change-tracker.js +0 -443
  294. claude_mpm/dashboard/static/built/components/file-change-viewer.js +0 -690
  295. claude_mpm/dashboard/static/built/components/file-tool-tracker.js +0 -2
  296. claude_mpm/dashboard/static/built/components/file-tool-tracker.js.map +0 -1
  297. claude_mpm/dashboard/static/built/components/file-viewer.js +0 -2
  298. claude_mpm/dashboard/static/built/components/file-viewer.js.map +0 -1
  299. claude_mpm/dashboard/static/built/components/hud-library-loader.js +0 -2
  300. claude_mpm/dashboard/static/built/components/hud-library-loader.js.map +0 -1
  301. claude_mpm/dashboard/static/built/components/hud-manager.js +0 -2
  302. claude_mpm/dashboard/static/built/components/hud-manager.js.map +0 -1
  303. claude_mpm/dashboard/static/built/components/hud-visualizer.js +0 -2
  304. claude_mpm/dashboard/static/built/components/hud-visualizer.js.map +0 -1
  305. claude_mpm/dashboard/static/built/components/module-viewer.js +0 -2
  306. claude_mpm/dashboard/static/built/components/module-viewer.js.map +0 -1
  307. claude_mpm/dashboard/static/built/components/nav-bar.js +0 -145
  308. claude_mpm/dashboard/static/built/components/page-structure.js +0 -429
  309. claude_mpm/dashboard/static/built/components/session-manager.js +0 -2
  310. claude_mpm/dashboard/static/built/components/session-manager.js.map +0 -1
  311. claude_mpm/dashboard/static/built/components/socket-manager.js +0 -2
  312. claude_mpm/dashboard/static/built/components/socket-manager.js.map +0 -1
  313. claude_mpm/dashboard/static/built/components/ui-state-manager.js +0 -2
  314. claude_mpm/dashboard/static/built/components/ui-state-manager.js.map +0 -1
  315. claude_mpm/dashboard/static/built/components/unified-data-viewer.js +0 -2
  316. claude_mpm/dashboard/static/built/components/unified-data-viewer.js.map +0 -1
  317. claude_mpm/dashboard/static/built/components/working-directory.js +0 -2
  318. claude_mpm/dashboard/static/built/components/working-directory.js.map +0 -1
  319. claude_mpm/dashboard/static/built/connection-manager.js +0 -536
  320. claude_mpm/dashboard/static/built/dashboard.js +0 -2
  321. claude_mpm/dashboard/static/built/dashboard.js.map +0 -1
  322. claude_mpm/dashboard/static/built/extension-error-handler.js +0 -164
  323. claude_mpm/dashboard/static/built/react/events.js +0 -30
  324. claude_mpm/dashboard/static/built/react/events.js.map +0 -1
  325. claude_mpm/dashboard/static/built/shared/dom-helpers.js +0 -396
  326. claude_mpm/dashboard/static/built/shared/event-bus.js +0 -330
  327. claude_mpm/dashboard/static/built/shared/event-filter-service.js +0 -540
  328. claude_mpm/dashboard/static/built/shared/logger.js +0 -385
  329. claude_mpm/dashboard/static/built/shared/page-structure.js +0 -249
  330. claude_mpm/dashboard/static/built/shared/tooltip-service.js +0 -253
  331. claude_mpm/dashboard/static/built/socket-client.js +0 -2
  332. claude_mpm/dashboard/static/built/socket-client.js.map +0 -1
  333. claude_mpm/dashboard/static/built/tab-isolation-fix.js +0 -185
  334. claude_mpm/dashboard/static/events.html +0 -607
  335. claude_mpm/dashboard/static/index.html +0 -635
  336. claude_mpm/dashboard/static/js/REFACTORING_SUMMARY.md +0 -170
  337. claude_mpm/dashboard/static/js/shared/dom-helpers.js +0 -396
  338. claude_mpm/dashboard/static/js/shared/event-bus.js +0 -330
  339. claude_mpm/dashboard/static/js/shared/logger.js +0 -385
  340. claude_mpm/dashboard/static/js/shared/tooltip-service.js +0 -253
  341. claude_mpm/dashboard/static/js/stores/dashboard-store.js +0 -562
  342. claude_mpm/dashboard/static/legacy/activity.html +0 -736
  343. claude_mpm/dashboard/static/legacy/agents.html +0 -786
  344. claude_mpm/dashboard/static/legacy/files.html +0 -747
  345. claude_mpm/dashboard/static/legacy/tools.html +0 -831
  346. claude_mpm/dashboard/static/monitors.html +0 -431
  347. claude_mpm/dashboard/static/navigation-test-results.md +0 -118
  348. claude_mpm/dashboard/static/production/events.html +0 -659
  349. claude_mpm/dashboard/static/production/main.html +0 -698
  350. claude_mpm/dashboard/static/production/monitors.html +0 -483
  351. claude_mpm/dashboard/static/test-archive/dashboard.html +0 -635
  352. claude_mpm/dashboard/static/test-archive/debug-events.html +0 -147
  353. claude_mpm/dashboard/static/test-archive/test-navigation.html +0 -256
  354. claude_mpm/dashboard/static/test-archive/test-react-exports.html +0 -180
  355. claude_mpm/dashboard/templates/.claude-mpm/memories/README.md +0 -36
  356. claude_mpm/dashboard/templates/.claude-mpm/memories/engineer_agent.md +0 -39
  357. claude_mpm/dashboard/templates/.claude-mpm/memories/version_control_agent.md +0 -38
  358. claude_mpm/hooks/README.md +0 -143
  359. claude_mpm/hooks/templates/README.md +0 -180
  360. claude_mpm/hooks/templates/settings.json.example +0 -147
  361. claude_mpm/schemas/agent_schema.json +0 -596
  362. claude_mpm/schemas/frontmatter_schema.json +0 -165
  363. claude_mpm/services/event_bus/README.md +0 -244
  364. claude_mpm/services/events/README.md +0 -303
  365. claude_mpm/services/framework_claude_md_generator/README.md +0 -119
  366. claude_mpm/services/mcp_gateway/README.md +0 -185
  367. claude_mpm/services/socketio/handlers/connection.py.backup +0 -217
  368. claude_mpm/services/socketio/handlers/hook.py.backup +0 -154
  369. claude_mpm/services/static/.gitkeep +0 -2
  370. claude_mpm/services/version_control/VERSION +0 -1
  371. claude_mpm/skills/bundled/.gitkeep +0 -2
  372. claude_mpm/skills/bundled/collaboration/brainstorming/SKILL.md +0 -79
  373. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/SKILL.md +0 -178
  374. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/agent-prompts.md +0 -577
  375. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/coordination-patterns.md +0 -467
  376. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/examples.md +0 -537
  377. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/troubleshooting.md +0 -730
  378. claude_mpm/skills/bundled/collaboration/git-worktrees.md +0 -317
  379. claude_mpm/skills/bundled/collaboration/requesting-code-review/SKILL.md +0 -112
  380. claude_mpm/skills/bundled/collaboration/requesting-code-review/references/code-reviewer-template.md +0 -146
  381. claude_mpm/skills/bundled/collaboration/requesting-code-review/references/review-examples.md +0 -412
  382. claude_mpm/skills/bundled/collaboration/stacked-prs.md +0 -251
  383. claude_mpm/skills/bundled/collaboration/writing-plans/SKILL.md +0 -81
  384. claude_mpm/skills/bundled/collaboration/writing-plans/references/best-practices.md +0 -362
  385. claude_mpm/skills/bundled/collaboration/writing-plans/references/plan-structure-templates.md +0 -312
  386. claude_mpm/skills/bundled/debugging/root-cause-tracing/SKILL.md +0 -152
  387. claude_mpm/skills/bundled/debugging/root-cause-tracing/find-polluter.sh +0 -63
  388. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/advanced-techniques.md +0 -668
  389. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/examples.md +0 -587
  390. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/integration.md +0 -438
  391. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/tracing-techniques.md +0 -391
  392. claude_mpm/skills/bundled/debugging/systematic-debugging/CREATION-LOG.md +0 -119
  393. claude_mpm/skills/bundled/debugging/systematic-debugging/SKILL.md +0 -148
  394. claude_mpm/skills/bundled/debugging/systematic-debugging/references/anti-patterns.md +0 -483
  395. claude_mpm/skills/bundled/debugging/systematic-debugging/references/examples.md +0 -452
  396. claude_mpm/skills/bundled/debugging/systematic-debugging/references/troubleshooting.md +0 -449
  397. claude_mpm/skills/bundled/debugging/systematic-debugging/references/workflow.md +0 -411
  398. claude_mpm/skills/bundled/debugging/systematic-debugging/test-academic.md +0 -14
  399. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-1.md +0 -58
  400. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-2.md +0 -68
  401. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-3.md +0 -69
  402. claude_mpm/skills/bundled/debugging/verification-before-completion/SKILL.md +0 -131
  403. claude_mpm/skills/bundled/debugging/verification-before-completion/references/gate-function.md +0 -325
  404. claude_mpm/skills/bundled/debugging/verification-before-completion/references/integration-and-workflows.md +0 -490
  405. claude_mpm/skills/bundled/debugging/verification-before-completion/references/red-flags-and-failures.md +0 -425
  406. claude_mpm/skills/bundled/debugging/verification-before-completion/references/verification-patterns.md +0 -499
  407. claude_mpm/skills/bundled/infrastructure/env-manager/INTEGRATION.md +0 -611
  408. claude_mpm/skills/bundled/infrastructure/env-manager/README.md +0 -596
  409. claude_mpm/skills/bundled/infrastructure/env-manager/SKILL.md +0 -260
  410. claude_mpm/skills/bundled/infrastructure/env-manager/examples/nextjs-env-structure.md +0 -315
  411. claude_mpm/skills/bundled/infrastructure/env-manager/references/frameworks.md +0 -436
  412. claude_mpm/skills/bundled/infrastructure/env-manager/references/security.md +0 -433
  413. claude_mpm/skills/bundled/infrastructure/env-manager/references/synchronization.md +0 -452
  414. claude_mpm/skills/bundled/infrastructure/env-manager/references/troubleshooting.md +0 -404
  415. claude_mpm/skills/bundled/infrastructure/env-manager/references/validation.md +0 -420
  416. claude_mpm/skills/bundled/main/artifacts-builder/LICENSE.txt +0 -202
  417. claude_mpm/skills/bundled/main/artifacts-builder/SKILL.md +0 -86
  418. claude_mpm/skills/bundled/main/artifacts-builder/scripts/bundle-artifact.sh +0 -54
  419. claude_mpm/skills/bundled/main/artifacts-builder/scripts/init-artifact.sh +0 -322
  420. claude_mpm/skills/bundled/main/artifacts-builder/scripts/shadcn-components.tar.gz +0 -0
  421. claude_mpm/skills/bundled/main/internal-comms/LICENSE.txt +0 -202
  422. claude_mpm/skills/bundled/main/internal-comms/SKILL.md +0 -43
  423. claude_mpm/skills/bundled/main/internal-comms/examples/3p-updates.md +0 -47
  424. claude_mpm/skills/bundled/main/internal-comms/examples/company-newsletter.md +0 -65
  425. claude_mpm/skills/bundled/main/internal-comms/examples/faq-answers.md +0 -30
  426. claude_mpm/skills/bundled/main/internal-comms/examples/general-comms.md +0 -16
  427. claude_mpm/skills/bundled/main/mcp-builder/LICENSE.txt +0 -202
  428. claude_mpm/skills/bundled/main/mcp-builder/SKILL.md +0 -160
  429. claude_mpm/skills/bundled/main/mcp-builder/reference/design_principles.md +0 -412
  430. claude_mpm/skills/bundled/main/mcp-builder/reference/evaluation.md +0 -602
  431. claude_mpm/skills/bundled/main/mcp-builder/reference/mcp_best_practices.md +0 -915
  432. claude_mpm/skills/bundled/main/mcp-builder/reference/node_mcp_server.md +0 -916
  433. claude_mpm/skills/bundled/main/mcp-builder/reference/python_mcp_server.md +0 -752
  434. claude_mpm/skills/bundled/main/mcp-builder/reference/workflow.md +0 -1237
  435. claude_mpm/skills/bundled/main/mcp-builder/scripts/example_evaluation.xml +0 -22
  436. claude_mpm/skills/bundled/main/mcp-builder/scripts/requirements.txt +0 -2
  437. claude_mpm/skills/bundled/main/skill-creator/LICENSE.txt +0 -202
  438. claude_mpm/skills/bundled/main/skill-creator/SKILL.md +0 -189
  439. claude_mpm/skills/bundled/main/skill-creator/references/best-practices.md +0 -500
  440. claude_mpm/skills/bundled/main/skill-creator/references/creation-workflow.md +0 -464
  441. claude_mpm/skills/bundled/main/skill-creator/references/examples.md +0 -619
  442. claude_mpm/skills/bundled/main/skill-creator/references/progressive-disclosure.md +0 -437
  443. claude_mpm/skills/bundled/main/skill-creator/references/skill-structure.md +0 -231
  444. claude_mpm/skills/bundled/php/espocrm-development/SKILL.md +0 -170
  445. claude_mpm/skills/bundled/php/espocrm-development/references/architecture.md +0 -602
  446. claude_mpm/skills/bundled/php/espocrm-development/references/common-tasks.md +0 -821
  447. claude_mpm/skills/bundled/php/espocrm-development/references/development-workflow.md +0 -742
  448. claude_mpm/skills/bundled/php/espocrm-development/references/frontend-customization.md +0 -726
  449. claude_mpm/skills/bundled/php/espocrm-development/references/hooks-and-services.md +0 -764
  450. claude_mpm/skills/bundled/php/espocrm-development/references/testing-debugging.md +0 -831
  451. claude_mpm/skills/bundled/react/flexlayout-react.md +0 -742
  452. claude_mpm/skills/bundled/rust/desktop-applications/SKILL.md +0 -226
  453. claude_mpm/skills/bundled/rust/desktop-applications/references/architecture-patterns.md +0 -901
  454. claude_mpm/skills/bundled/rust/desktop-applications/references/native-gui-frameworks.md +0 -901
  455. claude_mpm/skills/bundled/rust/desktop-applications/references/platform-integration.md +0 -775
  456. claude_mpm/skills/bundled/rust/desktop-applications/references/state-management.md +0 -937
  457. claude_mpm/skills/bundled/rust/desktop-applications/references/tauri-framework.md +0 -770
  458. claude_mpm/skills/bundled/rust/desktop-applications/references/testing-deployment.md +0 -961
  459. claude_mpm/skills/bundled/tauri/tauri-async-patterns.md +0 -495
  460. claude_mpm/skills/bundled/tauri/tauri-build-deploy.md +0 -599
  461. claude_mpm/skills/bundled/tauri/tauri-command-patterns.md +0 -535
  462. claude_mpm/skills/bundled/tauri/tauri-error-handling.md +0 -613
  463. claude_mpm/skills/bundled/tauri/tauri-event-system.md +0 -648
  464. claude_mpm/skills/bundled/tauri/tauri-file-system.md +0 -673
  465. claude_mpm/skills/bundled/tauri/tauri-frontend-integration.md +0 -767
  466. claude_mpm/skills/bundled/tauri/tauri-performance.md +0 -669
  467. claude_mpm/skills/bundled/tauri/tauri-state-management.md +0 -573
  468. claude_mpm/skills/bundled/tauri/tauri-testing.md +0 -384
  469. claude_mpm/skills/bundled/tauri/tauri-window-management.md +0 -628
  470. claude_mpm/skills/bundled/testing/condition-based-waiting/SKILL.md +0 -119
  471. claude_mpm/skills/bundled/testing/condition-based-waiting/example.ts +0 -158
  472. claude_mpm/skills/bundled/testing/condition-based-waiting/references/patterns-and-implementation.md +0 -253
  473. claude_mpm/skills/bundled/testing/test-driven-development/SKILL.md +0 -145
  474. claude_mpm/skills/bundled/testing/test-driven-development/references/anti-patterns.md +0 -543
  475. claude_mpm/skills/bundled/testing/test-driven-development/references/examples.md +0 -741
  476. claude_mpm/skills/bundled/testing/test-driven-development/references/integration.md +0 -470
  477. claude_mpm/skills/bundled/testing/test-driven-development/references/philosophy.md +0 -458
  478. claude_mpm/skills/bundled/testing/test-driven-development/references/workflow.md +0 -639
  479. claude_mpm/skills/bundled/testing/test-quality-inspector/SKILL.md +0 -458
  480. claude_mpm/skills/bundled/testing/test-quality-inspector/examples/example-inspection-report.md +0 -411
  481. claude_mpm/skills/bundled/testing/test-quality-inspector/references/assertion-quality.md +0 -317
  482. claude_mpm/skills/bundled/testing/test-quality-inspector/references/inspection-checklist.md +0 -270
  483. claude_mpm/skills/bundled/testing/test-quality-inspector/references/red-flags.md +0 -436
  484. claude_mpm/skills/bundled/testing/testing-anti-patterns/SKILL.md +0 -140
  485. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/completeness-anti-patterns.md +0 -572
  486. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/core-anti-patterns.md +0 -411
  487. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/detection-guide.md +0 -569
  488. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/tdd-connection.md +0 -695
  489. claude_mpm/skills/bundled/testing/webapp-testing/LICENSE.txt +0 -202
  490. claude_mpm/skills/bundled/testing/webapp-testing/SKILL.md +0 -184
  491. claude_mpm/skills/bundled/testing/webapp-testing/decision-tree.md +0 -459
  492. claude_mpm/skills/bundled/testing/webapp-testing/playwright-patterns.md +0 -479
  493. claude_mpm/skills/bundled/testing/webapp-testing/reconnaissance-pattern.md +0 -687
  494. claude_mpm/skills/bundled/testing/webapp-testing/server-management.md +0 -758
  495. claude_mpm/skills/bundled/testing/webapp-testing/troubleshooting.md +0 -868
  496. claude_mpm/templates/questions/EXAMPLES.md +0 -501
  497. claude_mpm/tools/README_SOCKETIO_DEBUG.md +0 -224
  498. claude_mpm/tools/code_tree_analyzer/README.md +0 -64
  499. /claude_mpm/agents/templates/{git_file_tracking.md → git-file-tracking.md} +0 -0
  500. /claude_mpm/agents/templates/{pm_examples.md → pm-examples.md} +0 -0
  501. /claude_mpm/agents/templates/{pm_red_flags.md → pm-red-flags.md} +0 -0
  502. /claude_mpm/agents/templates/{response_format.md → response-format.md} +0 -0
  503. /claude_mpm/agents/templates/{validation_templates.md → validation-templates.md} +0 -0
  504. {claude_mpm-4.25.10.dist-info → claude_mpm-5.1.8.dist-info}/WHEEL +0 -0
  505. {claude_mpm-4.25.10.dist-info → claude_mpm-5.1.8.dist-info}/entry_points.txt +0 -0
  506. {claude_mpm-4.25.10.dist-info → claude_mpm-5.1.8.dist-info}/licenses/LICENSE +0 -0
  507. {claude_mpm-4.25.10.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
 
@@ -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:
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:
181
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
- _, 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:
413
+ continue
414
+ elif choice_num == len(all_agents) + 4:
193
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:
418
+ continue
419
+ elif choice_num == len(all_agents) + 5:
199
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.