claude-mpm 4.1.26__py3-none-any.whl → 5.0.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 (792) hide show
  1. claude_mpm/BUILD_NUMBER +1 -1
  2. claude_mpm/VERSION +1 -1
  3. claude_mpm/__init__.py +20 -5
  4. claude_mpm/agents/BASE_AGENT_TEMPLATE.md +118 -0
  5. claude_mpm/agents/BASE_DOCUMENTATION.md +53 -0
  6. claude_mpm/agents/BASE_ENGINEER.md +658 -0
  7. claude_mpm/agents/BASE_OPS.md +219 -0
  8. claude_mpm/agents/BASE_PM.md +432 -158
  9. claude_mpm/agents/BASE_PROMPT_ENGINEER.md +787 -0
  10. claude_mpm/agents/BASE_QA.md +167 -0
  11. claude_mpm/agents/BASE_RESEARCH.md +53 -0
  12. claude_mpm/agents/OUTPUT_STYLE.md +254 -29
  13. claude_mpm/agents/PM_INSTRUCTIONS.md +969 -0
  14. claude_mpm/agents/PM_INSTRUCTIONS_TEACH.md +1322 -0
  15. claude_mpm/agents/WORKFLOW.md +355 -191
  16. claude_mpm/agents/__init__.py +6 -0
  17. claude_mpm/agents/agent_loader.py +41 -14
  18. claude_mpm/agents/agent_loader_integration.py +3 -2
  19. claude_mpm/agents/async_agent_loader.py +3 -3
  20. claude_mpm/agents/base_agent.json +6 -3
  21. claude_mpm/agents/base_agent_loader.py +21 -44
  22. claude_mpm/agents/frontmatter_validator.py +292 -252
  23. claude_mpm/agents/system_agent_config.py +3 -2
  24. claude_mpm/agents/templates/README.md +465 -0
  25. claude_mpm/agents/templates/circuit-breakers.md +1005 -0
  26. claude_mpm/agents/templates/context-management-examples.md +544 -0
  27. claude_mpm/agents/templates/git-file-tracking.md +584 -0
  28. claude_mpm/agents/templates/pm-examples.md +474 -0
  29. claude_mpm/agents/templates/pm-red-flags.md +310 -0
  30. claude_mpm/agents/templates/pr-workflow-examples.md +427 -0
  31. claude_mpm/agents/templates/research-gate-examples.md +669 -0
  32. claude_mpm/agents/templates/response-format.md +583 -0
  33. claude_mpm/agents/templates/structured-questions-examples.md +615 -0
  34. claude_mpm/agents/templates/ticket-completeness-examples.md +139 -0
  35. claude_mpm/agents/templates/ticketing-examples.md +277 -0
  36. claude_mpm/agents/templates/validation-templates.md +312 -0
  37. claude_mpm/cli/__init__.py +72 -376
  38. claude_mpm/cli/commands/__init__.py +4 -0
  39. claude_mpm/cli/commands/agent_manager.py +675 -20
  40. claude_mpm/cli/commands/agent_source.py +774 -0
  41. claude_mpm/cli/commands/agent_state_manager.py +344 -0
  42. claude_mpm/cli/commands/agents.py +1673 -178
  43. claude_mpm/cli/commands/agents_cleanup.py +210 -0
  44. claude_mpm/cli/commands/agents_detect.py +380 -0
  45. claude_mpm/cli/commands/agents_discover.py +338 -0
  46. claude_mpm/cli/commands/agents_recommend.py +309 -0
  47. claude_mpm/cli/commands/aggregate.py +11 -7
  48. claude_mpm/cli/commands/analyze.py +18 -13
  49. claude_mpm/cli/commands/analyze_code.py +8 -4
  50. claude_mpm/cli/commands/auto_configure.py +566 -0
  51. claude_mpm/cli/commands/cleanup.py +12 -12
  52. claude_mpm/cli/commands/config.py +54 -17
  53. claude_mpm/cli/commands/configure.py +1184 -1055
  54. claude_mpm/cli/commands/configure_agent_display.py +261 -0
  55. claude_mpm/cli/commands/configure_behavior_manager.py +204 -0
  56. claude_mpm/cli/commands/configure_hook_manager.py +225 -0
  57. claude_mpm/cli/commands/configure_models.py +18 -0
  58. claude_mpm/cli/commands/configure_navigation.py +184 -0
  59. claude_mpm/cli/commands/configure_paths.py +104 -0
  60. claude_mpm/cli/commands/configure_persistence.py +254 -0
  61. claude_mpm/cli/commands/configure_startup_manager.py +646 -0
  62. claude_mpm/cli/commands/configure_template_editor.py +497 -0
  63. claude_mpm/cli/commands/configure_validators.py +73 -0
  64. claude_mpm/cli/commands/dashboard.py +50 -52
  65. claude_mpm/cli/commands/debug.py +19 -19
  66. claude_mpm/cli/commands/doctor.py +51 -7
  67. claude_mpm/cli/commands/hook_errors.py +277 -0
  68. claude_mpm/cli/commands/info.py +3 -4
  69. claude_mpm/cli/commands/local_deploy.py +534 -0
  70. claude_mpm/cli/commands/mcp.py +17 -10
  71. claude_mpm/cli/commands/mcp_command_router.py +11 -0
  72. claude_mpm/cli/commands/mcp_config.py +154 -0
  73. claude_mpm/cli/commands/mcp_external_commands.py +249 -0
  74. claude_mpm/cli/commands/mcp_install_commands.py +101 -32
  75. claude_mpm/cli/commands/mcp_pipx_config.py +2 -2
  76. claude_mpm/cli/commands/mcp_setup_external.py +868 -0
  77. claude_mpm/cli/commands/memory.py +55 -21
  78. claude_mpm/cli/commands/monitor.py +160 -70
  79. claude_mpm/cli/commands/mpm_init/__init__.py +73 -0
  80. claude_mpm/cli/commands/mpm_init/core.py +573 -0
  81. claude_mpm/cli/commands/mpm_init/display.py +341 -0
  82. claude_mpm/cli/commands/mpm_init/git_activity.py +427 -0
  83. claude_mpm/cli/commands/mpm_init/modes.py +397 -0
  84. claude_mpm/cli/commands/mpm_init/prompts.py +442 -0
  85. claude_mpm/cli/commands/mpm_init_cli.py +396 -0
  86. claude_mpm/cli/commands/mpm_init_handler.py +114 -4
  87. claude_mpm/cli/commands/postmortem.py +401 -0
  88. claude_mpm/cli/commands/run.py +252 -167
  89. claude_mpm/cli/commands/search.py +458 -0
  90. claude_mpm/cli/commands/skill_source.py +694 -0
  91. claude_mpm/cli/commands/skills.py +1225 -0
  92. claude_mpm/cli/commands/uninstall.py +176 -0
  93. claude_mpm/cli/commands/upgrade.py +152 -0
  94. claude_mpm/cli/commands/verify.py +119 -0
  95. claude_mpm/cli/executor.py +279 -0
  96. claude_mpm/cli/helpers.py +105 -0
  97. claude_mpm/cli/interactive/__init__.py +21 -0
  98. claude_mpm/cli/interactive/agent_wizard.py +1872 -0
  99. claude_mpm/cli/interactive/skills_wizard.py +491 -0
  100. claude_mpm/cli/parser.py +79 -2
  101. claude_mpm/cli/parsers/__init__.py +7 -1
  102. claude_mpm/cli/parsers/agent_manager_parser.py +161 -1
  103. claude_mpm/cli/parsers/agent_source_parser.py +171 -0
  104. claude_mpm/cli/parsers/agents_parser.py +369 -1
  105. claude_mpm/cli/parsers/auto_configure_parser.py +245 -0
  106. claude_mpm/cli/parsers/base_parser.py +196 -3
  107. claude_mpm/cli/parsers/config_parser.py +96 -43
  108. claude_mpm/cli/parsers/configure_parser.py +11 -15
  109. claude_mpm/cli/parsers/local_deploy_parser.py +227 -0
  110. claude_mpm/cli/parsers/mcp_parser.py +15 -0
  111. claude_mpm/cli/parsers/monitor_parser.py +12 -2
  112. claude_mpm/cli/parsers/mpm_init_parser.py +179 -9
  113. claude_mpm/cli/parsers/run_parser.py +5 -0
  114. claude_mpm/cli/parsers/search_parser.py +245 -0
  115. claude_mpm/cli/parsers/skill_source_parser.py +169 -0
  116. claude_mpm/cli/parsers/skills_parser.py +282 -0
  117. claude_mpm/cli/parsers/source_parser.py +138 -0
  118. claude_mpm/cli/shared/argument_patterns.py +20 -13
  119. claude_mpm/cli/shared/base_command.py +2 -2
  120. claude_mpm/cli/shared/output_formatters.py +28 -19
  121. claude_mpm/cli/startup.py +994 -0
  122. claude_mpm/cli/startup_display.py +480 -0
  123. claude_mpm/cli/startup_logging.py +179 -13
  124. claude_mpm/cli/utils.py +54 -3
  125. claude_mpm/cli_module/commands.py +1 -1
  126. claude_mpm/commands/mpm-agents-auto-configure.md +278 -0
  127. claude_mpm/commands/mpm-agents-detect.md +177 -0
  128. claude_mpm/commands/mpm-agents-list.md +131 -0
  129. claude_mpm/commands/mpm-agents-recommend.md +223 -0
  130. claude_mpm/commands/mpm-config-view.md +150 -0
  131. claude_mpm/commands/mpm-doctor.md +9 -0
  132. claude_mpm/commands/mpm-help.md +297 -5
  133. claude_mpm/commands/mpm-init.md +401 -17
  134. claude_mpm/commands/mpm-monitor.md +418 -0
  135. claude_mpm/commands/mpm-postmortem.md +123 -0
  136. claude_mpm/commands/mpm-session-resume.md +381 -0
  137. claude_mpm/commands/mpm-status.md +79 -8
  138. claude_mpm/commands/mpm-ticket-organize.md +304 -0
  139. claude_mpm/commands/mpm-ticket-view.md +552 -0
  140. claude_mpm/commands/mpm-version.md +122 -0
  141. claude_mpm/commands/mpm.md +12 -0
  142. claude_mpm/config/agent_config.py +4 -4
  143. claude_mpm/config/agent_presets.py +488 -0
  144. claude_mpm/config/agent_sources.py +325 -0
  145. claude_mpm/config/experimental_features.py +7 -7
  146. claude_mpm/config/model_config.py +428 -0
  147. claude_mpm/config/paths.py +3 -2
  148. claude_mpm/config/skill_presets.py +392 -0
  149. claude_mpm/config/skill_sources.py +590 -0
  150. claude_mpm/config/socketio_config.py +3 -3
  151. claude_mpm/constants.py +28 -1
  152. claude_mpm/core/__init__.py +53 -17
  153. claude_mpm/core/agent_name_normalizer.py +3 -2
  154. claude_mpm/core/agent_registry.py +2 -2
  155. claude_mpm/core/agent_session_manager.py +10 -10
  156. claude_mpm/core/api_validator.py +330 -0
  157. claude_mpm/core/base_service.py +33 -23
  158. claude_mpm/core/cache.py +9 -9
  159. claude_mpm/core/claude_runner.py +24 -42
  160. claude_mpm/core/config.py +101 -8
  161. claude_mpm/core/config_aliases.py +7 -6
  162. claude_mpm/core/constants.py +66 -1
  163. claude_mpm/core/container.py +11 -5
  164. claude_mpm/core/enums.py +452 -0
  165. claude_mpm/core/error_handler.py +623 -0
  166. claude_mpm/core/factories.py +1 -1
  167. claude_mpm/core/file_utils.py +764 -0
  168. claude_mpm/core/framework/__init__.py +25 -0
  169. claude_mpm/core/framework/formatters/__init__.py +11 -0
  170. claude_mpm/core/framework/formatters/capability_generator.py +367 -0
  171. claude_mpm/core/framework/formatters/content_formatter.py +288 -0
  172. claude_mpm/core/framework/formatters/context_generator.py +185 -0
  173. claude_mpm/core/framework/loaders/__init__.py +13 -0
  174. claude_mpm/core/framework/loaders/agent_loader.py +210 -0
  175. claude_mpm/core/framework/loaders/file_loader.py +176 -0
  176. claude_mpm/core/framework/loaders/instruction_loader.py +181 -0
  177. claude_mpm/core/framework/loaders/packaged_loader.py +232 -0
  178. claude_mpm/core/framework/processors/__init__.py +11 -0
  179. claude_mpm/core/framework/processors/memory_processor.py +230 -0
  180. claude_mpm/core/framework/processors/metadata_processor.py +146 -0
  181. claude_mpm/core/framework/processors/template_processor.py +244 -0
  182. claude_mpm/core/framework_loader.py +321 -1631
  183. claude_mpm/core/hook_error_memory.py +381 -0
  184. claude_mpm/core/hook_manager.py +49 -8
  185. claude_mpm/core/injectable_service.py +11 -8
  186. claude_mpm/core/instruction_reinforcement_hook.py +4 -3
  187. claude_mpm/core/interactive_session.py +146 -18
  188. claude_mpm/core/interfaces.py +56 -1
  189. claude_mpm/core/lazy.py +3 -3
  190. claude_mpm/core/log_manager.py +92 -23
  191. claude_mpm/core/logger.py +22 -15
  192. claude_mpm/core/logging_config.py +6 -2
  193. claude_mpm/core/logging_utils.py +520 -0
  194. claude_mpm/core/oneshot_session.py +122 -15
  195. claude_mpm/core/optimized_agent_loader.py +9 -9
  196. claude_mpm/core/optimized_startup.py +1 -1
  197. claude_mpm/core/output_style_manager.py +12 -192
  198. claude_mpm/core/pm_hook_interceptor.py +18 -12
  199. claude_mpm/core/protocols/__init__.py +23 -0
  200. claude_mpm/core/protocols/runner_protocol.py +103 -0
  201. claude_mpm/core/protocols/session_protocol.py +131 -0
  202. claude_mpm/core/service_registry.py +7 -3
  203. claude_mpm/core/session_manager.py +14 -12
  204. claude_mpm/core/shared/config_loader.py +1 -1
  205. claude_mpm/core/shared/singleton_manager.py +11 -4
  206. claude_mpm/core/socketio_pool.py +15 -15
  207. claude_mpm/core/system_context.py +38 -0
  208. claude_mpm/core/tool_access_control.py +3 -2
  209. claude_mpm/core/types.py +4 -11
  210. claude_mpm/core/typing_utils.py +7 -6
  211. claude_mpm/core/unified_agent_registry.py +115 -11
  212. claude_mpm/core/unified_config.py +6 -6
  213. claude_mpm/core/unified_paths.py +23 -20
  214. claude_mpm/dashboard/analysis_runner.py +4 -4
  215. claude_mpm/dashboard/api/simple_directory.py +261 -0
  216. claude_mpm/dashboard/static/css/activity.css +69 -69
  217. claude_mpm/dashboard/static/css/connection-status.css +10 -10
  218. claude_mpm/dashboard/static/css/dashboard.css +600 -18
  219. claude_mpm/dashboard/static/js/components/activity-tree.js +181 -195
  220. claude_mpm/dashboard/static/js/components/agent-hierarchy.js +105 -102
  221. claude_mpm/dashboard/static/js/components/agent-inference.js +34 -31
  222. claude_mpm/dashboard/static/js/components/build-tracker.js +67 -59
  223. claude_mpm/dashboard/static/js/components/code-simple.js +857 -0
  224. claude_mpm/dashboard/static/js/components/connection-debug.js +101 -101
  225. claude_mpm/dashboard/static/js/components/diff-viewer.js +891 -0
  226. claude_mpm/dashboard/static/js/components/event-processor.js +3 -0
  227. claude_mpm/dashboard/static/js/components/event-viewer.js +50 -13
  228. claude_mpm/dashboard/static/js/components/export-manager.js +3 -0
  229. claude_mpm/dashboard/static/js/components/file-change-tracker.js +443 -0
  230. claude_mpm/dashboard/static/js/components/file-change-viewer.js +690 -0
  231. claude_mpm/dashboard/static/js/components/file-tool-tracker.js +36 -16
  232. claude_mpm/dashboard/static/js/components/file-viewer.js +580 -0
  233. claude_mpm/dashboard/static/js/components/module-viewer.js +49 -23
  234. claude_mpm/dashboard/static/js/components/session-manager.js +19 -19
  235. claude_mpm/dashboard/static/js/components/socket-manager.js +5 -1
  236. claude_mpm/dashboard/static/js/components/ui-state-manager.js +356 -41
  237. claude_mpm/dashboard/static/js/components/unified-data-viewer.js +520 -88
  238. claude_mpm/dashboard/static/js/components/working-directory.js +46 -11
  239. claude_mpm/dashboard/static/js/connection-manager.js +76 -76
  240. claude_mpm/dashboard/static/js/dashboard.js +309 -178
  241. claude_mpm/dashboard/static/js/extension-error-handler.js +22 -22
  242. claude_mpm/dashboard/static/js/socket-client.js +183 -139
  243. claude_mpm/dashboard/static/js/tab-isolation-fix.js +185 -0
  244. claude_mpm/dashboard/static/socket.io.min.js +7 -0
  245. claude_mpm/dashboard/static/socket.io.v4.8.1.backup.js +7 -0
  246. claude_mpm/dashboard/templates/code_simple.html +153 -0
  247. claude_mpm/dashboard/templates/index.html +125 -122
  248. claude_mpm/experimental/cli_enhancements.py +5 -7
  249. claude_mpm/generators/agent_profile_generator.py +5 -3
  250. claude_mpm/hooks/__init__.py +37 -1
  251. claude_mpm/hooks/base_hook.py +5 -4
  252. claude_mpm/hooks/claude_hooks/__pycache__/__init__.cpython-313.pyc +0 -0
  253. claude_mpm/hooks/claude_hooks/__pycache__/event_handlers.cpython-313.pyc +0 -0
  254. claude_mpm/hooks/claude_hooks/__pycache__/hook_handler.cpython-313.pyc +0 -0
  255. claude_mpm/hooks/claude_hooks/__pycache__/memory_integration.cpython-313.pyc +0 -0
  256. claude_mpm/hooks/claude_hooks/__pycache__/response_tracking.cpython-313.pyc +0 -0
  257. claude_mpm/hooks/claude_hooks/__pycache__/tool_analysis.cpython-313.pyc +0 -0
  258. claude_mpm/hooks/claude_hooks/connection_pool.py +4 -4
  259. claude_mpm/hooks/claude_hooks/event_handlers.py +24 -19
  260. claude_mpm/hooks/claude_hooks/hook_handler.py +29 -22
  261. claude_mpm/hooks/claude_hooks/installer.py +67 -22
  262. claude_mpm/hooks/claude_hooks/memory_integration.py +3 -3
  263. claude_mpm/hooks/claude_hooks/response_tracking.py +57 -17
  264. claude_mpm/hooks/claude_hooks/services/__pycache__/__init__.cpython-313.pyc +0 -0
  265. claude_mpm/hooks/claude_hooks/services/__pycache__/connection_manager_http.cpython-313.pyc +0 -0
  266. claude_mpm/hooks/claude_hooks/services/__pycache__/duplicate_detector.cpython-313.pyc +0 -0
  267. claude_mpm/hooks/claude_hooks/services/__pycache__/state_manager.cpython-313.pyc +0 -0
  268. claude_mpm/hooks/claude_hooks/services/__pycache__/subagent_processor.cpython-313.pyc +0 -0
  269. claude_mpm/hooks/claude_hooks/services/connection_manager.py +62 -64
  270. claude_mpm/hooks/claude_hooks/services/connection_manager_http.py +140 -76
  271. claude_mpm/hooks/claude_hooks/services/state_manager.py +11 -9
  272. claude_mpm/hooks/claude_hooks/services/subagent_processor.py +3 -3
  273. claude_mpm/hooks/failure_learning/__init__.py +54 -0
  274. claude_mpm/hooks/failure_learning/failure_detection_hook.py +230 -0
  275. claude_mpm/hooks/failure_learning/fix_detection_hook.py +212 -0
  276. claude_mpm/hooks/failure_learning/learning_extraction_hook.py +281 -0
  277. claude_mpm/hooks/instruction_reinforcement.py +301 -0
  278. claude_mpm/hooks/kuzu_enrichment_hook.py +263 -0
  279. claude_mpm/hooks/kuzu_memory_hook.py +386 -0
  280. claude_mpm/hooks/kuzu_response_hook.py +179 -0
  281. claude_mpm/hooks/memory_integration_hook.py +1 -1
  282. claude_mpm/hooks/session_resume_hook.py +121 -0
  283. claude_mpm/hooks/templates/pre_tool_use_simple.py +78 -0
  284. claude_mpm/hooks/templates/pre_tool_use_template.py +323 -0
  285. claude_mpm/hooks/tool_call_interceptor.py +8 -5
  286. claude_mpm/hooks/validation_hooks.py +3 -3
  287. claude_mpm/init.py +23 -4
  288. claude_mpm/models/agent_session.py +8 -6
  289. claude_mpm/models/git_repository.py +198 -0
  290. claude_mpm/models/resume_log.py +340 -0
  291. claude_mpm/scripts/claude-hook-handler.sh +35 -9
  292. claude_mpm/scripts/launch_monitor.py +85 -0
  293. claude_mpm/scripts/mcp_server.py +3 -5
  294. claude_mpm/scripts/mpm_doctor.py +3 -2
  295. claude_mpm/scripts/socketio_daemon.py +159 -512
  296. claude_mpm/scripts/start_activity_logging.py +3 -1
  297. claude_mpm/services/__init__.py +144 -160
  298. claude_mpm/services/agents/__init__.py +18 -5
  299. claude_mpm/services/agents/agent_builder.py +56 -18
  300. claude_mpm/services/agents/agent_preset_service.py +238 -0
  301. claude_mpm/services/agents/agent_selection_service.py +484 -0
  302. claude_mpm/services/agents/auto_config_manager.py +796 -0
  303. claude_mpm/services/agents/auto_deploy_index_parser.py +569 -0
  304. claude_mpm/services/agents/cache_git_manager.py +621 -0
  305. claude_mpm/services/agents/deployment/agent_configuration_manager.py +1 -1
  306. claude_mpm/services/agents/deployment/agent_deployment.py +164 -17
  307. claude_mpm/services/agents/deployment/agent_discovery_service.py +191 -41
  308. claude_mpm/services/agents/deployment/agent_filesystem_manager.py +5 -5
  309. claude_mpm/services/agents/deployment/agent_format_converter.py +56 -12
  310. claude_mpm/services/agents/deployment/agent_lifecycle_manager.py +5 -7
  311. claude_mpm/services/agents/deployment/agent_metrics_collector.py +3 -3
  312. claude_mpm/services/agents/deployment/agent_operation_service.py +2 -2
  313. claude_mpm/services/agents/deployment/agent_record_service.py +4 -4
  314. claude_mpm/services/agents/deployment/agent_restore_handler.py +1 -4
  315. claude_mpm/services/agents/deployment/agent_state_service.py +2 -2
  316. claude_mpm/services/agents/deployment/agent_template_builder.py +939 -50
  317. claude_mpm/services/agents/deployment/agent_validator.py +31 -7
  318. claude_mpm/services/agents/deployment/agent_version_manager.py +8 -5
  319. claude_mpm/services/agents/deployment/agent_versioning.py +1 -1
  320. claude_mpm/services/agents/deployment/agents_directory_resolver.py +101 -15
  321. claude_mpm/services/agents/deployment/async_agent_deployment.py +3 -2
  322. claude_mpm/services/agents/deployment/deployment_config_loader.py +131 -7
  323. claude_mpm/services/agents/deployment/deployment_type_detector.py +10 -14
  324. claude_mpm/services/agents/deployment/deployment_wrapper.py +58 -0
  325. claude_mpm/services/agents/deployment/facade/deployment_facade.py +3 -3
  326. claude_mpm/services/agents/deployment/interface_adapter.py +3 -2
  327. claude_mpm/services/agents/deployment/local_template_deployment.py +360 -0
  328. claude_mpm/services/agents/deployment/multi_source_deployment_service.py +249 -53
  329. claude_mpm/services/agents/deployment/pipeline/pipeline_executor.py +2 -2
  330. claude_mpm/services/agents/deployment/pipeline/steps/agent_processing_step.py +8 -7
  331. claude_mpm/services/agents/deployment/pipeline/steps/base_step.py +7 -16
  332. claude_mpm/services/agents/deployment/pipeline/steps/configuration_step.py +4 -3
  333. claude_mpm/services/agents/deployment/pipeline/steps/target_directory_step.py +7 -5
  334. claude_mpm/services/agents/deployment/pipeline/steps/validation_step.py +6 -5
  335. claude_mpm/services/agents/deployment/refactored_agent_deployment_service.py +10 -10
  336. claude_mpm/services/agents/deployment/remote_agent_discovery_service.py +363 -0
  337. claude_mpm/services/agents/deployment/single_agent_deployer.py +2 -2
  338. claude_mpm/services/agents/deployment/system_instructions_deployer.py +168 -43
  339. claude_mpm/services/agents/deployment/validation/__init__.py +3 -1
  340. claude_mpm/services/agents/deployment/validation/deployment_validator.py +2 -2
  341. claude_mpm/services/agents/deployment/validation/template_validator.py +64 -44
  342. claude_mpm/services/agents/deployment/validation/validation_result.py +1 -9
  343. claude_mpm/services/agents/git_source_manager.py +629 -0
  344. claude_mpm/services/agents/loading/agent_profile_loader.py +10 -9
  345. claude_mpm/services/agents/loading/base_agent_manager.py +16 -6
  346. claude_mpm/services/agents/loading/framework_agent_loader.py +11 -14
  347. claude_mpm/services/agents/local_template_manager.py +784 -0
  348. claude_mpm/services/agents/management/agent_capabilities_generator.py +3 -2
  349. claude_mpm/services/agents/management/agent_management_service.py +5 -5
  350. claude_mpm/services/agents/memory/agent_memory_manager.py +32 -29
  351. claude_mpm/services/agents/memory/content_manager.py +17 -9
  352. claude_mpm/services/agents/memory/memory_categorization_service.py +4 -2
  353. claude_mpm/services/agents/memory/memory_file_service.py +32 -6
  354. claude_mpm/services/agents/memory/memory_format_service.py +6 -4
  355. claude_mpm/services/agents/memory/memory_limits_service.py +4 -2
  356. claude_mpm/services/agents/memory/template_generator.py +3 -3
  357. claude_mpm/services/agents/observers.py +547 -0
  358. claude_mpm/services/agents/recommender.py +615 -0
  359. claude_mpm/services/agents/registry/deployed_agent_discovery.py +3 -3
  360. claude_mpm/services/agents/registry/modification_tracker.py +30 -19
  361. claude_mpm/services/agents/single_tier_deployment_service.py +696 -0
  362. claude_mpm/services/agents/sources/__init__.py +13 -0
  363. claude_mpm/services/agents/sources/agent_sync_state.py +516 -0
  364. claude_mpm/services/agents/sources/git_source_sync_service.py +1087 -0
  365. claude_mpm/services/agents/startup_sync.py +239 -0
  366. claude_mpm/services/agents/toolchain_detector.py +474 -0
  367. claude_mpm/services/analysis/__init__.py +25 -0
  368. claude_mpm/services/analysis/postmortem_reporter.py +474 -0
  369. claude_mpm/services/analysis/postmortem_service.py +765 -0
  370. claude_mpm/services/async_session_logger.py +141 -98
  371. claude_mpm/services/claude_session_logger.py +82 -74
  372. claude_mpm/services/cli/agent_cleanup_service.py +5 -0
  373. claude_mpm/services/cli/agent_listing_service.py +5 -5
  374. claude_mpm/services/cli/agent_validation_service.py +3 -1
  375. claude_mpm/services/cli/memory_crud_service.py +12 -7
  376. claude_mpm/services/cli/memory_output_formatter.py +2 -2
  377. claude_mpm/services/cli/resume_service.py +617 -0
  378. claude_mpm/services/cli/session_manager.py +104 -13
  379. claude_mpm/services/cli/session_pause_manager.py +504 -0
  380. claude_mpm/services/cli/session_resume_helper.py +372 -0
  381. claude_mpm/services/cli/startup_checker.py +13 -10
  382. claude_mpm/services/cli/unified_dashboard_manager.py +439 -0
  383. claude_mpm/services/command_deployment_service.py +209 -13
  384. claude_mpm/services/command_handler_service.py +11 -5
  385. claude_mpm/services/core/__init__.py +33 -1
  386. claude_mpm/services/core/base.py +31 -11
  387. claude_mpm/services/core/interfaces/__init__.py +88 -3
  388. claude_mpm/services/core/interfaces/agent.py +184 -0
  389. claude_mpm/services/core/interfaces/health.py +169 -0
  390. claude_mpm/services/core/interfaces/model.py +281 -0
  391. claude_mpm/services/core/interfaces/process.py +372 -0
  392. claude_mpm/services/core/interfaces/project.py +121 -0
  393. claude_mpm/services/core/interfaces/restart.py +307 -0
  394. claude_mpm/services/core/interfaces/stability.py +260 -0
  395. claude_mpm/services/core/interfaces.py +56 -1
  396. claude_mpm/services/core/memory_manager.py +92 -47
  397. claude_mpm/services/core/models/__init__.py +70 -0
  398. claude_mpm/services/core/models/agent_config.py +384 -0
  399. claude_mpm/services/core/models/health.py +162 -0
  400. claude_mpm/services/core/models/process.py +239 -0
  401. claude_mpm/services/core/models/restart.py +302 -0
  402. claude_mpm/services/core/models/stability.py +264 -0
  403. claude_mpm/services/core/models/toolchain.py +306 -0
  404. claude_mpm/services/core/path_resolver.py +36 -14
  405. claude_mpm/services/diagnostics/__init__.py +2 -2
  406. claude_mpm/services/diagnostics/checks/__init__.py +8 -2
  407. claude_mpm/services/diagnostics/checks/agent_check.py +30 -34
  408. claude_mpm/services/diagnostics/checks/agent_sources_check.py +577 -0
  409. claude_mpm/services/diagnostics/checks/claude_code_check.py +270 -0
  410. claude_mpm/services/diagnostics/checks/common_issues_check.py +28 -27
  411. claude_mpm/services/diagnostics/checks/configuration_check.py +26 -25
  412. claude_mpm/services/diagnostics/checks/filesystem_check.py +18 -17
  413. claude_mpm/services/diagnostics/checks/installation_check.py +165 -60
  414. claude_mpm/services/diagnostics/checks/instructions_check.py +21 -21
  415. claude_mpm/services/diagnostics/checks/mcp_check.py +57 -44
  416. claude_mpm/services/diagnostics/checks/mcp_services_check.py +1058 -0
  417. claude_mpm/services/diagnostics/checks/monitor_check.py +24 -24
  418. claude_mpm/services/diagnostics/checks/skill_sources_check.py +587 -0
  419. claude_mpm/services/diagnostics/checks/startup_log_check.py +14 -11
  420. claude_mpm/services/diagnostics/diagnostic_runner.py +31 -13
  421. claude_mpm/services/diagnostics/doctor_reporter.py +305 -47
  422. claude_mpm/services/diagnostics/models.py +37 -21
  423. claude_mpm/services/event_aggregator.py +5 -3
  424. claude_mpm/services/event_bus/direct_relay.py +11 -7
  425. claude_mpm/services/event_bus/event_bus.py +51 -9
  426. claude_mpm/services/event_bus/relay.py +33 -14
  427. claude_mpm/services/events/consumers/dead_letter.py +7 -5
  428. claude_mpm/services/events/consumers/logging.py +1 -2
  429. claude_mpm/services/events/core.py +5 -6
  430. claude_mpm/services/events/producers/hook.py +6 -6
  431. claude_mpm/services/events/producers/system.py +8 -8
  432. claude_mpm/services/exceptions.py +5 -5
  433. claude_mpm/services/framework_claude_md_generator/__init__.py +1 -1
  434. claude_mpm/services/framework_claude_md_generator/content_assembler.py +5 -5
  435. claude_mpm/services/framework_claude_md_generator/content_validator.py +2 -2
  436. claude_mpm/services/framework_claude_md_generator/deployment_manager.py +3 -3
  437. claude_mpm/services/framework_claude_md_generator/section_generators/__init__.py +2 -2
  438. claude_mpm/services/framework_claude_md_generator/version_manager.py +1 -1
  439. claude_mpm/services/git/__init__.py +21 -0
  440. claude_mpm/services/git/git_operations_service.py +494 -0
  441. claude_mpm/services/github/__init__.py +21 -0
  442. claude_mpm/services/github/github_cli_service.py +397 -0
  443. claude_mpm/services/hook_installer_service.py +506 -0
  444. claude_mpm/services/hook_service.py +5 -6
  445. claude_mpm/services/infrastructure/context_preservation.py +13 -11
  446. claude_mpm/services/infrastructure/daemon_manager.py +9 -9
  447. claude_mpm/services/infrastructure/logging.py +2 -2
  448. claude_mpm/services/infrastructure/monitoring/__init__.py +2 -6
  449. claude_mpm/services/infrastructure/monitoring/aggregator.py +13 -18
  450. claude_mpm/services/infrastructure/monitoring/base.py +5 -13
  451. claude_mpm/services/infrastructure/monitoring/network.py +7 -6
  452. claude_mpm/services/infrastructure/monitoring/process.py +13 -12
  453. claude_mpm/services/infrastructure/monitoring/resources.py +8 -7
  454. claude_mpm/services/infrastructure/monitoring/service.py +16 -15
  455. claude_mpm/services/infrastructure/resume_log_generator.py +439 -0
  456. claude_mpm/services/instructions/__init__.py +9 -0
  457. claude_mpm/services/instructions/instruction_cache_service.py +374 -0
  458. claude_mpm/services/local_ops/__init__.py +155 -0
  459. claude_mpm/services/local_ops/crash_detector.py +257 -0
  460. claude_mpm/services/local_ops/health_checks/__init__.py +26 -0
  461. claude_mpm/services/local_ops/health_checks/http_check.py +224 -0
  462. claude_mpm/services/local_ops/health_checks/process_check.py +236 -0
  463. claude_mpm/services/local_ops/health_checks/resource_check.py +255 -0
  464. claude_mpm/services/local_ops/health_manager.py +427 -0
  465. claude_mpm/services/local_ops/log_monitor.py +396 -0
  466. claude_mpm/services/local_ops/memory_leak_detector.py +294 -0
  467. claude_mpm/services/local_ops/process_manager.py +595 -0
  468. claude_mpm/services/local_ops/resource_monitor.py +331 -0
  469. claude_mpm/services/local_ops/restart_manager.py +401 -0
  470. claude_mpm/services/local_ops/restart_policy.py +387 -0
  471. claude_mpm/services/local_ops/state_manager.py +372 -0
  472. claude_mpm/services/local_ops/unified_manager.py +600 -0
  473. claude_mpm/services/mcp_config_manager.py +1542 -0
  474. claude_mpm/services/mcp_gateway/__init__.py +97 -93
  475. claude_mpm/services/mcp_gateway/auto_configure.py +43 -38
  476. claude_mpm/services/mcp_gateway/config/config_loader.py +3 -3
  477. claude_mpm/services/mcp_gateway/config/configuration.py +24 -5
  478. claude_mpm/services/mcp_gateway/core/__init__.py +1 -2
  479. claude_mpm/services/mcp_gateway/core/base.py +20 -33
  480. claude_mpm/services/mcp_gateway/core/process_pool.py +591 -31
  481. claude_mpm/services/mcp_gateway/core/singleton_manager.py +2 -2
  482. claude_mpm/services/mcp_gateway/core/startup_verification.py +3 -3
  483. claude_mpm/services/mcp_gateway/main.py +90 -15
  484. claude_mpm/services/mcp_gateway/registry/service_registry.py +4 -2
  485. claude_mpm/services/mcp_gateway/registry/tool_registry.py +12 -9
  486. claude_mpm/services/mcp_gateway/server/mcp_gateway.py +5 -10
  487. claude_mpm/services/mcp_gateway/server/stdio_server.py +9 -17
  488. claude_mpm/services/mcp_gateway/tools/__init__.py +14 -2
  489. claude_mpm/services/mcp_gateway/tools/base_adapter.py +15 -15
  490. claude_mpm/services/mcp_gateway/tools/document_summarizer.py +10 -9
  491. claude_mpm/services/mcp_gateway/tools/external_mcp_services.py +654 -0
  492. claude_mpm/services/mcp_gateway/tools/health_check_tool.py +36 -34
  493. claude_mpm/services/mcp_gateway/tools/hello_world.py +8 -8
  494. claude_mpm/services/mcp_gateway/tools/kuzu_memory_service.py +555 -0
  495. claude_mpm/services/mcp_gateway/utils/__init__.py +14 -0
  496. claude_mpm/services/mcp_gateway/utils/package_version_checker.py +160 -0
  497. claude_mpm/services/mcp_gateway/utils/update_preferences.py +170 -0
  498. claude_mpm/services/mcp_service_verifier.py +732 -0
  499. claude_mpm/services/memory/builder.py +9 -8
  500. claude_mpm/services/memory/cache/shared_prompt_cache.py +2 -1
  501. claude_mpm/services/memory/cache/simple_cache.py +2 -2
  502. claude_mpm/services/memory/failure_tracker.py +578 -0
  503. claude_mpm/services/memory/indexed_memory.py +8 -8
  504. claude_mpm/services/memory/optimizer.py +8 -9
  505. claude_mpm/services/memory/router.py +3 -3
  506. claude_mpm/services/memory_hook_service.py +165 -4
  507. claude_mpm/services/model/__init__.py +147 -0
  508. claude_mpm/services/model/base_provider.py +365 -0
  509. claude_mpm/services/model/claude_provider.py +412 -0
  510. claude_mpm/services/model/model_router.py +452 -0
  511. claude_mpm/services/model/ollama_provider.py +415 -0
  512. claude_mpm/services/monitor/__init__.py +20 -0
  513. claude_mpm/services/monitor/daemon.py +691 -0
  514. claude_mpm/services/monitor/daemon_manager.py +1040 -0
  515. claude_mpm/services/monitor/event_emitter.py +350 -0
  516. claude_mpm/services/monitor/handlers/__init__.py +21 -0
  517. claude_mpm/services/monitor/handlers/code_analysis.py +332 -0
  518. claude_mpm/services/monitor/handlers/dashboard.py +299 -0
  519. claude_mpm/services/monitor/handlers/file.py +264 -0
  520. claude_mpm/services/monitor/handlers/hooks.py +512 -0
  521. claude_mpm/services/monitor/management/__init__.py +18 -0
  522. claude_mpm/services/monitor/management/health.py +124 -0
  523. claude_mpm/services/monitor/management/lifecycle.py +724 -0
  524. claude_mpm/services/monitor/server.py +817 -0
  525. claude_mpm/services/monitor_build_service.py +2 -2
  526. claude_mpm/services/native_agent_converter.py +356 -0
  527. claude_mpm/services/orphan_detection.py +786 -0
  528. claude_mpm/services/port_manager.py +3 -3
  529. claude_mpm/services/pr/__init__.py +14 -0
  530. claude_mpm/services/pr/pr_template_service.py +329 -0
  531. claude_mpm/services/project/__init__.py +23 -0
  532. claude_mpm/services/project/analyzer.py +3 -3
  533. claude_mpm/services/project/architecture_analyzer.py +5 -5
  534. claude_mpm/services/project/archive_manager.py +1045 -0
  535. claude_mpm/services/project/dependency_analyzer.py +4 -4
  536. claude_mpm/services/project/detection_strategies.py +719 -0
  537. claude_mpm/services/project/documentation_manager.py +554 -0
  538. claude_mpm/services/project/enhanced_analyzer.py +572 -0
  539. claude_mpm/services/project/metrics_collector.py +4 -4
  540. claude_mpm/services/project/project_organizer.py +1005 -0
  541. claude_mpm/services/project/registry.py +13 -7
  542. claude_mpm/services/project/toolchain_analyzer.py +583 -0
  543. claude_mpm/services/project_port_allocator.py +596 -0
  544. claude_mpm/services/response_tracker.py +21 -10
  545. claude_mpm/services/runner_configuration_service.py +17 -3
  546. claude_mpm/services/self_upgrade_service.py +500 -0
  547. claude_mpm/services/session_management_service.py +23 -9
  548. claude_mpm/services/session_manager.py +380 -0
  549. claude_mpm/services/shared/__init__.py +2 -1
  550. claude_mpm/services/shared/async_service_base.py +16 -27
  551. claude_mpm/services/shared/config_service_base.py +17 -14
  552. claude_mpm/services/shared/lifecycle_service_base.py +1 -14
  553. claude_mpm/services/shared/service_factory.py +8 -5
  554. claude_mpm/services/skills/__init__.py +18 -0
  555. claude_mpm/services/skills/git_skill_source_manager.py +1169 -0
  556. claude_mpm/services/skills/skill_discovery_service.py +568 -0
  557. claude_mpm/services/skills_config.py +547 -0
  558. claude_mpm/services/skills_deployer.py +955 -0
  559. claude_mpm/services/socketio/client_proxy.py +60 -5
  560. claude_mpm/services/socketio/dashboard_server.py +361 -0
  561. claude_mpm/services/socketio/event_normalizer.py +10 -6
  562. claude_mpm/services/socketio/handlers/__init__.py +5 -2
  563. claude_mpm/services/socketio/handlers/base.py +2 -2
  564. claude_mpm/services/socketio/handlers/code_analysis.py +90 -27
  565. claude_mpm/services/socketio/handlers/connection.py +22 -41
  566. claude_mpm/services/socketio/handlers/connection_handler.py +13 -10
  567. claude_mpm/services/socketio/handlers/file.py +46 -10
  568. claude_mpm/services/socketio/handlers/git.py +9 -9
  569. claude_mpm/services/socketio/handlers/hook.py +29 -17
  570. claude_mpm/services/socketio/handlers/registry.py +4 -2
  571. claude_mpm/services/socketio/monitor_client.py +364 -0
  572. claude_mpm/services/socketio/server/broadcaster.py +9 -7
  573. claude_mpm/services/socketio/server/connection_manager.py +2 -2
  574. claude_mpm/services/socketio/server/core.py +142 -8
  575. claude_mpm/services/socketio/server/eventbus_integration.py +20 -14
  576. claude_mpm/services/socketio/server/main.py +24 -24
  577. claude_mpm/services/socketio_client_manager.py +4 -4
  578. claude_mpm/services/subprocess_launcher_service.py +19 -15
  579. claude_mpm/services/system_instructions_service.py +3 -5
  580. claude_mpm/services/ticket_services/formatter_service.py +1 -1
  581. claude_mpm/services/ticket_services/validation_service.py +5 -5
  582. claude_mpm/services/unified/__init__.py +65 -0
  583. claude_mpm/services/unified/analyzer_strategies/__init__.py +44 -0
  584. claude_mpm/services/unified/analyzer_strategies/code_analyzer.py +518 -0
  585. claude_mpm/services/unified/analyzer_strategies/dependency_analyzer.py +680 -0
  586. claude_mpm/services/unified/analyzer_strategies/performance_analyzer.py +900 -0
  587. claude_mpm/services/unified/analyzer_strategies/security_analyzer.py +745 -0
  588. claude_mpm/services/unified/analyzer_strategies/structure_analyzer.py +733 -0
  589. claude_mpm/services/unified/config_strategies/__init__.py +175 -0
  590. claude_mpm/services/unified/config_strategies/config_schema.py +731 -0
  591. claude_mpm/services/unified/config_strategies/context_strategy.py +747 -0
  592. claude_mpm/services/unified/config_strategies/error_handling_strategy.py +1005 -0
  593. claude_mpm/services/unified/config_strategies/file_loader_strategy.py +881 -0
  594. claude_mpm/services/unified/config_strategies/unified_config_service.py +823 -0
  595. claude_mpm/services/unified/config_strategies/validation_strategy.py +1148 -0
  596. claude_mpm/services/unified/deployment_strategies/__init__.py +97 -0
  597. claude_mpm/services/unified/deployment_strategies/base.py +553 -0
  598. claude_mpm/services/unified/deployment_strategies/cloud_strategies.py +573 -0
  599. claude_mpm/services/unified/deployment_strategies/local.py +607 -0
  600. claude_mpm/services/unified/deployment_strategies/utils.py +667 -0
  601. claude_mpm/services/unified/deployment_strategies/vercel.py +471 -0
  602. claude_mpm/services/unified/interfaces.py +475 -0
  603. claude_mpm/services/unified/migration.py +509 -0
  604. claude_mpm/services/unified/strategies.py +534 -0
  605. claude_mpm/services/unified/unified_analyzer.py +542 -0
  606. claude_mpm/services/unified/unified_config.py +691 -0
  607. claude_mpm/services/unified/unified_deployment.py +466 -0
  608. claude_mpm/services/utility_service.py +6 -3
  609. claude_mpm/services/version_control/branch_strategy.py +2 -2
  610. claude_mpm/services/version_control/conflict_resolution.py +14 -8
  611. claude_mpm/services/version_control/git_operations.py +26 -24
  612. claude_mpm/services/version_control/semantic_versioning.py +14 -14
  613. claude_mpm/services/version_control/version_parser.py +14 -11
  614. claude_mpm/services/version_service.py +104 -1
  615. claude_mpm/services/visualization/__init__.py +1 -5
  616. claude_mpm/services/visualization/mermaid_generator.py +2 -3
  617. claude_mpm/skills/__init__.py +42 -0
  618. claude_mpm/skills/agent_skills_injector.py +324 -0
  619. claude_mpm/skills/bundled/LICENSE_ATTRIBUTIONS.md +79 -0
  620. claude_mpm/skills/bundled/__init__.py +6 -0
  621. claude_mpm/skills/bundled/api-documentation.md +393 -0
  622. claude_mpm/skills/bundled/async-testing.md +571 -0
  623. claude_mpm/skills/bundled/code-review.md +143 -0
  624. claude_mpm/skills/bundled/database-migration.md +199 -0
  625. claude_mpm/skills/bundled/docker-containerization.md +194 -0
  626. claude_mpm/skills/bundled/express-local-dev.md +1429 -0
  627. claude_mpm/skills/bundled/fastapi-local-dev.md +1199 -0
  628. claude_mpm/skills/bundled/git-workflow.md +414 -0
  629. claude_mpm/skills/bundled/imagemagick.md +204 -0
  630. claude_mpm/skills/bundled/infrastructure/env-manager/scripts/validate_env.py +576 -0
  631. claude_mpm/skills/bundled/json-data-handling.md +223 -0
  632. claude_mpm/skills/bundled/main/mcp-builder/scripts/connections.py +157 -0
  633. claude_mpm/skills/bundled/main/mcp-builder/scripts/evaluation.py +425 -0
  634. claude_mpm/skills/bundled/main/skill-creator/scripts/init_skill.py +303 -0
  635. claude_mpm/skills/bundled/main/skill-creator/scripts/package_skill.py +113 -0
  636. claude_mpm/skills/bundled/main/skill-creator/scripts/quick_validate.py +72 -0
  637. claude_mpm/skills/bundled/nextjs-local-dev.md +807 -0
  638. claude_mpm/skills/bundled/pdf.md +141 -0
  639. claude_mpm/skills/bundled/performance-profiling.md +573 -0
  640. claude_mpm/skills/bundled/refactoring-patterns.md +180 -0
  641. claude_mpm/skills/bundled/security-scanning.md +327 -0
  642. claude_mpm/skills/bundled/systematic-debugging.md +473 -0
  643. claude_mpm/skills/bundled/test-driven-development.md +378 -0
  644. claude_mpm/skills/bundled/testing/webapp-testing/examples/console_logging.py +35 -0
  645. claude_mpm/skills/bundled/testing/webapp-testing/examples/element_discovery.py +44 -0
  646. claude_mpm/skills/bundled/testing/webapp-testing/examples/static_html_automation.py +34 -0
  647. claude_mpm/skills/bundled/testing/webapp-testing/scripts/with_server.py +129 -0
  648. claude_mpm/skills/bundled/vite-local-dev.md +1061 -0
  649. claude_mpm/skills/bundled/web-performance-optimization.md +2305 -0
  650. claude_mpm/skills/bundled/xlsx.md +157 -0
  651. claude_mpm/skills/registry.py +286 -0
  652. claude_mpm/skills/skill_manager.py +310 -0
  653. claude_mpm/skills/skills_registry.py +347 -0
  654. claude_mpm/skills/skills_service.py +739 -0
  655. claude_mpm/storage/state_storage.py +31 -31
  656. claude_mpm/templates/questions/__init__.py +38 -0
  657. claude_mpm/templates/questions/base.py +193 -0
  658. claude_mpm/templates/questions/pr_strategy.py +311 -0
  659. claude_mpm/templates/questions/project_init.py +385 -0
  660. claude_mpm/templates/questions/ticket_mgmt.py +394 -0
  661. claude_mpm/tools/__main__.py +9 -9
  662. claude_mpm/tools/code_tree_analyzer/__init__.py +45 -0
  663. claude_mpm/tools/code_tree_analyzer/analysis.py +299 -0
  664. claude_mpm/tools/code_tree_analyzer/cache.py +131 -0
  665. claude_mpm/tools/code_tree_analyzer/core.py +380 -0
  666. claude_mpm/tools/code_tree_analyzer/discovery.py +403 -0
  667. claude_mpm/tools/code_tree_analyzer/events.py +168 -0
  668. claude_mpm/tools/code_tree_analyzer/gitignore.py +308 -0
  669. claude_mpm/tools/code_tree_analyzer/models.py +39 -0
  670. claude_mpm/tools/code_tree_analyzer/multilang_analyzer.py +224 -0
  671. claude_mpm/tools/code_tree_analyzer/python_analyzer.py +284 -0
  672. claude_mpm/tools/code_tree_builder.py +6 -6
  673. claude_mpm/tools/code_tree_events.py +14 -10
  674. claude_mpm/tools/socketio_debug.py +11 -11
  675. claude_mpm/utils/agent_dependency_loader.py +184 -36
  676. claude_mpm/utils/agent_filters.py +288 -0
  677. claude_mpm/utils/common.py +544 -0
  678. claude_mpm/utils/config_manager.py +12 -6
  679. claude_mpm/utils/database_connector.py +298 -0
  680. claude_mpm/utils/dependency_cache.py +5 -3
  681. claude_mpm/utils/dependency_strategies.py +15 -10
  682. claude_mpm/utils/display_helper.py +260 -0
  683. claude_mpm/utils/environment_context.py +4 -3
  684. claude_mpm/utils/error_handler.py +5 -3
  685. claude_mpm/utils/file_utils.py +13 -14
  686. claude_mpm/utils/git_analyzer.py +407 -0
  687. claude_mpm/utils/gitignore.py +241 -0
  688. claude_mpm/utils/log_cleanup.py +627 -0
  689. claude_mpm/utils/migration.py +372 -0
  690. claude_mpm/utils/path_operations.py +7 -4
  691. claude_mpm/utils/progress.py +387 -0
  692. claude_mpm/utils/robust_installer.py +131 -24
  693. claude_mpm/utils/session_logging.py +2 -2
  694. claude_mpm/utils/structured_questions.py +619 -0
  695. claude_mpm/utils/subprocess_utils.py +9 -8
  696. claude_mpm/validation/agent_validator.py +6 -6
  697. claude_mpm/validation/frontmatter_validator.py +6 -6
  698. claude_mpm-5.0.9.dist-info/METADATA +1028 -0
  699. claude_mpm-5.0.9.dist-info/RECORD +864 -0
  700. {claude_mpm-4.1.26.dist-info → claude_mpm-5.0.9.dist-info}/entry_points.txt +1 -0
  701. claude_mpm/agents/INSTRUCTIONS.md +0 -261
  702. claude_mpm/agents/templates/.claude-mpm/memories/README.md +0 -17
  703. claude_mpm/agents/templates/.claude-mpm/memories/engineer_memories.md +0 -3
  704. claude_mpm/agents/templates/agent-manager.json +0 -270
  705. claude_mpm/agents/templates/agent-manager.md +0 -619
  706. claude_mpm/agents/templates/agentic_coder_optimizer.json +0 -222
  707. claude_mpm/agents/templates/api_qa.json +0 -171
  708. claude_mpm/agents/templates/code_analyzer.json +0 -95
  709. claude_mpm/agents/templates/data_engineer.json +0 -152
  710. claude_mpm/agents/templates/documentation.json +0 -175
  711. claude_mpm/agents/templates/engineer.json +0 -176
  712. claude_mpm/agents/templates/imagemagick.json +0 -261
  713. claude_mpm/agents/templates/logs/prompts/agent_engineer_20250826_014258_728.md +0 -39
  714. claude_mpm/agents/templates/memory_manager.json +0 -155
  715. claude_mpm/agents/templates/ops.json +0 -175
  716. claude_mpm/agents/templates/project_organizer.json +0 -130
  717. claude_mpm/agents/templates/qa.json +0 -223
  718. claude_mpm/agents/templates/refactoring_engineer.json +0 -266
  719. claude_mpm/agents/templates/research.json +0 -163
  720. claude_mpm/agents/templates/security.json +0 -153
  721. claude_mpm/agents/templates/ticketing.json +0 -169
  722. claude_mpm/agents/templates/vercel_ops_agent.json +0 -281
  723. claude_mpm/agents/templates/version_control.json +0 -147
  724. claude_mpm/agents/templates/web_qa.json +0 -254
  725. claude_mpm/agents/templates/web_ui.json +0 -176
  726. claude_mpm/cli/commands/configure_tui.py +0 -1927
  727. claude_mpm/cli/commands/mpm_init.py +0 -594
  728. claude_mpm/cli/commands/socketio_monitor.py +0 -233
  729. claude_mpm/commands/mpm-agents.md +0 -12
  730. claude_mpm/commands/mpm-config.md +0 -18
  731. claude_mpm/commands/mpm-tickets.md +0 -102
  732. claude_mpm/dashboard/.claude-mpm/socketio-instances.json +0 -1
  733. claude_mpm/dashboard/static/built/components/activity-tree.js +0 -2
  734. claude_mpm/dashboard/static/built/components/agent-inference.js +0 -2
  735. claude_mpm/dashboard/static/built/components/code-tree.js +0 -2
  736. claude_mpm/dashboard/static/built/components/code-viewer.js +0 -2
  737. claude_mpm/dashboard/static/built/components/event-processor.js +0 -2
  738. claude_mpm/dashboard/static/built/components/event-viewer.js +0 -2
  739. claude_mpm/dashboard/static/built/components/export-manager.js +0 -2
  740. claude_mpm/dashboard/static/built/components/file-tool-tracker.js +0 -2
  741. claude_mpm/dashboard/static/built/components/hud-library-loader.js +0 -2
  742. claude_mpm/dashboard/static/built/components/hud-manager.js +0 -2
  743. claude_mpm/dashboard/static/built/components/hud-visualizer.js +0 -2
  744. claude_mpm/dashboard/static/built/components/module-viewer.js +0 -2
  745. claude_mpm/dashboard/static/built/components/session-manager.js +0 -2
  746. claude_mpm/dashboard/static/built/components/socket-manager.js +0 -2
  747. claude_mpm/dashboard/static/built/components/ui-state-manager.js +0 -2
  748. claude_mpm/dashboard/static/built/components/unified-data-viewer.js +0 -2
  749. claude_mpm/dashboard/static/built/components/working-directory.js +0 -2
  750. claude_mpm/dashboard/static/built/dashboard.js +0 -2
  751. claude_mpm/dashboard/static/built/socket-client.js +0 -2
  752. claude_mpm/dashboard/static/css/code-tree.css +0 -1408
  753. claude_mpm/dashboard/static/dist/components/activity-tree.js +0 -2
  754. claude_mpm/dashboard/static/dist/components/agent-inference.js +0 -2
  755. claude_mpm/dashboard/static/dist/components/code-tree.js +0 -2
  756. claude_mpm/dashboard/static/dist/components/code-viewer.js +0 -2
  757. claude_mpm/dashboard/static/dist/components/event-processor.js +0 -2
  758. claude_mpm/dashboard/static/dist/components/event-viewer.js +0 -2
  759. claude_mpm/dashboard/static/dist/components/export-manager.js +0 -2
  760. claude_mpm/dashboard/static/dist/components/file-tool-tracker.js +0 -2
  761. claude_mpm/dashboard/static/dist/components/hud-library-loader.js +0 -2
  762. claude_mpm/dashboard/static/dist/components/hud-manager.js +0 -2
  763. claude_mpm/dashboard/static/dist/components/hud-visualizer.js +0 -2
  764. claude_mpm/dashboard/static/dist/components/module-viewer.js +0 -2
  765. claude_mpm/dashboard/static/dist/components/session-manager.js +0 -2
  766. claude_mpm/dashboard/static/dist/components/socket-manager.js +0 -2
  767. claude_mpm/dashboard/static/dist/components/ui-state-manager.js +0 -2
  768. claude_mpm/dashboard/static/dist/components/unified-data-viewer.js +0 -2
  769. claude_mpm/dashboard/static/dist/components/working-directory.js +0 -2
  770. claude_mpm/dashboard/static/dist/dashboard.js +0 -2
  771. claude_mpm/dashboard/static/dist/socket-client.js +0 -2
  772. claude_mpm/dashboard/static/js/components/code-tree.js +0 -3220
  773. claude_mpm/dashboard/static/js/components/code-viewer.js +0 -480
  774. claude_mpm/hooks/claude_hooks/hook_handler_eventbus.py +0 -425
  775. claude_mpm/hooks/claude_hooks/hook_handler_original.py +0 -1040
  776. claude_mpm/hooks/claude_hooks/hook_handler_refactored.py +0 -347
  777. claude_mpm/scripts/socketio_daemon_hardened.py +0 -937
  778. claude_mpm/scripts/socketio_daemon_wrapper.py +0 -78
  779. claude_mpm/scripts/socketio_server_manager.py +0 -349
  780. claude_mpm/services/agents/deployment/agent_lifecycle_manager_refactored.py +0 -575
  781. claude_mpm/services/cli/dashboard_launcher.py +0 -423
  782. claude_mpm/services/cli/socketio_manager.py +0 -537
  783. claude_mpm/services/diagnostics/checks/claude_desktop_check.py +0 -286
  784. claude_mpm/services/mcp_gateway/tools/ticket_tools.py +0 -645
  785. claude_mpm/services/mcp_gateway/tools/unified_ticket_tool.py +0 -602
  786. claude_mpm/services/project/analyzer_refactored.py +0 -450
  787. claude_mpm/tools/code_tree_analyzer.py +0 -1693
  788. claude_mpm-4.1.26.dist-info/METADATA +0 -332
  789. claude_mpm-4.1.26.dist-info/RECORD +0 -606
  790. {claude_mpm-4.1.26.dist-info → claude_mpm-5.0.9.dist-info}/WHEEL +0 -0
  791. {claude_mpm-4.1.26.dist-info → claude_mpm-5.0.9.dist-info}/licenses/LICENSE +0 -0
  792. {claude_mpm-4.1.26.dist-info → claude_mpm-5.0.9.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,1872 @@
1
+ """Interactive Agent Creation Wizard for Claude MPM.
2
+
3
+ This module provides a step-by-step interactive wizard for creating and managing
4
+ local agents with user-friendly prompts, intelligent defaults, and validation.
5
+ """
6
+
7
+ import json
8
+ import re
9
+ import sys
10
+ from pathlib import Path
11
+ from typing import Any, Dict, List, Optional, Tuple
12
+
13
+ import questionary
14
+ from questionary import Style
15
+
16
+ from claude_mpm.core.logging_config import get_logger
17
+ from claude_mpm.services.agents.local_template_manager import (
18
+ LocalAgentTemplate,
19
+ LocalAgentTemplateManager,
20
+ )
21
+ from claude_mpm.utils.agent_filters import apply_all_filters
22
+
23
+ logger = get_logger(__name__)
24
+
25
+ # Questionary style matching Rich cyan theme (consistent with configure.py)
26
+ QUESTIONARY_STYLE = Style(
27
+ [
28
+ ("selected", "fg:cyan bold"),
29
+ ("pointer", "fg:cyan bold"),
30
+ ("highlighted", "fg:cyan"),
31
+ ("question", "fg:cyan bold"),
32
+ ]
33
+ )
34
+
35
+
36
+ class AgentWizard:
37
+ """
38
+ Interactive wizard for agent creation and management.
39
+
40
+ DEPRECATED: This interface has been superseded by the unified
41
+ configuration interface. Please use 'claude-mpm config' instead.
42
+
43
+ This class is retained for backward compatibility but will be
44
+ removed in a future version.
45
+ """
46
+
47
+ def __init__(self):
48
+ """Initialize the agent wizard."""
49
+ self.manager = LocalAgentTemplateManager()
50
+ self.logger = logger
51
+
52
+ # Initialize remote discovery services
53
+ try:
54
+ from claude_mpm.services.agents.git_source_manager import GitSourceManager
55
+
56
+ self.source_manager = GitSourceManager()
57
+ self.discovery_enabled = True
58
+ self.logger.debug("Remote agent discovery enabled")
59
+ except Exception as e:
60
+ self.logger.warning(f"Failed to initialize remote discovery: {e}")
61
+ self.source_manager = None
62
+ self.discovery_enabled = False
63
+
64
+ def run_interactive_create(self) -> Tuple[bool, str]:
65
+ """Run interactive agent creation wizard.
66
+
67
+ Returns:
68
+ Tuple of (success, message)
69
+ """
70
+ try:
71
+ print("\n" + "=" * 60)
72
+ print("🧙‍♂️ Agent Creation Wizard")
73
+ print("=" * 60)
74
+ print("\nI'll guide you through creating a custom local agent.")
75
+ print("Press Ctrl+C anytime to cancel.\n")
76
+
77
+ # Step 1: Agent ID
78
+ agent_id = self._get_agent_id()
79
+ if not agent_id:
80
+ return False, "Agent creation cancelled"
81
+
82
+ # Step 2: Agent Name
83
+ agent_name = self._get_agent_name(agent_id)
84
+
85
+ # Step 3: Agent Type/Category
86
+ agent_type = self._get_agent_type()
87
+
88
+ # Step 4: Model Selection
89
+ model = self._get_model_choice()
90
+
91
+ # Step 5: Inheritance Option
92
+ parent_agent, _base_template = self._get_inheritance_option()
93
+
94
+ # Step 6: Capabilities Configuration
95
+ capabilities = self._get_capabilities_configuration()
96
+
97
+ # Step 7: Description
98
+ description = self._get_agent_description(agent_type)
99
+
100
+ # Step 8: Instructions
101
+ instructions = self._get_agent_instructions(
102
+ agent_id, agent_type, parent_agent
103
+ )
104
+
105
+ # Step 9: Preview Configuration
106
+ config_preview = self._create_preview_config(
107
+ agent_id,
108
+ agent_name,
109
+ agent_type,
110
+ model,
111
+ parent_agent,
112
+ capabilities,
113
+ description,
114
+ instructions,
115
+ )
116
+
117
+ # Step 10: Confirmation
118
+ if not self._confirm_creation(config_preview):
119
+ return False, "Agent creation cancelled"
120
+
121
+ # Step 11: Create and Save Agent
122
+ template = self._create_agent_template(
123
+ agent_id,
124
+ agent_name,
125
+ agent_type,
126
+ model,
127
+ parent_agent,
128
+ capabilities,
129
+ description,
130
+ instructions,
131
+ )
132
+
133
+ template_file = self.manager.save_local_template(template, "project")
134
+
135
+ print(f"\n✅ Agent '{agent_id}' created successfully!")
136
+ print(f"📁 Saved to: {template_file}")
137
+ print("\n💡 Next steps:")
138
+ print(
139
+ f" • Deploy: claude-mpm agent-manager deploy-local --agent-id {agent_id}"
140
+ )
141
+ print(f" • Edit: Edit {template_file} directly")
142
+ print(f" • Test: claude-mpm run --agent {agent_id}")
143
+
144
+ return True, f"Agent '{agent_id}' created successfully"
145
+
146
+ except KeyboardInterrupt:
147
+ print("\n\n❌ Agent creation cancelled by user")
148
+ return False, "Agent creation cancelled"
149
+ except Exception as e:
150
+ error_msg = f"Failed to create agent: {e}"
151
+ self.logger.error(error_msg, exc_info=True)
152
+ return False, error_msg
153
+
154
+ def _merge_agent_sources(self) -> List[Dict[str, Any]]:
155
+ """
156
+ Merge agents from all sources with precedence: local > discovered.
157
+
158
+ Returns list of agents with metadata:
159
+ {
160
+ "agent_id": "engineer/backend/python-engineer",
161
+ "name": "Python Engineer",
162
+ "description": "...",
163
+ "source_type": "system" | "project",
164
+ "source_identifier": "bobmatnyc/claude-mpm-agents",
165
+ "category": "engineer/backend",
166
+ "deployed": True | False,
167
+ "path": "/path/to/agent.md"
168
+ }
169
+ """
170
+ agents = {}
171
+
172
+ # Get discovered agents (system/user sources)
173
+ if self.discovery_enabled and self.source_manager:
174
+ try:
175
+ discovered = self.source_manager.list_cached_agents()
176
+ self.logger.debug(f"Discovered {len(discovered)} remote agents")
177
+
178
+ for agent in discovered:
179
+ agent_id = agent.get("agent_id", "")
180
+ if not agent_id:
181
+ continue
182
+
183
+ # Extract metadata
184
+ metadata = agent.get("metadata", {})
185
+ agents[agent_id] = {
186
+ "agent_id": agent_id,
187
+ "name": metadata.get("name", agent_id),
188
+ "description": metadata.get("description", ""),
189
+ "source_type": "system",
190
+ "source_identifier": agent.get("source", "unknown"),
191
+ "category": agent.get("category", ""),
192
+ "deployed": False, # Will be updated below
193
+ "path": agent.get("path", agent.get("source_file", "")),
194
+ }
195
+ except Exception as e:
196
+ self.logger.warning(f"Failed to discover remote agents: {e}")
197
+
198
+ # Get local agents (project-level, highest precedence)
199
+ local_templates = self.manager.list_local_templates()
200
+ for template in local_templates:
201
+ agent_id = template.agent_id
202
+ agents[agent_id] = {
203
+ "agent_id": agent_id,
204
+ "name": template.metadata.get("name", agent_id),
205
+ "description": template.metadata.get("description", ""),
206
+ "source_type": "project",
207
+ "source_identifier": "local",
208
+ "category": template.metadata.get("category", ""),
209
+ "deployed": True, # Local templates are deployed
210
+ "path": str(self._get_template_path(template)),
211
+ }
212
+
213
+ # Check deployment status for discovered agents
214
+ deployed_dir = Path.cwd() / ".claude" / "agents"
215
+ if deployed_dir.exists():
216
+ for agent_id, agent_data in agents.items():
217
+ deployed_file = deployed_dir / f"{agent_id.replace('/', '-')}.md"
218
+ # Also check hierarchical path
219
+ deployed_file_alt = deployed_dir / f"{agent_id.split('/')[-1]}.md"
220
+ if deployed_file.exists() or deployed_file_alt.exists():
221
+ agent_data["deployed"] = True
222
+
223
+ # Filter BASE_AGENT from all agent lists (1M-502 Phase 1)
224
+ # BASE_AGENT is a build tool, not a deployable agent
225
+ agent_list = list(agents.values())
226
+ return apply_all_filters(agent_list, filter_base=True, filter_deployed=False)
227
+
228
+ def run_interactive_manage(self) -> Tuple[bool, str]:
229
+ """Run interactive agent management menu.
230
+
231
+ Returns:
232
+ Tuple of (success, message)
233
+ """
234
+ try:
235
+ while True:
236
+ # Get merged agents from all sources
237
+ all_agents = self._merge_agent_sources()
238
+
239
+ print("\n" + "=" * 60)
240
+ print("🔧 Agent Management Menu")
241
+ print("=" * 60)
242
+
243
+ if not all_agents:
244
+ print(
245
+ "\n📭 No agents found. Configure sources with 'claude-mpm agents discover'"
246
+ )
247
+ print("\n1. Create new agent")
248
+ print("2. Import agents")
249
+ print("3. Exit")
250
+
251
+ choice = input("\nSelect option [1-3]: ").strip()
252
+
253
+ if choice == "1":
254
+ return self.run_interactive_create()
255
+ if choice == "2":
256
+ return self._interactive_import()
257
+ if choice == "3":
258
+ return True, "Management menu exited"
259
+ print("❌ Invalid choice. Please try again.")
260
+ continue
261
+
262
+ # Show existing agents in a table
263
+ print(f"\n📋 Found {len(all_agents)} agent(s):\n")
264
+ print(
265
+ f"{'#':<4} {'Agent ID':<40} {'Name':<25} {'Source':<20} {'Status':<10}"
266
+ )
267
+ print("-" * 105)
268
+
269
+ for i, agent in enumerate(all_agents, 1):
270
+ agent_id = agent["agent_id"]
271
+ name = (
272
+ agent["name"][:24] if len(agent["name"]) > 24 else agent["name"]
273
+ )
274
+ source_label = (
275
+ f"[{agent['source_type']}] {agent['source_identifier']}"[:19]
276
+ )
277
+ status = "✓ Deployed" if agent["deployed"] else "Available"
278
+
279
+ print(
280
+ f"{i:<4} {agent_id:<40} {name:<25} {source_label:<20} {status:<10}"
281
+ )
282
+
283
+ # Build menu choices with arrow-key navigation
284
+ menu_choices = []
285
+
286
+ # Add agent viewing options (1-N)
287
+ for i, agent in enumerate(all_agents, 1):
288
+ menu_choices.append(f"{i}. View agent: {agent['agent_id']}")
289
+
290
+ # Add action options
291
+ menu_choices.append(f"{len(all_agents) + 1}. Deploy agent")
292
+ menu_choices.append(f"{len(all_agents) + 2}. Create new agent")
293
+ menu_choices.append(f"{len(all_agents) + 3}. Delete agent(s)")
294
+ menu_choices.append(f"{len(all_agents) + 4}. Import agents")
295
+ menu_choices.append(f"{len(all_agents) + 5}. Export all agents")
296
+
297
+ if self.discovery_enabled:
298
+ menu_choices.append(
299
+ f"{len(all_agents) + 6}. Browse & filter agents"
300
+ )
301
+ menu_choices.append(f"{len(all_agents) + 7}. Deploy preset")
302
+ menu_choices.append(f"{len(all_agents) + 8}. Manage agent sources")
303
+ menu_choices.append(f"{len(all_agents) + 9}. Exit")
304
+ exit_num = len(all_agents) + 9
305
+ else:
306
+ menu_choices.append(f"{len(all_agents) + 6}. Exit")
307
+ exit_num = len(all_agents) + 6
308
+
309
+ choice = questionary.select(
310
+ "Agent Management Menu:",
311
+ choices=menu_choices,
312
+ style=QUESTIONARY_STYLE,
313
+ ).ask()
314
+
315
+ if not choice: # User pressed Esc
316
+ return True, "Management menu exited"
317
+
318
+ # Parse choice number from "N. Description" format
319
+ choice_num = int(choice.split(".")[0])
320
+
321
+ if 1 <= choice_num <= len(all_agents):
322
+ # View agent details
323
+ selected_agent = all_agents[choice_num - 1]
324
+ self._show_agent_details(selected_agent)
325
+ continue
326
+ if choice_num == len(all_agents) + 1:
327
+ self._deploy_agent_interactive(all_agents)
328
+ elif choice_num == len(all_agents) + 2:
329
+ _, message = self.run_interactive_create()
330
+ if message:
331
+ print(f"\n{message}")
332
+ continue
333
+ elif choice_num == len(all_agents) + 3:
334
+ local_templates = self.manager.list_local_templates()
335
+ _, message = self._interactive_delete_menu(local_templates)
336
+ if message:
337
+ print(f"\n{message}")
338
+ continue
339
+ elif choice_num == len(all_agents) + 4:
340
+ _, message = self._interactive_import()
341
+ if message:
342
+ print(f"\n{message}")
343
+ continue
344
+ elif choice_num == len(all_agents) + 5:
345
+ _success, message = self._interactive_export()
346
+ if message:
347
+ print(f"\n{message}")
348
+ continue
349
+ elif choice_num == len(all_agents) + 6 and self.discovery_enabled:
350
+ self._browse_agents_interactive()
351
+ continue
352
+ elif choice_num == len(all_agents) + 7 and self.discovery_enabled:
353
+ self._deploy_preset_interactive()
354
+ continue
355
+ elif choice_num == len(all_agents) + 8 and self.discovery_enabled:
356
+ self._manage_sources_interactive()
357
+ continue
358
+ elif choice_num == exit_num:
359
+ return True, "Management menu exited"
360
+ else:
361
+ print("❌ Invalid choice. Please try again.")
362
+
363
+ except KeyboardInterrupt:
364
+ print("\n\n❌ Management menu cancelled")
365
+ return False, "Management cancelled"
366
+ except Exception as e:
367
+ error_msg = f"Management menu error: {e}"
368
+ self.logger.error(error_msg, exc_info=True)
369
+ return False, error_msg
370
+
371
+ def _get_agent_id(self) -> Optional[str]:
372
+ """Get and validate agent ID from user."""
373
+ while True:
374
+ agent_id = input(
375
+ "1. What would you like to name your agent?\n Agent ID (lowercase, hyphens): "
376
+ ).strip()
377
+
378
+ if not agent_id:
379
+ print("❌ Agent ID is required.")
380
+ continue
381
+
382
+ # Validate format
383
+ if (
384
+ not re.match(r"^[a-z][a-z0-9-]*[a-z0-9]$", agent_id)
385
+ or len(agent_id) > 50
386
+ ):
387
+ print("❌ Agent ID must:")
388
+ print(" • Start with a letter")
389
+ print(" • Contain only lowercase letters, numbers, and hyphens")
390
+ print(" • End with a letter or number")
391
+ print(" • Be 50 characters or less")
392
+ continue
393
+
394
+ # Check for conflicts
395
+ existing_template = self.manager.get_local_template(agent_id)
396
+ if existing_template:
397
+ print(f"❌ Agent '{agent_id}' already exists.")
398
+ overwrite = (
399
+ input(" Overwrite existing agent? [y/N]: ").strip().lower()
400
+ )
401
+ if overwrite not in ["y", "yes"]:
402
+ continue
403
+
404
+ return agent_id
405
+
406
+ def _get_agent_name(self, agent_id: str) -> str:
407
+ """Get agent display name."""
408
+ default_name = agent_id.replace("-", " ").title()
409
+ agent_name = input(
410
+ f"\n2. What should be the display name? [{default_name}]: "
411
+ ).strip()
412
+ return agent_name or default_name
413
+
414
+ def _get_agent_type(self) -> str:
415
+ """Get agent type/category from user."""
416
+ print("\n3. What type of agent is this?")
417
+ agent_types = [
418
+ (
419
+ "research",
420
+ "Research & Analysis",
421
+ "Gathering information, analyzing data, investigating topics",
422
+ ),
423
+ (
424
+ "engineer",
425
+ "Implementation & Engineering",
426
+ "Writing code, building features, technical development",
427
+ ),
428
+ (
429
+ "qa",
430
+ "Quality Assurance & Testing",
431
+ "Testing code, reviewing quality, finding bugs",
432
+ ),
433
+ (
434
+ "docs",
435
+ "Documentation & Writing",
436
+ "Creating docs, writing content, technical writing",
437
+ ),
438
+ (
439
+ "ops",
440
+ "Operations & Deployment",
441
+ "DevOps, deployment, system administration",
442
+ ),
443
+ ("custom", "Custom/Other", "Specialized or unique functionality"),
444
+ ]
445
+
446
+ for i, (_type_id, name, desc) in enumerate(agent_types, 1):
447
+ print(f" [{i}] {name}")
448
+ print(f" {desc}")
449
+
450
+ while True:
451
+ choice = input(f"\nSelect type [1-{len(agent_types)}]: ").strip()
452
+
453
+ try:
454
+ choice_num = int(choice)
455
+ if 1 <= choice_num <= len(agent_types):
456
+ return agent_types[choice_num - 1][0]
457
+ except ValueError:
458
+ pass
459
+
460
+ print("❌ Invalid choice. Please select a number from the list.")
461
+
462
+ def _get_model_choice(self) -> str:
463
+ """Get model selection from user."""
464
+ print("\n4. Which model should this agent use?")
465
+ models = [
466
+ (
467
+ "sonnet",
468
+ "claude-3-sonnet (balanced - recommended)",
469
+ "Good balance of capability and speed",
470
+ ),
471
+ (
472
+ "opus",
473
+ "claude-3-opus (powerful)",
474
+ "Most capable but slower and more expensive",
475
+ ),
476
+ ("haiku", "claude-3-haiku (fast)", "Fastest and most economical"),
477
+ ]
478
+
479
+ for i, (_model_id, name, desc) in enumerate(models, 1):
480
+ print(f" [{i}] {name}")
481
+ print(f" {desc}")
482
+
483
+ while True:
484
+ choice = input(f"\nSelect model [1-{len(models)}] [1]: ").strip() or "1"
485
+
486
+ try:
487
+ choice_num = int(choice)
488
+ if 1 <= choice_num <= len(models):
489
+ return models[choice_num - 1][0]
490
+ except ValueError:
491
+ pass
492
+
493
+ print("❌ Invalid choice. Please select a number from the list.")
494
+
495
+ def _get_inheritance_option(self) -> Tuple[Optional[str], Optional[Dict]]:
496
+ """Get inheritance option from user."""
497
+ print("\n5. Would you like to inherit from an existing agent?")
498
+ print(" [1] No, start fresh")
499
+ print(" [2] Yes, inherit from system agent")
500
+
501
+ while True:
502
+ choice = input("\nSelect option [1-2] [1]: ").strip() or "1"
503
+
504
+ if choice == "1":
505
+ return None, None
506
+ if choice == "2":
507
+ return self._select_system_agent()
508
+ print("❌ Invalid choice. Please select 1 or 2.")
509
+
510
+ def _select_system_agent(self) -> Tuple[Optional[str], Optional[Dict]]:
511
+ """Let user select a system agent to inherit from."""
512
+ try:
513
+ # Get available system agents
514
+ from claude_mpm.services.agents.agent_builder import AgentBuilderService
515
+
516
+ builder = AgentBuilderService()
517
+ templates = builder.list_available_templates()
518
+
519
+ if not templates:
520
+ print("❌ No system agents found to inherit from.")
521
+ return None, None
522
+
523
+ print("\n Select system agent to inherit from:")
524
+ for i, template in enumerate(templates, 1):
525
+ name = template.get("name", template.get("id", "Unknown"))
526
+ description = template.get("description", "")
527
+ print(f" [{i}] {name}")
528
+ if description:
529
+ print(f" {description[:80]}...")
530
+
531
+ while True:
532
+ choice = input(f"\n Select agent [1-{len(templates)}]: ").strip()
533
+
534
+ try:
535
+ choice_num = int(choice)
536
+ if 1 <= choice_num <= len(templates):
537
+ selected = templates[choice_num - 1]
538
+ return selected.get("id"), selected
539
+ except ValueError:
540
+ pass
541
+
542
+ print("❌ Invalid choice. Please select a number from the list.")
543
+
544
+ except Exception as e:
545
+ self.logger.warning(f"Failed to load system agents: {e}")
546
+ print("❌ Could not load system agents for inheritance.")
547
+ return None, None
548
+
549
+ def _get_capabilities_configuration(self) -> Dict[str, Any]:
550
+ """Get capabilities configuration from user."""
551
+ print("\n6. What additional capabilities should this agent have?")
552
+
553
+ capabilities_options = [
554
+ ("code_analysis", "Code analysis and review"),
555
+ ("test_generation", "Test generation and validation"),
556
+ ("security_scanning", "Security analysis and scanning"),
557
+ ("performance_profiling", "Performance analysis and optimization"),
558
+ ("documentation", "Documentation generation"),
559
+ ("api_design", "API design and documentation"),
560
+ ("data_processing", "Data processing and analysis"),
561
+ ("web_scraping", "Web scraping and data extraction"),
562
+ ]
563
+
564
+ print(" Select capabilities (enter multiple numbers separated by spaces):")
565
+ for i, (_cap_id, desc) in enumerate(capabilities_options, 1):
566
+ print(f" [{i}] {desc}")
567
+
568
+ selected_capabilities = []
569
+ while True:
570
+ choices = input(
571
+ f"\nSelect capabilities [1-{len(capabilities_options)}] (space-separated) [none]: "
572
+ ).strip()
573
+
574
+ if not choices or choices.lower() == "none":
575
+ break
576
+
577
+ try:
578
+ choice_nums = [int(x) for x in choices.split()]
579
+ valid_choices = []
580
+
581
+ for num in choice_nums:
582
+ if 1 <= num <= len(capabilities_options):
583
+ selected_capabilities.append(capabilities_options[num - 1][0])
584
+ valid_choices.append(str(num))
585
+
586
+ if valid_choices:
587
+ print(f"✅ Selected: {', '.join(selected_capabilities)}")
588
+ break
589
+ print("❌ No valid choices selected.")
590
+ except ValueError:
591
+ print("❌ Please enter numbers separated by spaces.")
592
+
593
+ return {"specializations": selected_capabilities}
594
+
595
+ def _get_agent_description(self, agent_type: str) -> str:
596
+ """Get agent description from user."""
597
+ type_examples = {
598
+ "research": "Specializes in analyzing market trends and competitive intelligence",
599
+ "engineer": "Focused on building scalable web applications using React and Node.js",
600
+ "qa": "Expert in automated testing and code quality assurance",
601
+ "docs": "Creates clear technical documentation and user guides",
602
+ "ops": "Manages CI/CD pipelines and cloud infrastructure",
603
+ "custom": "Handles specialized domain-specific tasks",
604
+ }
605
+
606
+ example = type_examples.get(agent_type, "Performs specialized tasks")
607
+
608
+ print("\n7. Describe this agent's specialty (one line):")
609
+ print(f" Example: {example}")
610
+
611
+ while True:
612
+ description = input("\n Description: ").strip()
613
+
614
+ if not description:
615
+ print("❌ Description is required.")
616
+ continue
617
+
618
+ if len(description) > 200:
619
+ print("❌ Description should be 200 characters or less.")
620
+ continue
621
+
622
+ return description
623
+
624
+ def _get_agent_instructions(
625
+ self, agent_id: str, agent_type: str, parent_agent: Optional[str]
626
+ ) -> str:
627
+ """Get agent instructions from user."""
628
+ print("\n8. Agent Instructions:")
629
+
630
+ if parent_agent:
631
+ print(f" Since you're inheriting from '{parent_agent}', you can:")
632
+ print(" [1] Use default inherited instructions")
633
+ print(" [2] Add custom instructions")
634
+ print(" [3] Write completely new instructions")
635
+
636
+ choice = input("\n Select option [1-3] [1]: ").strip() or "1"
637
+
638
+ if choice == "1":
639
+ return f"Extends the {parent_agent} agent with project-specific enhancements."
640
+ if choice == "2":
641
+ additional = input("\n Enter additional instructions: ").strip()
642
+ return f"Extends the {parent_agent} agent.\n\nAdditional instructions:\n{additional}"
643
+
644
+ # Get custom instructions
645
+ print(" Enter custom instructions for this agent:")
646
+ print(" (Type 'DONE' on a new line when finished)")
647
+
648
+ lines = []
649
+ while True:
650
+ line = input(" ")
651
+ if line.strip() == "DONE":
652
+ break
653
+ lines.append(line)
654
+
655
+ instructions = "\n".join(lines).strip()
656
+
657
+ if not instructions:
658
+ # Provide default based on type
659
+ type_defaults = {
660
+ "research": f"You are {agent_id}, a research and analysis agent. Focus on gathering accurate information, analyzing data, and providing well-researched insights.",
661
+ "engineer": f"You are {agent_id}, a software engineering agent. Focus on writing clean, efficient code and implementing technical solutions.",
662
+ "qa": f"You are {agent_id}, a quality assurance agent. Focus on testing, code review, and ensuring high quality standards.",
663
+ "docs": f"You are {agent_id}, a documentation agent. Focus on creating clear, comprehensive documentation and technical writing.",
664
+ "ops": f"You are {agent_id}, an operations agent. Focus on deployment, infrastructure, and system administration tasks.",
665
+ "custom": f"You are {agent_id}, a specialized agent. Focus on your specific domain expertise.",
666
+ }
667
+ instructions = type_defaults.get(
668
+ agent_type, f"You are {agent_id}, a specialized agent."
669
+ )
670
+
671
+ return instructions
672
+
673
+ def _create_preview_config(
674
+ self,
675
+ agent_id: str,
676
+ agent_name: str,
677
+ agent_type: str,
678
+ model: str,
679
+ parent_agent: Optional[str],
680
+ capabilities: Dict,
681
+ description: str,
682
+ instructions: str,
683
+ ) -> Dict[str, Any]:
684
+ """Create preview configuration dictionary."""
685
+ config = {
686
+ "agent_id": agent_id,
687
+ "name": agent_name,
688
+ "type": agent_type,
689
+ "model": model,
690
+ "description": description,
691
+ "capabilities": capabilities.get("specializations", []),
692
+ "instructions_preview": (
693
+ instructions[:100] + "..." if len(instructions) > 100 else instructions
694
+ ),
695
+ }
696
+
697
+ if parent_agent:
698
+ config["inherits_from"] = parent_agent
699
+
700
+ return config
701
+
702
+ def _confirm_creation(self, config: Dict[str, Any]) -> bool:
703
+ """Show preview and get confirmation from user."""
704
+ print("\n" + "=" * 60)
705
+ print("📋 Agent Configuration Preview")
706
+ print("=" * 60)
707
+
708
+ print(f"Agent ID: {config['agent_id']}")
709
+ print(f"Name: {config['name']}")
710
+ print(f"Type: {config['type']}")
711
+ print(f"Model: {config['model']}")
712
+ print(f"Description: {config['description']}")
713
+
714
+ if config.get("inherits_from"):
715
+ print(f"Inherits: {config['inherits_from']}")
716
+
717
+ if config.get("capabilities"):
718
+ print(f"Capabilities: {', '.join(config['capabilities'])}")
719
+
720
+ print("\nInstructions Preview:")
721
+ print(f" {config['instructions_preview']}")
722
+
723
+ print("\n" + "=" * 60)
724
+
725
+ while True:
726
+ confirm = input("\nCreate this agent? [Y/n]: ").strip().lower()
727
+
728
+ if confirm in ["", "y", "yes"]:
729
+ return True
730
+ if confirm in ["n", "no"]:
731
+ return False
732
+ print("❌ Please enter 'y' for yes or 'n' for no.")
733
+
734
+ def _create_agent_template(
735
+ self,
736
+ agent_id: str,
737
+ agent_name: str,
738
+ agent_type: str,
739
+ model: str,
740
+ parent_agent: Optional[str],
741
+ capabilities: Dict,
742
+ description: str,
743
+ instructions: str,
744
+ ) -> LocalAgentTemplate:
745
+ """Create the actual agent template."""
746
+ return self.manager.create_local_template(
747
+ agent_id=agent_id,
748
+ name=agent_name,
749
+ description=description,
750
+ instructions=instructions,
751
+ model=model,
752
+ tools="*",
753
+ parent_agent=parent_agent,
754
+ tier="project",
755
+ )
756
+
757
+ def _manage_single_agent(self, template: LocalAgentTemplate) -> Tuple[bool, str]:
758
+ """Manage a single agent."""
759
+ print(f"\n🔧 Managing Agent: {template.agent_id}")
760
+ print(f" Name: {template.metadata.get('name', template.agent_id)}")
761
+ print(f" Tier: {template.tier}")
762
+ print(f" Version: {template.agent_version}")
763
+
764
+ print("\n1. View details")
765
+ print("2. Edit configuration")
766
+ print("3. Deploy agent")
767
+ print("4. Delete agent")
768
+ print("5. Export agent")
769
+ print("6. Back to menu")
770
+
771
+ choice = input("\nSelect option [1-6]: ").strip()
772
+
773
+ if choice == "1":
774
+ return self._view_agent_details(template)
775
+ if choice == "2":
776
+ return self._edit_agent_config(template)
777
+ if choice == "3":
778
+ return self._deploy_single_agent(template)
779
+ if choice == "4":
780
+ return self._delete_agent(template)
781
+ if choice == "5":
782
+ return self._export_single_agent(template)
783
+ if choice == "6":
784
+ return True, "Back to menu"
785
+ return False, "Invalid choice"
786
+
787
+ def _view_agent_details(self, template: LocalAgentTemplate) -> Tuple[bool, str]:
788
+ """View detailed agent information."""
789
+ print(f"\n📄 Agent Details: {template.agent_id}")
790
+ print("=" * 50)
791
+ print(f"Name: {template.metadata.get('name', template.agent_id)}")
792
+ print(f"Version: {template.agent_version}")
793
+ print(f"Author: {template.author}")
794
+ print(f"Tier: {template.tier}")
795
+ print(f"Model: {template.capabilities.get('model', 'unknown')}")
796
+ print(f"Tools: {template.capabilities.get('tools', '*')}")
797
+
798
+ if template.parent_agent:
799
+ print(f"Inherits: {template.parent_agent}")
800
+
801
+ print("\nDescription:")
802
+ print(f" {template.metadata.get('description', 'No description')}")
803
+
804
+ print("\nInstructions:")
805
+ print(
806
+ f" {template.instructions[:200]}{'...' if len(template.instructions) > 200 else ''}"
807
+ )
808
+
809
+ input("\nPress Enter to continue...")
810
+ return True, "Agent details viewed"
811
+
812
+ def _edit_agent_config(self, template: LocalAgentTemplate) -> Tuple[bool, str]:
813
+ """Edit agent configuration."""
814
+ print(f"\n✏️ Editing Agent: {template.agent_id}")
815
+ print("This will open the JSON file in your default editor.")
816
+
817
+ confirm = input("Continue? [y/N]: ").strip().lower()
818
+ if confirm not in ["y", "yes"]:
819
+ return False, "Edit cancelled"
820
+
821
+ # Get template file path
822
+ if template.tier == "project":
823
+ template_file = (
824
+ self.manager.project_agents_dir / f"{template.agent_id}.json"
825
+ )
826
+ else:
827
+ template_file = self.manager.user_agents_dir / f"{template.agent_id}.json"
828
+
829
+ # Open in editor
830
+ import os
831
+ import subprocess
832
+
833
+ editor = os.environ.get("EDITOR", "nano")
834
+ try:
835
+ subprocess.run([editor, str(template_file)], check=True)
836
+ return True, f"Agent {template.agent_id} edited"
837
+ except Exception as e:
838
+ return False, f"Failed to open editor: {e}"
839
+
840
+ def _deploy_single_agent(self, template: LocalAgentTemplate) -> Tuple[bool, str]:
841
+ """Deploy a single agent."""
842
+ try:
843
+ from claude_mpm.services.agents.deployment.local_template_deployment import (
844
+ LocalTemplateDeploymentService,
845
+ )
846
+
847
+ service = LocalTemplateDeploymentService()
848
+ success = service.deploy_single_local_template(
849
+ template.agent_id, force=True
850
+ )
851
+
852
+ if success:
853
+ return True, f"Agent '{template.agent_id}' deployed successfully"
854
+ return False, f"Failed to deploy agent '{template.agent_id}'"
855
+
856
+ except Exception as e:
857
+ return False, f"Deployment error: {e}"
858
+
859
+ def _delete_agent(self, template: LocalAgentTemplate) -> Tuple[bool, str]:
860
+ """Delete an agent with comprehensive options."""
861
+ print(f"\n🗑️ Delete Agent: {template.agent_id}")
862
+ print(f" Name: {template.metadata.get('name', template.agent_id)}")
863
+ print(f" Tier: {template.tier}")
864
+ print(f" Location: {self._get_template_path(template)}")
865
+
866
+ # Check if deployed
867
+ deployment_file = Path.cwd() / ".claude" / "agents" / f"{template.agent_id}.md"
868
+ if deployment_file.exists():
869
+ print(f" Deployed: Yes ({deployment_file})")
870
+ else:
871
+ print(" Deployed: No")
872
+
873
+ print("\nDelete options:")
874
+ print(" [1] Delete template and deployment")
875
+ print(" [2] Delete template only (keep deployment)")
876
+ print(" [3] Cancel")
877
+
878
+ option = input("\nSelect option [1-3]: ").strip()
879
+
880
+ if option == "3":
881
+ return False, "Deletion cancelled"
882
+
883
+ delete_deployment = option == "1"
884
+
885
+ # Confirmation
886
+ print("\n⚠️ This will permanently delete:")
887
+ print(f" - Template: {self._get_template_path(template)}")
888
+ if delete_deployment and deployment_file.exists():
889
+ print(f" - Deployment: {deployment_file}")
890
+
891
+ # Ask about backup
892
+ backup_choice = (
893
+ input("\nCreate backup before deletion? [y/N]: ").strip().lower()
894
+ )
895
+ backup_first = backup_choice in ["y", "yes"]
896
+
897
+ confirm = input("\nAre you sure? Type 'DELETE' to confirm: ").strip()
898
+
899
+ if confirm != "DELETE":
900
+ return False, "Deletion cancelled"
901
+
902
+ # Perform deletion
903
+ result = self.manager.delete_local_template(
904
+ agent_id=template.agent_id,
905
+ tier=template.tier,
906
+ delete_deployment=delete_deployment,
907
+ backup_first=backup_first,
908
+ )
909
+
910
+ if result["success"]:
911
+ message = f"✅ Agent '{template.agent_id}' deleted successfully"
912
+ if result["backup_location"]:
913
+ message += f"\n Backup saved to: {result['backup_location']}"
914
+ message += f"\n Removed {len(result['deleted_files'])} file(s)"
915
+ return True, message
916
+ errors = "\n".join(result["errors"])
917
+ return False, f"Failed to delete agent:\n{errors}"
918
+
919
+ def _export_single_agent(self, template: LocalAgentTemplate) -> Tuple[bool, str]:
920
+ """Export a single agent."""
921
+ output_dir = Path("./exported-agents")
922
+ output_dir.mkdir(exist_ok=True)
923
+
924
+ output_file = output_dir / f"{template.agent_id}.json"
925
+
926
+ with output_file.open("w") as f:
927
+ json.dump(template.to_json(), f, indent=2)
928
+
929
+ return True, f"Agent exported to {output_file}"
930
+
931
+ def _interactive_import(self) -> Tuple[bool, str]:
932
+ """Interactive agent import."""
933
+ input_dir = input("\nEnter directory path to import from: ").strip()
934
+
935
+ if not input_dir:
936
+ return False, "Import cancelled"
937
+
938
+ input_path = Path(input_dir)
939
+ if not input_path.exists():
940
+ return False, f"Directory does not exist: {input_path}"
941
+
942
+ # Select tier
943
+ print("\nImport to which tier?")
944
+ print(" [1] Project (recommended)")
945
+ print(" [2] User")
946
+
947
+ tier_choice = input("Select tier [1-2] [1]: ").strip() or "1"
948
+ tier = "project" if tier_choice == "1" else "user"
949
+
950
+ count = self.manager.import_local_templates(input_path, tier)
951
+ return True, f"Imported {count} agents from {input_path}"
952
+
953
+ def _interactive_export(self) -> Tuple[bool, str]:
954
+ """Interactive agent export."""
955
+ output_dir = input(
956
+ "\nEnter directory path to export to [./exported-agents]: "
957
+ ).strip()
958
+
959
+ if not output_dir:
960
+ output_dir = "./exported-agents"
961
+
962
+ output_path = Path(output_dir)
963
+ count = self.manager.export_local_templates(output_path)
964
+ return True, f"Exported {count} agents to {output_path}"
965
+
966
+ def _get_template_path(self, template: LocalAgentTemplate) -> Path:
967
+ """Get the file path for a template."""
968
+ if template.tier == "project":
969
+ return self.manager.project_agents_dir / f"{template.agent_id}.json"
970
+ return self.manager.user_agents_dir / f"{template.agent_id}.json"
971
+
972
+ def _interactive_delete_menu(self, templates: list) -> Tuple[bool, str]:
973
+ """Interactive deletion menu for multiple agents."""
974
+ print("\n🗑️ Delete Agents")
975
+ print("=" * 50)
976
+
977
+ if not templates:
978
+ return False, "No agents available to delete"
979
+
980
+ print("\nAvailable agents:")
981
+ for i, template in enumerate(templates, 1):
982
+ tier_icon = "🏢" if template.tier == "project" else "👤"
983
+ print(
984
+ f" [{i}] {tier_icon} {template.agent_id} - {template.metadata.get('name', template.agent_id)}"
985
+ )
986
+
987
+ print("\n[all] Select all agents")
988
+ print("[0] Cancel")
989
+
990
+ selection = input(
991
+ "\nSelect agents to delete (comma-separated numbers or 'all'): "
992
+ ).strip()
993
+
994
+ if selection == "0" or not selection:
995
+ return False, "Deletion cancelled"
996
+
997
+ # Parse selection
998
+ selected_templates = []
999
+ if selection.lower() == "all":
1000
+ selected_templates = templates
1001
+ else:
1002
+ try:
1003
+ indices = [int(x.strip()) - 1 for x in selection.split(",")]
1004
+ for idx in indices:
1005
+ if 0 <= idx < len(templates):
1006
+ selected_templates.append(templates[idx])
1007
+ else:
1008
+ print(f"⚠️ Invalid selection: {idx + 1}")
1009
+ except ValueError:
1010
+ return False, "Invalid selection format"
1011
+
1012
+ if not selected_templates:
1013
+ return False, "No valid agents selected"
1014
+
1015
+ # Show what will be deleted
1016
+ print(f"\n📋 Selected {len(selected_templates)} agent(s) for deletion:")
1017
+ for template in selected_templates:
1018
+ tier_icon = "🏢" if template.tier == "project" else "👤"
1019
+ print(f" - {tier_icon} {template.agent_id}")
1020
+
1021
+ # Deletion options
1022
+ print("\nDelete options:")
1023
+ print(" [1] Delete templates and deployments")
1024
+ print(" [2] Delete templates only (keep deployments)")
1025
+ print(" [3] Cancel")
1026
+
1027
+ option = input("\nSelect option [1-3]: ").strip()
1028
+
1029
+ if option == "3":
1030
+ return False, "Deletion cancelled"
1031
+
1032
+ delete_deployment = option == "1"
1033
+
1034
+ # Ask about backup
1035
+ backup_choice = (
1036
+ input("\nCreate backups before deletion? [y/N]: ").strip().lower()
1037
+ )
1038
+ backup_first = backup_choice in ["y", "yes"]
1039
+
1040
+ # Strong confirmation for multiple deletions
1041
+ if len(selected_templates) > 1:
1042
+ print(f"\n⚠️ WARNING: This will delete {len(selected_templates)} agents!")
1043
+
1044
+ confirm = input("\nAre you sure? Type 'DELETE ALL' to confirm: ").strip()
1045
+
1046
+ if confirm != "DELETE ALL":
1047
+ return False, "Deletion cancelled"
1048
+
1049
+ # Perform bulk deletion
1050
+ agent_ids = [t.agent_id for t in selected_templates]
1051
+ results = self.manager.delete_multiple_templates(
1052
+ agent_ids=agent_ids,
1053
+ tier="all", # Check all tiers since we have mixed selection
1054
+ delete_deployment=delete_deployment,
1055
+ backup_first=backup_first,
1056
+ )
1057
+
1058
+ # Format results
1059
+ if results["successful"]:
1060
+ message = (
1061
+ f"✅ Successfully deleted {len(results['successful'])} agent(s):\n"
1062
+ )
1063
+ for agent_id in results["successful"]:
1064
+ message += f" - {agent_id}\n"
1065
+ else:
1066
+ message = ""
1067
+
1068
+ if results["failed"]:
1069
+ message += f"❌ Failed to delete {len(results['failed'])} agent(s):\n"
1070
+ for agent_id in results["failed"]:
1071
+ errors = results["details"][agent_id]["errors"]
1072
+ message += f" - {agent_id}: {', '.join(errors)}\n"
1073
+
1074
+ return len(results["successful"]) > 0, message.strip()
1075
+
1076
+ def _show_agent_details(self, agent: Dict[str, Any]) -> None:
1077
+ """Show detailed information about an agent.
1078
+
1079
+ Args:
1080
+ agent: Agent metadata dictionary
1081
+ """
1082
+ print("\n" + "=" * 60)
1083
+ print(f"📄 Agent Details: {agent['agent_id']}")
1084
+ print("=" * 60)
1085
+ print(f"Name: {agent['name']}")
1086
+ print(f"Category: {agent['category'] or 'N/A'}")
1087
+ print(f"Source: [{agent['source_type']}] {agent['source_identifier']}")
1088
+ print(f"Status: {'✓ Deployed' if agent['deployed'] else 'Available'}")
1089
+ print(f"Path: {agent['path']}")
1090
+
1091
+ if agent["description"]:
1092
+ print("\nDescription:")
1093
+ print(
1094
+ f" {agent['description'][:200]}{'...' if len(agent['description']) > 200 else ''}"
1095
+ )
1096
+
1097
+ input("\nPress Enter to continue...")
1098
+
1099
+ def _deploy_agent_interactive(self, available_agents: List[Dict[str, Any]]):
1100
+ """Interactive agent deployment.
1101
+
1102
+ Args:
1103
+ available_agents: List of all available agents
1104
+ """
1105
+ # Filter to non-deployed agents using improved detection (1M-502 Phase 1)
1106
+ # This checks both .claude-mpm/agents/ and .claude/agents/
1107
+ deployable = apply_all_filters(
1108
+ available_agents, filter_base=True, filter_deployed=True
1109
+ )
1110
+
1111
+ if not deployable:
1112
+ print("\n✅ All agents are already deployed!")
1113
+ input("\nPress Enter to continue...")
1114
+ return
1115
+
1116
+ print("\n" + "=" * 60)
1117
+ print("📦 Deploy Agent")
1118
+ print("=" * 60)
1119
+ print(f"\n{len(deployable)} agent(s) available to deploy:\n")
1120
+
1121
+ # Build agent selection choices with arrow-key navigation
1122
+ agent_choices = [
1123
+ f"{i}. {agent['agent_id']} - {agent['description'][:60]}{'...' if len(agent['description']) > 60 else ''}"
1124
+ for i, agent in enumerate(deployable, 1)
1125
+ ]
1126
+
1127
+ choice = questionary.select(
1128
+ "Select agent to deploy:", choices=agent_choices, style=QUESTIONARY_STYLE
1129
+ ).ask()
1130
+
1131
+ if not choice: # User pressed Esc
1132
+ return
1133
+
1134
+ # Parse agent index from "N. agent_id - description" format
1135
+ idx = int(choice.split(".")[0]) - 1
1136
+ agent = deployable[idx]
1137
+
1138
+ # Deploy agent using deployment service
1139
+ print(f"\n🚀 Deploying {agent['agent_id']}...")
1140
+
1141
+ try:
1142
+ # Use SingleAgentDeployer for deployment
1143
+ from claude_mpm.services.agents.deployment.agent_template_builder import (
1144
+ AgentTemplateBuilder,
1145
+ )
1146
+ from claude_mpm.services.agents.deployment.agent_version_manager import (
1147
+ AgentVersionManager,
1148
+ )
1149
+ from claude_mpm.services.agents.deployment.deployment_results_manager import (
1150
+ DeploymentResultsManager,
1151
+ )
1152
+ from claude_mpm.services.agents.deployment.single_agent_deployer import (
1153
+ SingleAgentDeployer,
1154
+ )
1155
+
1156
+ # Initialize deployment services
1157
+ template_builder = AgentTemplateBuilder()
1158
+ version_manager = AgentVersionManager()
1159
+ results_manager = DeploymentResultsManager(self.logger)
1160
+ deployer = SingleAgentDeployer(
1161
+ template_builder=template_builder,
1162
+ version_manager=version_manager,
1163
+ results_manager=results_manager,
1164
+ logger=self.logger,
1165
+ )
1166
+
1167
+ # Prepare deployment parameters
1168
+ template_path = Path(agent["path"])
1169
+ target_dir = Path.cwd() / ".claude" / "agents"
1170
+
1171
+ # Find base_agent.json in multiple possible locations
1172
+ base_agent_candidates = [
1173
+ Path.home()
1174
+ / ".claude-mpm"
1175
+ / "agents"
1176
+ / "templates"
1177
+ / "base_agent.json",
1178
+ Path.home() / ".claude-mpm" / "cache" / "base_agent.json",
1179
+ Path(__file__).parent.parent.parent
1180
+ / "agents"
1181
+ / "templates"
1182
+ / "base_agent.json",
1183
+ ]
1184
+ base_agent_path = None
1185
+ for candidate in base_agent_candidates:
1186
+ if candidate.exists():
1187
+ base_agent_path = candidate
1188
+ break
1189
+
1190
+ if not base_agent_path:
1191
+ base_agent_path = base_agent_candidates[
1192
+ 0
1193
+ ] # Use default even if not exists
1194
+
1195
+ # Deploy the agent
1196
+ success = deployer.deploy_agent(
1197
+ agent_name=agent["agent_id"],
1198
+ templates_dir=template_path.parent,
1199
+ target_dir=target_dir,
1200
+ base_agent_path=base_agent_path,
1201
+ force_rebuild=True,
1202
+ working_directory=Path.cwd(),
1203
+ )
1204
+
1205
+ if success:
1206
+ print(f"\n✅ Successfully deployed {agent['agent_id']}")
1207
+ else:
1208
+ print(f"\n❌ Failed to deploy {agent['agent_id']}")
1209
+
1210
+ except Exception as e:
1211
+ self.logger.error(f"Deployment failed: {e}", exc_info=True)
1212
+ print(f"\n❌ Deployment error: {e}")
1213
+
1214
+ input("\nPress Enter to continue...")
1215
+
1216
+ def _browse_agents_interactive(self):
1217
+ """Interactive agent browsing with filters."""
1218
+ if not self.discovery_enabled or not self.source_manager:
1219
+ print("\n❌ Discovery service not available")
1220
+ input("\nPress Enter to continue...")
1221
+ return
1222
+
1223
+ while True:
1224
+ print("\n" + "=" * 60)
1225
+ print("🔍 Browse & Filter Agents")
1226
+ print("=" * 60)
1227
+
1228
+ # Show filter menu with arrow-key navigation
1229
+ print("\n[bold]Filter by:[/bold]")
1230
+
1231
+ filter_choices = [
1232
+ "1. Category (engineer/backend, qa, ops, etc.)",
1233
+ "2. Language (python, typescript, rust, etc.)",
1234
+ "3. Framework (react, nextjs, flask, etc.)",
1235
+ "4. Show all agents",
1236
+ "← Back to main menu",
1237
+ ]
1238
+
1239
+ choice = questionary.select(
1240
+ "Browse & Filter Agents:",
1241
+ choices=filter_choices,
1242
+ style=QUESTIONARY_STYLE,
1243
+ ).ask()
1244
+
1245
+ if not choice or "Back" in choice:
1246
+ break
1247
+
1248
+ # Parse choice number if it starts with a digit
1249
+ if choice[0].isdigit():
1250
+ choice_num = choice.split(".")[0]
1251
+ else:
1252
+ break
1253
+
1254
+ filtered_agents = []
1255
+ filter_description = ""
1256
+
1257
+ if choice_num == "1":
1258
+ # Category filtering with arrow-key navigation
1259
+ categories = [
1260
+ "engineer/backend",
1261
+ "engineer/frontend",
1262
+ "qa",
1263
+ "ops",
1264
+ "documentation",
1265
+ "universal",
1266
+ ]
1267
+
1268
+ cat_choices = [f"{idx}. {cat}" for idx, cat in enumerate(categories, 1)]
1269
+
1270
+ cat_choice = questionary.select(
1271
+ "Select category:", choices=cat_choices, style=QUESTIONARY_STYLE
1272
+ ).ask()
1273
+
1274
+ if not cat_choice: # User pressed Esc
1275
+ continue
1276
+
1277
+ # Parse category from "N. category" format
1278
+ cat_idx = int(cat_choice.split(".")[0]) - 1
1279
+ category = categories[cat_idx]
1280
+ all_agents = self._merge_agent_sources()
1281
+ filtered_agents = [
1282
+ a for a in all_agents if a.get("category", "").startswith(category)
1283
+ ]
1284
+ filter_description = f"Category: {category}"
1285
+
1286
+ elif choice_num == "2":
1287
+ # Language filtering (using AUTO-DEPLOY-INDEX if available)
1288
+ language = input(
1289
+ "\nEnter language (python, typescript, rust, go, etc.): "
1290
+ ).strip()
1291
+
1292
+ try:
1293
+ # Find AUTO-DEPLOY-INDEX.md in agent repository
1294
+ from claude_mpm.services.agents.auto_deploy_index_parser import (
1295
+ AutoDeployIndexParser,
1296
+ )
1297
+
1298
+ index_path = (
1299
+ Path.home()
1300
+ / ".claude-mpm"
1301
+ / "cache"
1302
+ / "remote-agents"
1303
+ / "bobmatnyc"
1304
+ / "claude-mpm-agents"
1305
+ / "AUTO-DEPLOY-INDEX.md"
1306
+ )
1307
+ if not index_path.exists():
1308
+ print(
1309
+ f"[yellow]Could not find AUTO-DEPLOY-INDEX.md at: {index_path}[/yellow]"
1310
+ )
1311
+ input("\nPress Enter to continue...")
1312
+ continue
1313
+
1314
+ parser = AutoDeployIndexParser(index_path)
1315
+ lang_agents = parser.get_agents_by_language(language.lower())
1316
+
1317
+ # Get full agent details from discovery
1318
+ all_agents = self._merge_agent_sources()
1319
+ agent_ids = lang_agents.get("core", []) + lang_agents.get(
1320
+ "optional", []
1321
+ )
1322
+ filtered_agents = [
1323
+ a for a in all_agents if a["agent_id"] in agent_ids
1324
+ ]
1325
+ filter_description = f"Language: {language}"
1326
+ except Exception as e:
1327
+ self.logger.error(f"Language filter error: {e}", exc_info=True)
1328
+ print(f"[yellow]Could not filter by language: {e}[/yellow]")
1329
+ input("\nPress Enter to continue...")
1330
+ continue
1331
+
1332
+ elif choice_num == "3":
1333
+ # Framework filtering
1334
+ framework = input(
1335
+ "\nEnter framework (react, nextjs, flask, django, etc.): "
1336
+ ).strip()
1337
+
1338
+ try:
1339
+ from claude_mpm.services.agents.auto_deploy_index_parser import (
1340
+ AutoDeployIndexParser,
1341
+ )
1342
+
1343
+ index_path = (
1344
+ Path.home()
1345
+ / ".claude-mpm"
1346
+ / "cache"
1347
+ / "remote-agents"
1348
+ / "bobmatnyc"
1349
+ / "claude-mpm-agents"
1350
+ / "AUTO-DEPLOY-INDEX.md"
1351
+ )
1352
+ if not index_path.exists():
1353
+ print(
1354
+ f"[yellow]Could not find AUTO-DEPLOY-INDEX.md at: {index_path}[/yellow]"
1355
+ )
1356
+ input("\nPress Enter to continue...")
1357
+ continue
1358
+
1359
+ parser = AutoDeployIndexParser(index_path)
1360
+ framework_agent_ids = parser.get_agents_by_framework(
1361
+ framework.lower()
1362
+ )
1363
+
1364
+ all_agents = self._merge_agent_sources()
1365
+ filtered_agents = [
1366
+ a for a in all_agents if a["agent_id"] in framework_agent_ids
1367
+ ]
1368
+ filter_description = f"Framework: {framework}"
1369
+ except Exception as e:
1370
+ self.logger.error(f"Framework filter error: {e}", exc_info=True)
1371
+ print(f"[yellow]Could not filter by framework: {e}[/yellow]")
1372
+ input("\nPress Enter to continue...")
1373
+ continue
1374
+
1375
+ elif choice_num == "4":
1376
+ # Show all agents
1377
+ filtered_agents = self._merge_agent_sources()
1378
+ filter_description = "All agents"
1379
+ else:
1380
+ print("❌ Invalid choice")
1381
+ input("\nPress Enter to continue...")
1382
+ continue
1383
+
1384
+ # Display filtered results
1385
+ print("\n" + "=" * 60)
1386
+ print(f"📋 {filter_description} ({len(filtered_agents)} agents)")
1387
+ print("=" * 60)
1388
+
1389
+ if not filtered_agents:
1390
+ print("\n[yellow]No agents found matching filter[/yellow]")
1391
+ else:
1392
+ print(f"\n{'#':<4} {'Agent ID':<40} {'Name':<25} {'Status':<12}")
1393
+ print("-" * 85)
1394
+
1395
+ for idx, agent in enumerate(filtered_agents, 1):
1396
+ agent_id = (
1397
+ agent["agent_id"][:39]
1398
+ if len(agent["agent_id"]) > 39
1399
+ else agent["agent_id"]
1400
+ )
1401
+ name = (
1402
+ agent["name"][:24] if len(agent["name"]) > 24 else agent["name"]
1403
+ )
1404
+ status = "✓ Deployed" if agent.get("deployed") else "Available"
1405
+ print(f"{idx:<4} {agent_id:<40} {name:<25} {status:<12}")
1406
+
1407
+ print("\n[bold]Actions:[/bold]")
1408
+ print(" [d] Deploy agent from this list")
1409
+ print(" [v] View agent details")
1410
+ print(" [n] New filter")
1411
+ print(" [b] Back to main menu")
1412
+
1413
+ action = input("\nSelect action: ").strip()
1414
+
1415
+ if action == "b":
1416
+ break
1417
+ if action == "n":
1418
+ continue
1419
+ if action == "d":
1420
+ self._deploy_from_filtered_list(filtered_agents)
1421
+ elif action == "v":
1422
+ self._view_from_filtered_list(filtered_agents)
1423
+ else:
1424
+ print("❌ Invalid choice")
1425
+ input("\nPress Enter to continue...")
1426
+
1427
+ def _deploy_from_filtered_list(self, agents: List[Dict[str, Any]]):
1428
+ """Deploy an agent from a filtered list.
1429
+
1430
+ Args:
1431
+ agents: List of agent dictionaries with metadata
1432
+ """
1433
+ if not agents:
1434
+ print("\n[yellow]No agents in list[/yellow]")
1435
+ input("\nPress Enter to continue...")
1436
+ return
1437
+
1438
+ deployable = [a for a in agents if not a.get("deployed")]
1439
+
1440
+ if not deployable:
1441
+ print("\n[yellow]All agents in this list are already deployed[/yellow]")
1442
+ input("\nPress Enter to continue...")
1443
+ return
1444
+
1445
+ # Build agent selection choices
1446
+ agent_choices = [
1447
+ f"{i}. {agent['agent_id']}" for i, agent in enumerate(agents, 1)
1448
+ ]
1449
+
1450
+ agent_choice = questionary.select(
1451
+ "Select agent to deploy:", choices=agent_choices, style=QUESTIONARY_STYLE
1452
+ ).ask()
1453
+
1454
+ if not agent_choice: # User pressed Esc
1455
+ return
1456
+
1457
+ # Parse agent index from "N. agent_id" format
1458
+ idx = int(agent_choice.split(".")[0]) - 1
1459
+ agent = agents[idx]
1460
+
1461
+ if agent.get("deployed"):
1462
+ print(f"\n[yellow]{agent['agent_id']} is already deployed[/yellow]")
1463
+ else:
1464
+ print(f"\n🚀 Deploying {agent['agent_id']}...")
1465
+
1466
+ try:
1467
+ from claude_mpm.services.agents.deployment.agent_template_builder import (
1468
+ AgentTemplateBuilder,
1469
+ )
1470
+ from claude_mpm.services.agents.deployment.agent_version_manager import (
1471
+ AgentVersionManager,
1472
+ )
1473
+ from claude_mpm.services.agents.deployment.deployment_results_manager import (
1474
+ DeploymentResultsManager,
1475
+ )
1476
+ from claude_mpm.services.agents.deployment.single_agent_deployer import (
1477
+ SingleAgentDeployer,
1478
+ )
1479
+
1480
+ # Initialize deployment services
1481
+ template_builder = AgentTemplateBuilder()
1482
+ version_manager = AgentVersionManager()
1483
+ results_manager = DeploymentResultsManager(self.logger)
1484
+ deployer = SingleAgentDeployer(
1485
+ template_builder=template_builder,
1486
+ version_manager=version_manager,
1487
+ results_manager=results_manager,
1488
+ logger=self.logger,
1489
+ )
1490
+
1491
+ # Prepare deployment parameters
1492
+ template_path = Path(agent["path"])
1493
+ target_dir = Path.cwd() / ".claude" / "agents"
1494
+
1495
+ # Find base_agent.json in multiple possible locations
1496
+ base_agent_candidates = [
1497
+ Path.home()
1498
+ / ".claude-mpm"
1499
+ / "agents"
1500
+ / "templates"
1501
+ / "base_agent.json",
1502
+ Path.home() / ".claude-mpm" / "cache" / "base_agent.json",
1503
+ Path(__file__).parent.parent.parent
1504
+ / "agents"
1505
+ / "templates"
1506
+ / "base_agent.json",
1507
+ ]
1508
+ base_agent_path = None
1509
+ for candidate in base_agent_candidates:
1510
+ if candidate.exists():
1511
+ base_agent_path = candidate
1512
+ break
1513
+
1514
+ if not base_agent_path:
1515
+ base_agent_path = base_agent_candidates[
1516
+ 0
1517
+ ] # Use default even if not exists
1518
+
1519
+ # Deploy the agent
1520
+ success = deployer.deploy_agent(
1521
+ agent_name=agent["agent_id"],
1522
+ templates_dir=template_path.parent,
1523
+ target_dir=target_dir,
1524
+ base_agent_path=base_agent_path,
1525
+ force_rebuild=True,
1526
+ working_directory=Path.cwd(),
1527
+ )
1528
+
1529
+ if success:
1530
+ print(f"[green]✓ Successfully deployed {agent['agent_id']}[/green]")
1531
+ else:
1532
+ print(f"[red]✗ Failed to deploy {agent['agent_id']}[/red]")
1533
+
1534
+ except Exception as e:
1535
+ self.logger.error(f"Deployment error: {e}", exc_info=True)
1536
+ print(f"❌ Deployment error: {e}")
1537
+
1538
+ input("\nPress Enter to continue...")
1539
+
1540
+ def _view_from_filtered_list(self, agents: List[Dict[str, Any]]):
1541
+ """View details of an agent from filtered list.
1542
+
1543
+ Args:
1544
+ agents: List of agent dictionaries with metadata
1545
+ """
1546
+ if not agents:
1547
+ print("\n[yellow]No agents in list[/yellow]")
1548
+ input("\nPress Enter to continue...")
1549
+ return
1550
+
1551
+ # Build agent selection choices
1552
+ agent_choices = [
1553
+ f"{i}. {agent['agent_id']}" for i, agent in enumerate(agents, 1)
1554
+ ]
1555
+
1556
+ agent_choice = questionary.select(
1557
+ "Select agent to view:", choices=agent_choices, style=QUESTIONARY_STYLE
1558
+ ).ask()
1559
+
1560
+ if not agent_choice: # User pressed Esc
1561
+ return
1562
+
1563
+ # Parse agent index from "N. agent_id" format
1564
+ idx = int(agent_choice.split(".")[0]) - 1
1565
+ agent = agents[idx]
1566
+ self._show_agent_details(agent)
1567
+
1568
+ def _deploy_preset_interactive(self):
1569
+ """Interactive preset deployment with preview and confirmation."""
1570
+ from claude_mpm.services.agents.agent_preset_service import AgentPresetService
1571
+
1572
+ if not self.source_manager:
1573
+ print("\n❌ Source manager not available")
1574
+ input("\nPress Enter to continue...")
1575
+ return
1576
+
1577
+ preset_service = AgentPresetService(self.source_manager)
1578
+
1579
+ while True:
1580
+ print("\n" + "=" * 60)
1581
+ print("📦 Deploy Agent Preset")
1582
+ print("=" * 60)
1583
+
1584
+ # List available presets
1585
+ presets = preset_service.list_presets()
1586
+
1587
+ print(f"\n{len(presets)} preset(s) available:\n")
1588
+ print(f"{'#':<4} {'Preset':<20} {'Agents':<10} {'Description':<50}")
1589
+ print("-" * 90)
1590
+
1591
+ for idx, preset in enumerate(presets, 1):
1592
+ description = (
1593
+ preset["description"][:48] + "..."
1594
+ if len(preset["description"]) > 50
1595
+ else preset["description"]
1596
+ )
1597
+ print(
1598
+ f"{idx:<4} {preset['name']:<20} {len(preset.get('agents', [])):<10} {description:<50}"
1599
+ )
1600
+
1601
+ print("\n[bold]Actions:[/bold]")
1602
+ print(" [1-11] Select preset number")
1603
+ print(" [b] Back to main menu")
1604
+
1605
+ choice = input("\nSelect preset number or action: ").strip()
1606
+
1607
+ if choice.lower() == "b":
1608
+ break
1609
+
1610
+ try:
1611
+ idx = int(choice) - 1
1612
+ if idx < 0 or idx >= len(presets):
1613
+ raise ValueError("Out of range")
1614
+
1615
+ preset_name = presets[idx]["name"]
1616
+
1617
+ # Show preset details
1618
+ print("\n" + "=" * 60)
1619
+ print(f"📦 Preset: {preset_name}")
1620
+ print("=" * 60)
1621
+ print(f"\n[bold]Description:[/bold] {presets[idx]['description']}\n")
1622
+
1623
+ # Resolve preset
1624
+ print("🔍 Resolving preset agents...")
1625
+ resolution = preset_service.resolve_agents(
1626
+ preset_name, validate_availability=True
1627
+ )
1628
+
1629
+ if resolution.get("missing_agents"):
1630
+ print(
1631
+ f"\n⚠️ [red]Missing agents ({len(resolution['missing_agents'])}):[/red]"
1632
+ )
1633
+ for agent_id in resolution["missing_agents"]:
1634
+ print(f" • {agent_id}")
1635
+ print("\n[yellow]Cannot deploy preset with missing agents[/yellow]")
1636
+ input("\nPress Enter to continue...")
1637
+ continue
1638
+
1639
+ # Show agents to deploy
1640
+ agents = resolution.get("agents", [])
1641
+ print(f"\n[bold]Agents to deploy ({len(agents)}):[/bold]\n")
1642
+
1643
+ print(f"{'Agent ID':<40} {'Name':<25} {'Source':<25}")
1644
+ print("-" * 95)
1645
+
1646
+ for agent in agents:
1647
+ # Get agent metadata
1648
+ agent_metadata = agent.get("metadata", {})
1649
+ agent_meta_data = agent_metadata.get("metadata", {})
1650
+
1651
+ agent_id = (
1652
+ agent.get("agent_id", "")[:39]
1653
+ if len(agent.get("agent_id", "")) > 39
1654
+ else agent.get("agent_id", "")
1655
+ )
1656
+ name = (
1657
+ agent_meta_data.get("name", "")[:24]
1658
+ if len(agent_meta_data.get("name", "")) > 24
1659
+ else agent_meta_data.get("name", "")
1660
+ )
1661
+ source = (
1662
+ agent.get("source", "unknown")[:24]
1663
+ if len(agent.get("source", "unknown")) > 24
1664
+ else agent.get("source", "unknown")
1665
+ )
1666
+
1667
+ print(f"{agent_id:<40} {name:<25} {source:<25}")
1668
+
1669
+ # Confirm deployment
1670
+ print("\n[bold]Options:[/bold]")
1671
+ print(" [y] Deploy all agents")
1672
+ print(" [n] Cancel")
1673
+
1674
+ confirm = input("\nProceed with deployment? ").strip()
1675
+
1676
+ if confirm.lower() == "y":
1677
+ print(f"\n🚀 Deploying preset '{preset_name}'...\n")
1678
+
1679
+ from claude_mpm.services.agents.deployment.agent_template_builder import (
1680
+ AgentTemplateBuilder,
1681
+ )
1682
+ from claude_mpm.services.agents.deployment.agent_version_manager import (
1683
+ AgentVersionManager,
1684
+ )
1685
+ from claude_mpm.services.agents.deployment.deployment_results_manager import (
1686
+ DeploymentResultsManager,
1687
+ )
1688
+ from claude_mpm.services.agents.deployment.single_agent_deployer import (
1689
+ SingleAgentDeployer,
1690
+ )
1691
+
1692
+ # Initialize deployment services once for all agents
1693
+ template_builder = AgentTemplateBuilder()
1694
+ version_manager = AgentVersionManager()
1695
+ results_manager = DeploymentResultsManager(self.logger)
1696
+ deployer = SingleAgentDeployer(
1697
+ template_builder=template_builder,
1698
+ version_manager=version_manager,
1699
+ results_manager=results_manager,
1700
+ logger=self.logger,
1701
+ )
1702
+
1703
+ target_dir = Path.cwd() / ".claude" / "agents"
1704
+
1705
+ # Find base_agent.json
1706
+ base_agent_candidates = [
1707
+ Path.home()
1708
+ / ".claude-mpm"
1709
+ / "agents"
1710
+ / "templates"
1711
+ / "base_agent.json",
1712
+ Path.home() / ".claude-mpm" / "cache" / "base_agent.json",
1713
+ Path(__file__).parent.parent.parent
1714
+ / "agents"
1715
+ / "templates"
1716
+ / "base_agent.json",
1717
+ ]
1718
+ base_agent_path = None
1719
+ for candidate in base_agent_candidates:
1720
+ if candidate.exists():
1721
+ base_agent_path = candidate
1722
+ break
1723
+
1724
+ if not base_agent_path:
1725
+ base_agent_path = base_agent_candidates[0]
1726
+
1727
+ deployed = 0
1728
+ failed = 0
1729
+
1730
+ for agent in agents:
1731
+ agent_id = agent["agent_id"]
1732
+ agent_metadata = agent.get("metadata", {})
1733
+ agent_path = agent_metadata.get(
1734
+ "path", agent_metadata.get("source_file", "")
1735
+ )
1736
+
1737
+ if not agent_path:
1738
+ print(f" Deploying {agent_id}... [red]✗ (no path)[/red]")
1739
+ failed += 1
1740
+ continue
1741
+
1742
+ print(f" Deploying {agent_id}...", end=" ", flush=True)
1743
+
1744
+ try:
1745
+ template_path = Path(agent_path)
1746
+ success = deployer.deploy_agent(
1747
+ agent_name=agent_id,
1748
+ templates_dir=template_path.parent,
1749
+ target_dir=target_dir,
1750
+ base_agent_path=base_agent_path,
1751
+ force_rebuild=True,
1752
+ working_directory=Path.cwd(),
1753
+ )
1754
+
1755
+ if success:
1756
+ print("[green]✓[/green]")
1757
+ deployed += 1
1758
+ else:
1759
+ print("[red]✗[/red]")
1760
+ failed += 1
1761
+ except Exception as e:
1762
+ print(f"[red]✗ ({e})[/red]")
1763
+ self.logger.error(
1764
+ f"Failed to deploy {agent_id}: {e}", exc_info=True
1765
+ )
1766
+ failed += 1
1767
+
1768
+ print("\n[bold]Summary:[/bold]")
1769
+ print(f" • Deployed: {deployed}")
1770
+ print(f" • Failed: {failed}")
1771
+ print(f" • Total: {len(agents)}")
1772
+
1773
+ if failed == 0:
1774
+ print(
1775
+ f"\n[green]✓ Preset '{preset_name}' deployed successfully![/green]"
1776
+ )
1777
+ else:
1778
+ print(
1779
+ f"\n[yellow]⚠ Preset deployed with {failed} failures[/yellow]"
1780
+ )
1781
+
1782
+ input("\nPress Enter to continue...")
1783
+ break
1784
+
1785
+ except (ValueError, IndexError):
1786
+ print("❌ Invalid preset selection")
1787
+ input("\nPress Enter to continue...")
1788
+ except Exception as e:
1789
+ self.logger.error(f"Preset deployment error: {e}", exc_info=True)
1790
+ print(f"❌ Error: {e}")
1791
+ input("\nPress Enter to continue...")
1792
+
1793
+ def _manage_sources_interactive(self):
1794
+ """Interactive source management."""
1795
+ if not self.discovery_enabled or not self.source_manager:
1796
+ print("\n❌ Source manager not available")
1797
+ input("\nPress Enter to continue...")
1798
+ return
1799
+
1800
+ print("\n" + "=" * 60)
1801
+ print("🔗 Manage Agent Sources")
1802
+ print("=" * 60)
1803
+
1804
+ try:
1805
+ from claude_mpm.config.agent_sources import AgentSourceConfiguration
1806
+
1807
+ config = AgentSourceConfiguration()
1808
+ sources = config.list_sources()
1809
+
1810
+ if not sources:
1811
+ print("\n📭 No sources configured.")
1812
+ else:
1813
+ print(f"\n{len(sources)} source(s) configured:\n")
1814
+ print(f"{'Source':<40} {'Priority':<10} {'Status':<10}")
1815
+ print("-" * 60)
1816
+
1817
+ for source in sources:
1818
+ identifier = source.get("identifier", "unknown")[:39]
1819
+ priority = str(source.get("priority", 100))
1820
+ status = "✓ Active" if source.get("enabled", True) else "Disabled"
1821
+ print(f"{identifier:<40} {priority:<10} {status:<10}")
1822
+
1823
+ print("\n💡 Use 'claude-mpm agent-source' command to add/remove sources")
1824
+ print("💡 Use 'claude-mpm agents discover' command to refresh agent cache")
1825
+
1826
+ except Exception as e:
1827
+ self.logger.error(f"Failed to list sources: {e}", exc_info=True)
1828
+ print(f"\n❌ Error: {e}")
1829
+
1830
+ input("\nPress Enter to continue...")
1831
+
1832
+
1833
+ def run_interactive_agent_wizard() -> int:
1834
+ """Entry point for interactive agent wizard.
1835
+
1836
+ Returns:
1837
+ Exit code (0 for success, non-zero for failure)
1838
+ """
1839
+ try:
1840
+ wizard = AgentWizard()
1841
+ success, message = wizard.run_interactive_create()
1842
+
1843
+ if success:
1844
+ print(f"\n✅ {message}")
1845
+ return 0
1846
+ print(f"\n❌ {message}", file=sys.stderr)
1847
+ return 1
1848
+
1849
+ except Exception as e:
1850
+ print(f"\n❌ Wizard error: {e}", file=sys.stderr)
1851
+ return 1
1852
+
1853
+
1854
+ def run_interactive_agent_manager() -> int:
1855
+ """Entry point for interactive agent management.
1856
+
1857
+ Returns:
1858
+ Exit code (0 for success, non-zero for failure)
1859
+ """
1860
+ try:
1861
+ wizard = AgentWizard()
1862
+ success, message = wizard.run_interactive_manage()
1863
+
1864
+ if success:
1865
+ print(f"\n✅ {message}")
1866
+ return 0
1867
+ print(f"\n❌ {message}", file=sys.stderr)
1868
+ return 1
1869
+
1870
+ except Exception as e:
1871
+ print(f"\n❌ Management error: {e}", file=sys.stderr)
1872
+ return 1