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