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,1369 @@
1
+ """Agent Template Builder Service
2
+
3
+ This service handles the building and generation of agent templates,
4
+ including YAML and Markdown generation, template merging, and metadata extraction.
5
+
6
+ Extracted from AgentDeploymentService as part of the refactoring to improve
7
+ maintainability and testability.
8
+ """
9
+
10
+ import json
11
+ import re
12
+ from pathlib import Path
13
+ from typing import Any, Dict, List
14
+
15
+ import yaml
16
+
17
+ from claude_mpm.core.logging_config import get_logger
18
+
19
+
20
+ class AgentTemplateBuilder:
21
+ """Service for building agent templates from JSON and base agent data.
22
+
23
+ This service handles:
24
+ - Building agent markdown files with YAML frontmatter
25
+ - Building agent YAML files
26
+ - Merging narrative and configuration fields
27
+ - Extracting agent metadata
28
+ - Formatting YAML lists
29
+ """
30
+
31
+ def __init__(self):
32
+ """Initialize the template builder."""
33
+ self.logger = get_logger(__name__)
34
+
35
+ def normalize_tools_input(self, tools):
36
+ """Normalize various tool input formats to a consistent list.
37
+
38
+ Handles multiple input formats:
39
+ - None/empty: Returns default tools
40
+ - String: Splits by comma and strips whitespace
41
+ - List: Ensures all items are strings and strips whitespace
42
+ - Dict: Takes enabled tools (where value is True)
43
+
44
+ Args:
45
+ tools: Tools input in various formats (str, list, dict, or None)
46
+
47
+ Returns:
48
+ List of tool names, normalized and cleaned
49
+ """
50
+ default_tools = ["Read", "Write", "Edit", "Grep", "Glob", "Bash"]
51
+
52
+ # Handle None or empty
53
+ if not tools:
54
+ self.logger.debug("No tools provided, using defaults")
55
+ return default_tools
56
+
57
+ # Convert to list format
58
+ if isinstance(tools, str):
59
+ # Split by comma, strip whitespace
60
+ tool_list = [t.strip() for t in tools.split(",") if t.strip()]
61
+ self.logger.debug(f"Converted string tools '{tools}' to list: {tool_list}")
62
+ elif isinstance(tools, list):
63
+ # Ensure all items are strings and strip whitespace
64
+ tool_list = [str(t).strip() for t in tools if t and str(t).strip()]
65
+ self.logger.debug(f"Normalized list tools: {tool_list}")
66
+ elif isinstance(tools, dict):
67
+ # Handle dict format - take enabled tools
68
+ tool_list = [k for k, v in tools.items() if v]
69
+ self.logger.info(f"Converting dict tools format: {tools} -> {tool_list}")
70
+ else:
71
+ self.logger.warning(f"Unknown tools format: {type(tools)}, using defaults")
72
+ return default_tools
73
+
74
+ # Return processed list or defaults if empty
75
+ if not tool_list:
76
+ self.logger.debug("Tools list empty after processing, using defaults")
77
+ return default_tools
78
+
79
+ return tool_list
80
+
81
+ def _discover_base_agent_templates(self, agent_file: Path) -> List[Path]:
82
+ """Discover BASE-AGENT.md files in hierarchy from agent file to repository root.
83
+
84
+ This method implements hierarchical BASE template discovery by walking up the
85
+ directory tree from the agent file location and collecting all BASE-AGENT.md
86
+ files found along the way.
87
+
88
+ Composition Order (closest to farthest):
89
+ 1. Local BASE-AGENT.md (same directory as agent)
90
+ 2. Parent BASE-AGENT.md (parent directory)
91
+ 3. Grandparent BASE-AGENT.md (grandparent directory)
92
+ ... continuing to repository root
93
+
94
+ Args:
95
+ agent_file: Path to the agent template file
96
+
97
+ Returns:
98
+ List of BASE-AGENT.md paths ordered from closest to farthest
99
+ (same directory to root)
100
+
101
+ Example:
102
+ Given structure:
103
+ repo/
104
+ BASE-AGENT.md # Root (index 2)
105
+ engineering/
106
+ BASE-AGENT.md # Parent (index 1)
107
+ python/
108
+ BASE-AGENT.md # Local (index 0)
109
+ fastapi-engineer.md # Agent file
110
+
111
+ Returns: [
112
+ repo/engineering/python/BASE-AGENT.md,
113
+ repo/engineering/BASE-AGENT.md,
114
+ repo/BASE-AGENT.md
115
+ ]
116
+ """
117
+ base_templates = []
118
+ current_dir = agent_file.parent
119
+
120
+ # Walk up directory tree until we reach root or a reasonable limit
121
+ # Stop at repository root or after 10 levels (safety limit)
122
+ max_depth = 10
123
+ depth = 0
124
+
125
+ while current_dir and depth < max_depth:
126
+ # Check for BASE-AGENT.md in current directory
127
+ base_agent_file = current_dir / "BASE-AGENT.md"
128
+ if base_agent_file.exists() and base_agent_file.is_file():
129
+ base_templates.append(base_agent_file)
130
+ self.logger.debug(f"Found BASE-AGENT.md at: {base_agent_file}")
131
+
132
+ # Stop at git repository root if detected (check AFTER finding BASE-AGENT.md)
133
+ if (current_dir / ".git").exists():
134
+ self.logger.debug(f"Reached git repository root at: {current_dir}")
135
+ break
136
+
137
+ # Stop at common repository root indicators (check AFTER finding BASE-AGENT.md)
138
+ # Stop at cache root or .claude-mpm directory
139
+ if current_dir.name in [".claude-mpm", "cache"]:
140
+ self.logger.debug(
141
+ f"Reached repository root indicator at: {current_dir}"
142
+ )
143
+ break
144
+
145
+ # Move to parent directory
146
+ parent = current_dir.parent
147
+ if parent == current_dir: # Reached filesystem root
148
+ break
149
+
150
+ current_dir = parent
151
+ depth += 1
152
+
153
+ if base_templates:
154
+ self.logger.info(
155
+ f"Discovered {len(base_templates)} BASE-AGENT.md file(s) for {agent_file.name}"
156
+ )
157
+
158
+ return base_templates
159
+
160
+ def _parse_markdown_template(self, template_path: Path) -> dict:
161
+ """Parse Markdown template with YAML frontmatter.
162
+
163
+ Extracts metadata from YAML frontmatter and content from Markdown body.
164
+ Supports the new agent template format with YAML frontmatter between --- delimiters.
165
+
166
+ Args:
167
+ template_path: Path to the Markdown template file
168
+
169
+ Returns:
170
+ Dictionary containing metadata and instructions
171
+
172
+ Raises:
173
+ ValueError: If frontmatter is missing or malformed
174
+ yaml.YAMLError: If YAML parsing fails
175
+ """
176
+ content = template_path.read_text(encoding="utf-8")
177
+
178
+ # Split frontmatter and body
179
+ # Format: ---\n<yaml>\n---\n<markdown>
180
+ if not content.startswith("---"):
181
+ raise ValueError(
182
+ f"Markdown template missing YAML frontmatter: {template_path}"
183
+ )
184
+
185
+ # Split by --- delimiters
186
+ parts = content.split("---", 2)
187
+ if len(parts) < 3:
188
+ raise ValueError(f"Malformed YAML frontmatter in template: {template_path}")
189
+
190
+ # parts[0] is empty (before first ---)
191
+ # parts[1] is YAML frontmatter
192
+ # parts[2] is Markdown content
193
+ yaml_content = parts[1].strip()
194
+ markdown_content = parts[2].strip()
195
+
196
+ # Parse YAML frontmatter
197
+ try:
198
+ metadata = yaml.safe_load(yaml_content)
199
+ if not isinstance(metadata, dict):
200
+ raise ValueError(
201
+ f"YAML frontmatter must be a dictionary: {template_path}"
202
+ )
203
+ except yaml.YAMLError as e:
204
+ self.logger.error(
205
+ f"Failed to parse YAML frontmatter in {template_path}: {e}"
206
+ )
207
+ raise
208
+
209
+ # Validate required fields
210
+ required_fields = ["name", "description", "version"]
211
+ missing_fields = [field for field in required_fields if field not in metadata]
212
+ if missing_fields:
213
+ raise ValueError(
214
+ f"Missing required fields in template {template_path}: {missing_fields}"
215
+ )
216
+
217
+ # Add the markdown content as instructions field
218
+ metadata["instructions"] = markdown_content
219
+
220
+ # Normalize metadata structure to match JSON template format
221
+ # JSON templates have these fields at top level, Markdown may have them nested
222
+ self._normalize_metadata_structure(metadata)
223
+
224
+ return metadata
225
+
226
+ def _normalize_metadata_structure(self, metadata: dict) -> None:
227
+ """Normalize metadata structure to match expected JSON template format.
228
+
229
+ This ensures both Markdown and JSON templates produce the same metadata structure
230
+ for downstream processing.
231
+
232
+ Args:
233
+ metadata: Metadata dictionary to normalize (modified in-place)
234
+ """
235
+ # Map Markdown frontmatter fields to JSON template structure
236
+ # Handle tags: YAML list vs JSON comma-separated string
237
+ if "tags" in metadata and isinstance(metadata["tags"], list):
238
+ # Keep as list for now, normalize_tools_input will handle both formats
239
+ pass
240
+
241
+ # Map agent_id to name if name is missing
242
+ if "agent_id" in metadata and "name" not in metadata:
243
+ metadata["name"] = metadata["agent_id"]
244
+
245
+ # Ensure capabilities dict exists
246
+ if "capabilities" not in metadata:
247
+ metadata["capabilities"] = {}
248
+
249
+ # Merge top-level capability fields into capabilities dict
250
+ capability_fields = ["memory_limit", "cpu_limit", "network_access"]
251
+ for field in capability_fields:
252
+ if field in metadata:
253
+ metadata["capabilities"][field] = metadata.pop(field)
254
+
255
+ # Add model to capabilities if present at top level
256
+ if "model" in metadata and "model" not in metadata["capabilities"]:
257
+ metadata["capabilities"]["model"] = metadata["model"]
258
+
259
+ def _load_base_agent_instructions(self, agent_type: str) -> str:
260
+ """Load BASE instructions for a specific agent type.
261
+
262
+ DEPRECATED: This method loads BASE_{TYPE}.md files (old pattern).
263
+ New pattern uses hierarchical BASE-AGENT.md files discovered via
264
+ _discover_base_agent_templates() and composed in build_agent_markdown().
265
+
266
+ Args:
267
+ agent_type: The type of agent (engineer, qa, ops, research, documentation)
268
+
269
+ Returns:
270
+ The BASE instructions content or empty string if not found
271
+ """
272
+ if not agent_type:
273
+ return ""
274
+
275
+ try:
276
+ # Construct BASE file name
277
+ base_file = f"BASE_{agent_type.upper()}.md"
278
+
279
+ # Try to find BASE file in agents directory
280
+ # First try current working directory structure
281
+ agents_dir = Path(__file__).parent.parent.parent.parent / "agents"
282
+ base_path = agents_dir / base_file
283
+
284
+ if not base_path.exists():
285
+ # Try packaged resources if available
286
+ try:
287
+ from importlib.resources import files
288
+
289
+ agents_package = files("claude_mpm.agents")
290
+ base_resource = agents_package / base_file
291
+ if base_resource.is_file():
292
+ content = base_resource.read_text(encoding="utf-8")
293
+ self.logger.debug(
294
+ f"Loaded BASE instructions from package: {base_file}"
295
+ )
296
+ return content
297
+ except (ImportError, Exception) as e:
298
+ self.logger.debug(
299
+ f"Could not load BASE instructions from package: {e}"
300
+ )
301
+
302
+ # Final fallback - try multiple possible locations
303
+ possible_paths = [
304
+ Path.cwd() / "src" / "claude_mpm" / "agents" / base_file,
305
+ Path(__file__).parent.parent.parent.parent / "agents" / base_file,
306
+ Path.home() / ".claude-mpm" / "agents" / base_file,
307
+ ]
308
+
309
+ for path in possible_paths:
310
+ if path.exists():
311
+ base_path = path
312
+ break
313
+ else:
314
+ self.logger.debug(
315
+ f"No BASE instructions found for type: {agent_type}"
316
+ )
317
+ return ""
318
+
319
+ if base_path.exists():
320
+ self.logger.debug(f"Loading BASE instructions from {base_path}")
321
+ return base_path.read_text(encoding="utf-8")
322
+ self.logger.debug(f"No BASE instructions found for type: {agent_type}")
323
+ return ""
324
+
325
+ except Exception as e:
326
+ self.logger.warning(
327
+ f"Error loading BASE instructions for {agent_type}: {e}"
328
+ )
329
+ return ""
330
+
331
+ def build_agent_markdown(
332
+ self,
333
+ agent_name: str,
334
+ template_path: Path,
335
+ base_agent_data: dict,
336
+ source_info: str = "unknown",
337
+ ) -> str:
338
+ """
339
+ Build a complete agent markdown file with YAML frontmatter.
340
+
341
+ Args:
342
+ agent_name: Name of the agent
343
+ template_path: Path to the agent template (JSON or Markdown file)
344
+ base_agent_data: Base agent configuration data
345
+ source_info: Source of the agent (system/project/user)
346
+
347
+ Returns:
348
+ Complete markdown content with YAML frontmatter
349
+
350
+ Raises:
351
+ FileNotFoundError: If template file doesn't exist
352
+ json.JSONDecodeError: If template JSON is invalid
353
+ yaml.YAMLError: If template YAML is invalid
354
+ ValueError: If template format is unsupported or malformed
355
+ """
356
+ if not template_path.exists():
357
+ raise FileNotFoundError(f"Template file not found: {template_path}")
358
+
359
+ # Format detection: route to appropriate parser
360
+ try:
361
+ if template_path.suffix == ".md":
362
+ # Parse Markdown template with YAML frontmatter
363
+ self.logger.debug(f"Parsing Markdown template: {template_path}")
364
+ template_data = self._parse_markdown_template(template_path)
365
+ elif template_path.suffix == ".json":
366
+ # Parse JSON template (legacy format)
367
+ self.logger.debug(f"Parsing JSON template: {template_path}")
368
+ template_content = template_path.read_text()
369
+ template_data = json.loads(template_content)
370
+ else:
371
+ raise ValueError(
372
+ f"Unsupported template format: {template_path.suffix}. "
373
+ f"Expected .md or .json"
374
+ )
375
+ except json.JSONDecodeError as e:
376
+ self.logger.error(f"Invalid JSON in template {template_path}: {e}")
377
+ raise
378
+ except (yaml.YAMLError, ValueError) as e:
379
+ self.logger.error(f"Invalid template {template_path}: {e}")
380
+ raise
381
+
382
+ # Extract tools from template with fallback
383
+ # Handle both dict and list formats for capabilities (backward compatibility)
384
+ capabilities = template_data.get("capabilities", {})
385
+ capabilities_tools = (
386
+ capabilities.get("tools") if isinstance(capabilities, dict) else None
387
+ )
388
+
389
+ # Get raw tools from various possible locations
390
+ raw_tools = (
391
+ template_data.get("tools")
392
+ or capabilities_tools
393
+ or template_data.get("configuration_fields", {}).get("tools")
394
+ )
395
+
396
+ # Normalize tools to a consistent list format
397
+ tools = self.normalize_tools_input(raw_tools)
398
+
399
+ # Log if we see non-standard tool names (info level, not warning)
400
+ standard_tools = {
401
+ "Read",
402
+ "Write",
403
+ "Edit",
404
+ "MultiEdit", # File operations
405
+ "Grep",
406
+ "Glob",
407
+ "LS", # Search and navigation
408
+ "Bash",
409
+ "BashOutput",
410
+ "KillShell", # Command execution
411
+ "TodoWrite",
412
+ "ExitPlanMode", # Task management
413
+ "WebSearch",
414
+ "WebFetch", # Web operations
415
+ "NotebookRead",
416
+ "NotebookEdit", # Jupyter notebook support
417
+ }
418
+ non_standard = [t for t in tools if t not in standard_tools]
419
+ if non_standard:
420
+ self.logger.info(f"Using non-standard tools: {non_standard}")
421
+
422
+ # Extract model from template (no fallback - preserve None if not specified)
423
+ capabilities_model = (
424
+ capabilities.get("model") if isinstance(capabilities, dict) else None
425
+ )
426
+
427
+ model = (
428
+ template_data.get("model")
429
+ or capabilities_model
430
+ or template_data.get("configuration_fields", {}).get("model")
431
+ # No default fallback - preserve None if not set
432
+ )
433
+
434
+ # Convert tools list to comma-separated string (without spaces for compatibility)
435
+ ",".join(tools)
436
+
437
+ # Map model names to Claude Code format (as required)
438
+ model_map = {
439
+ "claude-3-5-sonnet-20241022": "sonnet",
440
+ "claude-3-5-haiku-20241022": "haiku",
441
+ "claude-3-opus-20240229": "opus",
442
+ "claude-3-5-sonnet": "sonnet",
443
+ "claude-3-sonnet": "sonnet",
444
+ "claude-3-haiku": "haiku",
445
+ "claude-3-opus": "opus",
446
+ "sonnet": "sonnet",
447
+ "haiku": "haiku",
448
+ "opus": "opus",
449
+ }
450
+
451
+ # Only map model if it's not None
452
+ if model is not None:
453
+ if model in model_map:
454
+ model = model_map[model]
455
+ # If model is specified but not in map, keep as-is (no default)
456
+
457
+ # Get response format from template or use base agent default
458
+ template_data.get("response", {}).get("format", "structured")
459
+
460
+ # Create Claude Code compatible name (lowercase, hyphens only)
461
+ claude_code_name = agent_name.lower().replace("_", "-")
462
+
463
+ # CRITICAL: NO underscores allowed - they cause silent failures!
464
+
465
+ # Validate the name before proceeding
466
+ if not re.match(r"^[a-z0-9]+(-[a-z0-9]+)*$", claude_code_name):
467
+ self.logger.error(
468
+ f"Invalid agent name '{claude_code_name}' - must match ^[a-z0-9]+(-[a-z0-9]+)*$"
469
+ )
470
+ raise ValueError(
471
+ f"Agent name '{claude_code_name}' does not meet Claude Code requirements"
472
+ )
473
+
474
+ # Extract description from template with fallback
475
+ raw_description = (
476
+ template_data.get("description")
477
+ or template_data.get("metadata", {}).get("description")
478
+ or f"{agent_name.title()} agent for specialized tasks"
479
+ )
480
+
481
+ # Convert to multiline format with examples for Claude Code compatibility
482
+ description = self._create_multiline_description(
483
+ raw_description, agent_name, template_data
484
+ )
485
+
486
+ # Extract custom metadata fields
487
+ metadata = template_data.get("metadata", {})
488
+ agent_version = (
489
+ template_data.get("agent_version")
490
+ or template_data.get("version")
491
+ or metadata.get("version", "1.0.0")
492
+ )
493
+ agent_type = template_data.get("agent_type", "general")
494
+ # Use the capabilities_model we already extracted earlier
495
+ model_type = capabilities_model or "sonnet"
496
+
497
+ # Map our model types to Claude Code format
498
+ if model_type in ["opus", "sonnet", "haiku"]:
499
+ # Use inherit for now - Claude Code seems to prefer this
500
+ pass
501
+ else:
502
+ pass
503
+
504
+ # Determine color - prefer template's color, fallback to type-based defaults
505
+ template_metadata = template_data.get("metadata", {})
506
+ template_color = template_metadata.get("color")
507
+
508
+ if template_color:
509
+ # Use the color specified in the template
510
+ pass
511
+ else:
512
+ # Fallback to default color map based on agent type
513
+ color_map = {
514
+ "engineer": "blue",
515
+ "qa": "green",
516
+ "security": "red",
517
+ "research": "purple",
518
+ "documentation": "cyan", # Changed default to match template preference
519
+ "ops": "gray",
520
+ }
521
+ color_map.get(agent_type, "blue")
522
+
523
+ # Check if we should include tools field (only if significantly restricting)
524
+ # Claude Code approach: omit tools field unless specifically restricting
525
+
526
+ # Convert tools to set for comparison
527
+ agent_tools = set(tools) if isinstance(tools, list) else set(tools.split(","))
528
+
529
+ # Only include tools field if agent is missing several important tools
530
+ # This matches Claude Code's approach of omitting tools for general-purpose agents
531
+ core_tools = {"Read", "Write", "Edit", "Bash", "Grep", "Glob"}
532
+ has_core_tools = len(agent_tools.intersection(core_tools)) >= 5
533
+
534
+ # Include tools field only if agent is clearly restricted (missing core tools or very few tools)
535
+ not has_core_tools or len(agent_tools) < 6
536
+
537
+ # Build YAML frontmatter with all relevant metadata from JSON template
538
+ # Include all fields that are useful for agent management and functionality
539
+ #
540
+ # COMPREHENSIVE AGENT FRONTMATTER FORMAT:
541
+ # - name: kebab-case agent name (required)
542
+ # - description: when/why to use this agent with examples (required, multiline)
543
+ # - model: mapped model name (required)
544
+ # - type: agent type for categorization and functionality (optional but important)
545
+ # - category: organizational category (optional)
546
+ # - color: visual identifier (optional)
547
+ # - version: agent version for update tracking (optional)
548
+ # - author: creator information (optional)
549
+ # - created_at: creation timestamp (optional)
550
+ # - updated_at: last update timestamp (optional)
551
+ # - tags: list of tags for search and categorization (optional)
552
+ frontmatter_lines = [
553
+ "---",
554
+ f"name: {claude_code_name}",
555
+ ]
556
+
557
+ # Add description as single-line YAML string with \n escapes
558
+ frontmatter_lines.append(
559
+ f"description: {self._format_description_for_yaml(description)}"
560
+ )
561
+
562
+ # Add model field only if explicitly set (not required for Claude Code)
563
+ if model is not None:
564
+ frontmatter_lines.append(f"model: {model}")
565
+
566
+ # Add type field (important for agent categorization)
567
+ if agent_type and agent_type != "general":
568
+ frontmatter_lines.append(f"type: {agent_type}")
569
+
570
+ # Add optional metadata fields
571
+ if metadata.get("color"):
572
+ frontmatter_lines.append(f"color: {metadata['color']}")
573
+ if metadata.get("category"):
574
+ frontmatter_lines.append(f"category: {metadata['category']}")
575
+ # Always include version field to prevent deployment comparison issues
576
+ if agent_version:
577
+ frontmatter_lines.append(f'version: "{agent_version}"')
578
+ if metadata.get("author"):
579
+ frontmatter_lines.append(f'author: "{metadata["author"]}"')
580
+ if metadata.get("created_at"):
581
+ frontmatter_lines.append(f"created_at: {metadata['created_at']}")
582
+ if metadata.get("updated_at"):
583
+ frontmatter_lines.append(f"updated_at: {metadata['updated_at']}")
584
+ # Add tags as comma-separated string if they exist (consistent with tools format)
585
+ if metadata.get("tags") and isinstance(metadata["tags"], list):
586
+ tags_str = ",".join(metadata["tags"])
587
+ frontmatter_lines.append(f"tags: {tags_str}")
588
+
589
+ frontmatter_lines.extend(
590
+ [
591
+ "---",
592
+ "",
593
+ ]
594
+ )
595
+
596
+ frontmatter = "\n".join(frontmatter_lines)
597
+
598
+ # Get agent instructions from template data (primary) or base agent data (fallback)
599
+ raw_instructions = template_data.get("instructions")
600
+
601
+ # Handle dictionary instructions format
602
+ if isinstance(raw_instructions, dict):
603
+ agent_specific_instructions = self._convert_instructions_dict_to_markdown(
604
+ raw_instructions
605
+ )
606
+ else:
607
+ agent_specific_instructions = (
608
+ raw_instructions
609
+ or base_agent_data.get("content")
610
+ or base_agent_data.get("instructions")
611
+ or "# Agent Instructions\n\nThis agent provides specialized assistance."
612
+ )
613
+
614
+ # Compose hierarchical BASE-AGENT.md templates
615
+ # Order: agent-specific + local BASE + parent BASE + ... + root BASE
616
+ content_parts = [agent_specific_instructions]
617
+
618
+ # Discover BASE-AGENT.md files in directory hierarchy
619
+ base_templates = self._discover_base_agent_templates(template_path)
620
+
621
+ # Append each BASE template (order: closest to farthest)
622
+ for base_template_path in base_templates:
623
+ try:
624
+ base_content = base_template_path.read_text(encoding="utf-8")
625
+ if base_content.strip():
626
+ content_parts.append(base_content)
627
+ self.logger.debug(
628
+ f"Composed BASE template: {base_template_path.relative_to(template_path.parent.parent) if template_path.parent.parent in base_template_path.parents else base_template_path.name}"
629
+ )
630
+ except Exception as e:
631
+ self.logger.warning(
632
+ f"Failed to read BASE template {base_template_path}: {e}"
633
+ )
634
+
635
+ # Fallback: Load legacy BASE_{TYPE}.md if no hierarchical templates found
636
+ if len(content_parts) == 1: # Only agent-specific instructions
637
+ legacy_base_instructions = self._load_base_agent_instructions(agent_type)
638
+ if legacy_base_instructions:
639
+ content_parts.append(legacy_base_instructions)
640
+ self.logger.debug(
641
+ f"Using legacy BASE_{agent_type.upper()}.md (no hierarchical BASE-AGENT.md found)"
642
+ )
643
+
644
+ # Join all parts with separator
645
+ content = "\n\n---\n\n".join(content_parts)
646
+
647
+ # Add memory update instructions if not already present
648
+ if "memory-update" not in content and "Remember" not in content:
649
+ memory_instructions = """
650
+
651
+ ## Memory Updates
652
+
653
+ When you learn something important about this project that would be useful for future tasks, include it in your response JSON block:
654
+
655
+ ```json
656
+ {
657
+ "memory-update": {
658
+ "Project Architecture": ["Key architectural patterns or structures"],
659
+ "Implementation Guidelines": ["Important coding standards or practices"],
660
+ "Current Technical Context": ["Project-specific technical details"]
661
+ }
662
+ }
663
+ ```
664
+
665
+ Or use the simpler "remember" field for general learnings:
666
+
667
+ ```json
668
+ {
669
+ "remember": ["Learning 1", "Learning 2"]
670
+ }
671
+ ```
672
+
673
+ Only include memories that are:
674
+ - Project-specific (not generic programming knowledge)
675
+ - Likely to be useful in future tasks
676
+ - Not already documented elsewhere
677
+ """
678
+ content = content + memory_instructions
679
+
680
+ # Combine frontmatter and content
681
+ return frontmatter + content
682
+
683
+ def build_agent_yaml(
684
+ self, agent_name: str, template_path: Path, base_agent_data: dict
685
+ ) -> str:
686
+ """
687
+ Build a complete agent YAML file by combining base agent and template.
688
+ Only includes essential fields for Claude Code best practices.
689
+
690
+ Args:
691
+ agent_name: Name of the agent
692
+ template_path: Path to the agent template JSON file
693
+ base_agent_data: Base agent configuration data
694
+
695
+ Returns:
696
+ Complete YAML content
697
+
698
+ Raises:
699
+ FileNotFoundError: If template file doesn't exist
700
+ json.JSONDecodeError: If template JSON is invalid
701
+ """
702
+ if not template_path.exists():
703
+ raise FileNotFoundError(f"Template file not found: {template_path}")
704
+
705
+ try:
706
+ template_content = template_path.read_text()
707
+ template_data = json.loads(template_content)
708
+ except json.JSONDecodeError as e:
709
+ self.logger.error(f"Invalid JSON in template {template_path}: {e}")
710
+ raise
711
+
712
+ # Merge narrative and configuration fields
713
+ self.merge_narrative_fields(base_agent_data, template_data)
714
+ merged_config = self.merge_configuration_fields(base_agent_data, template_data)
715
+
716
+ # Extract essential fields for Claude Code
717
+ name = template_data.get("name", agent_name)
718
+ description = template_data.get(
719
+ "description", f"{name} agent for specialized tasks"
720
+ )
721
+
722
+ # Get tools and model (no fallback for model)
723
+ raw_tools = merged_config.get("tools")
724
+ tools = self.normalize_tools_input(raw_tools)
725
+ model = merged_config.get("model") # No default - preserve None
726
+
727
+ # Format tools as YAML list
728
+ tools_yaml = self.format_yaml_list(tools, 2)
729
+
730
+ # Build YAML content with only essential fields
731
+ yaml_lines = [
732
+ f"name: {name}",
733
+ f"description: {description}",
734
+ ]
735
+
736
+ # Only include model if explicitly set
737
+ if model is not None:
738
+ yaml_lines.append(f"model: {model}")
739
+
740
+ yaml_lines.extend(
741
+ [
742
+ "tools:",
743
+ tools_yaml,
744
+ ]
745
+ )
746
+
747
+ return "\n".join(yaml_lines) + "\n"
748
+
749
+ def merge_narrative_fields(self, base_data: dict, template_data: dict) -> dict:
750
+ """
751
+ Merge narrative fields from base and template, combining arrays.
752
+
753
+ Args:
754
+ base_data: Base agent data
755
+ template_data: Template agent data
756
+
757
+ Returns:
758
+ Merged narrative fields
759
+ """
760
+ merged = {}
761
+
762
+ # Fields that should be combined (arrays)
763
+ combinable_fields = [
764
+ "when_to_use",
765
+ "specialized_knowledge",
766
+ "unique_capabilities",
767
+ ]
768
+
769
+ for field in combinable_fields:
770
+ base_value = base_data.get(field, [])
771
+ template_value = template_data.get(field, [])
772
+
773
+ # Ensure both are lists
774
+ if not isinstance(base_value, list):
775
+ base_value = [base_value] if base_value else []
776
+ if not isinstance(template_value, list):
777
+ template_value = [template_value] if template_value else []
778
+
779
+ # Combine and deduplicate
780
+ combined = list(set(base_value + template_value))
781
+ merged[field] = combined
782
+
783
+ return merged
784
+
785
+ def merge_configuration_fields(self, base_data: dict, template_data: dict) -> dict:
786
+ """
787
+ Merge configuration fields, with template overriding base.
788
+
789
+ Args:
790
+ base_data: Base agent data
791
+ template_data: Template agent data
792
+
793
+ Returns:
794
+ Merged configuration fields
795
+ """
796
+ merged = {}
797
+
798
+ # Start with base configuration
799
+ if "configuration_fields" in base_data:
800
+ merged.update(base_data["configuration_fields"])
801
+
802
+ # Override with template configuration
803
+ if "configuration_fields" in template_data:
804
+ merged.update(template_data["configuration_fields"])
805
+
806
+ # Also check for direct fields in template
807
+ direct_fields = ["tools", "model", "timeout", "max_tokens"]
808
+ for field in direct_fields:
809
+ if field in template_data:
810
+ merged[field] = template_data[field]
811
+
812
+ return merged
813
+
814
+ def extract_agent_metadata(self, template_content: str) -> Dict[str, Any]:
815
+ """
816
+ Extract metadata from simplified agent template content.
817
+
818
+ Args:
819
+ template_content: Agent template markdown content
820
+
821
+ Returns:
822
+ Dictionary containing extracted metadata
823
+ """
824
+ metadata = {}
825
+ current_section = None
826
+ section_content = []
827
+
828
+ lines = template_content.split("\n")
829
+
830
+ for line in lines:
831
+ line = line.strip()
832
+
833
+ # Check for section headers
834
+ if line.startswith("## "):
835
+ # Save previous section
836
+ if current_section and section_content:
837
+ metadata[current_section] = section_content.copy()
838
+
839
+ # Start new section
840
+ current_section = line[3:].lower().replace(" ", "_")
841
+ section_content = []
842
+
843
+ elif line.startswith("- ") and current_section:
844
+ # Add list item to current section
845
+ section_content.append(line[2:])
846
+
847
+ elif line and current_section and not line.startswith("#"):
848
+ # Add non-empty, non-header line to current section
849
+ section_content.append(line)
850
+
851
+ # Save final section
852
+ if current_section and section_content:
853
+ metadata[current_section] = section_content.copy()
854
+
855
+ # Ensure all required fields have defaults
856
+ metadata.setdefault("when_to_use", [])
857
+ metadata.setdefault("specialized_knowledge", [])
858
+ metadata.setdefault("unique_capabilities", [])
859
+
860
+ return metadata
861
+
862
+ def format_yaml_list(self, items: List[str], indent: int) -> str:
863
+ """
864
+ Format a list for YAML with proper indentation.
865
+
866
+ Args:
867
+ items: List of items to format
868
+ indent: Number of spaces for indentation
869
+
870
+ Returns:
871
+ Formatted YAML list string
872
+ """
873
+ if not items:
874
+ return ""
875
+
876
+ indent_str = " " * indent
877
+ formatted_items = []
878
+
879
+ for item in items:
880
+ formatted_items.append(f"{indent_str}- {item}")
881
+
882
+ return "\n".join(formatted_items)
883
+
884
+ def _create_multiline_description(
885
+ self, raw_description: str, agent_name: str, template_data: dict
886
+ ) -> str:
887
+ """
888
+ Create a comprehensive multiline description with examples for Claude Code compatibility.
889
+ Based on Claude's software-engineer.md format: detailed when/why description with examples.
890
+
891
+ Args:
892
+ raw_description: Original single-line description
893
+ agent_name: Name of the agent
894
+ template_data: Template data for extracting examples
895
+
896
+ Returns:
897
+ Formatted multiline description with examples in Claude Code format
898
+ """
899
+ raw_description = self._format_to_single_line(raw_description)
900
+
901
+ # Get agent type for creating targeted descriptions
902
+ agent_type = template_data.get("agent_type", "general")
903
+
904
+ # Create enhanced description based on agent type
905
+ enhanced_description = self._create_enhanced_description(
906
+ raw_description, agent_name, agent_type, template_data
907
+ )
908
+
909
+ # Add examples
910
+ examples = self._extract_examples_from_template(template_data, agent_name)
911
+ if not examples:
912
+ examples = self._generate_default_examples(agent_name, template_data)
913
+
914
+ # Combine enhanced description with examples
915
+ if examples:
916
+ description_parts = [enhanced_description, "", *examples]
917
+ else:
918
+ description_parts = [enhanced_description]
919
+
920
+ return "\n".join(description_parts)
921
+
922
+ def _format_to_single_line(self, description: str) -> str:
923
+ """
924
+ Format description to single line by removing line breaks and normalizing whitespace.
925
+
926
+ Args:
927
+ description: Raw description text
928
+
929
+ Returns:
930
+ Single-line formatted description
931
+ """
932
+ if not description:
933
+ return description
934
+
935
+ # Remove all line breaks and normalize whitespace
936
+ single_line = " ".join(description.strip().split())
937
+
938
+ # Remove redundant spaces around punctuation
939
+ single_line = re.sub(r"\s+([,.!?;:])", r"\1", single_line)
940
+ return re.sub(r"([,.!?;:])\s+", r"\1 ", single_line)
941
+
942
+ def _create_enhanced_description(
943
+ self,
944
+ raw_description: str,
945
+ agent_name: str,
946
+ agent_type: str,
947
+ template_data: dict,
948
+ ) -> str:
949
+ """
950
+ Create an enhanced description based on agent type that follows Claude's format.
951
+
952
+ Args:
953
+ raw_description: Original description
954
+ agent_name: Name of the agent
955
+ agent_type: Type of agent (engineer, qa, research, etc.)
956
+ template_data: Template data for additional context
957
+
958
+ Returns:
959
+ Enhanced description string
960
+ """
961
+ # Type-specific enhanced descriptions following Claude's software-engineer.md pattern
962
+ enhanced_descriptions = {
963
+ "engineer": "Use this agent when you need to implement new features, write production-quality code, refactor existing code, or solve complex programming challenges. This agent excels at translating requirements into well-architected, maintainable code solutions across various programming languages and frameworks.",
964
+ "qa": "Use this agent when you need comprehensive testing, quality assurance validation, or test automation. This agent specializes in creating robust test suites, identifying edge cases, and ensuring code quality through systematic testing approaches across different testing methodologies.",
965
+ "research": "Use this agent when you need to investigate codebases, analyze system architecture, or gather technical insights. This agent excels at code exploration, pattern identification, and providing comprehensive analysis of existing systems while maintaining strict memory efficiency.",
966
+ "ops": "Use this agent when you need infrastructure management, deployment automation, or operational excellence. This agent specializes in DevOps practices, cloud operations, monitoring setup, and maintaining reliable production systems.",
967
+ "security": "Use this agent when you need security analysis, vulnerability assessment, or secure coding practices. This agent excels at identifying security risks, implementing security best practices, and ensuring applications meet security standards.",
968
+ "documentation": "Use this agent when you need to create, update, or maintain technical documentation. This agent specializes in writing clear, comprehensive documentation including API docs, user guides, and technical specifications.",
969
+ }
970
+
971
+ # Get the enhanced description or fallback to the original with improvements
972
+ if agent_type in enhanced_descriptions:
973
+ return enhanced_descriptions[agent_type]
974
+ # Enhance the raw description if it's a custom type
975
+ if raw_description and len(raw_description) > 10:
976
+ return f"Use this agent when you need specialized assistance with {raw_description.lower()}. This agent provides targeted expertise and follows best practices for {agent_name.replace('-', ' ')} related tasks."
977
+ return f"Use this agent when you need specialized assistance from the {agent_name.replace('-', ' ')} agent. This agent provides targeted expertise and follows established best practices."
978
+
979
+ def _extract_examples_from_template(
980
+ self, template_data: dict, agent_name: str
981
+ ) -> List[str]:
982
+ """
983
+ Extract examples from template data and format with commentary.
984
+ Creates ONE example with commentary from template data.
985
+
986
+ Args:
987
+ template_data: Template data
988
+ agent_name: Name of the agent
989
+
990
+ Returns:
991
+ List of example strings (single example with commentary)
992
+ """
993
+ examples = []
994
+
995
+ # Check for examples in knowledge section
996
+ knowledge = template_data.get("knowledge", {})
997
+ template_examples = knowledge.get("examples", [])
998
+
999
+ if template_examples:
1000
+ # Take only the first example and add commentary
1001
+ example = template_examples[0]
1002
+ scenario = example.get("scenario", "")
1003
+ approach = example.get("approach", "")
1004
+ commentary = example.get("commentary", "")
1005
+
1006
+ if scenario and approach:
1007
+ examples.extend(
1008
+ [
1009
+ "<example>",
1010
+ f"Context: {scenario}",
1011
+ f'user: "I need help with {scenario.lower()}"',
1012
+ f'assistant: "I\'ll use the {agent_name} agent to {approach.lower()}."',
1013
+ "<commentary>",
1014
+ (
1015
+ commentary
1016
+ if commentary
1017
+ else f"This agent is well-suited for {scenario.lower()} because it specializes in {approach.lower()} with targeted expertise."
1018
+ ),
1019
+ "</commentary>",
1020
+ "</example>",
1021
+ ]
1022
+ )
1023
+
1024
+ # Check for triggers that can be converted to examples
1025
+ interactions = template_data.get("interactions", {})
1026
+ triggers = interactions.get("triggers", [])
1027
+
1028
+ if triggers and not examples:
1029
+ # Convert first trigger to example with commentary
1030
+ trigger = triggers[0]
1031
+
1032
+ # Handle both string and dict trigger formats
1033
+ if isinstance(trigger, dict):
1034
+ # New format with pattern and confidence
1035
+ trigger_text = trigger.get("pattern", "")
1036
+ else:
1037
+ # Old format with simple string
1038
+ trigger_text = str(trigger)
1039
+
1040
+ # Skip if we don't have valid trigger text
1041
+ if not trigger_text:
1042
+ return examples
1043
+
1044
+ agent_type = template_data.get("agent_type", "general")
1045
+
1046
+ examples.extend(
1047
+ [
1048
+ "<example>",
1049
+ f"Context: When user needs {trigger_text}",
1050
+ f'user: "{trigger_text}"',
1051
+ f'assistant: "I\'ll use the {agent_name} agent for {trigger_text}."',
1052
+ "<commentary>",
1053
+ f"This {agent_type} agent is appropriate because it has specialized capabilities for {trigger_text.lower()} tasks.",
1054
+ "</commentary>",
1055
+ "</example>",
1056
+ ]
1057
+ )
1058
+
1059
+ return examples
1060
+
1061
+ def _generate_default_examples(
1062
+ self, agent_name: str, template_data: dict
1063
+ ) -> List[str]:
1064
+ """
1065
+ Generate default examples when none are available in template.
1066
+ Creates ONE example with commentary for each agent type.
1067
+
1068
+ Args:
1069
+ agent_name: Name of the agent
1070
+ template_data: Template data for context
1071
+
1072
+ Returns:
1073
+ List of example strings (single example with commentary)
1074
+ """
1075
+ agent_type = template_data.get("agent_type", "general")
1076
+
1077
+ # Create type-specific examples with commentary inside
1078
+ type_examples = {
1079
+ "engineer": [
1080
+ "<example>",
1081
+ "Context: When you need to implement new features or write code.",
1082
+ 'user: "I need to add authentication to my API"',
1083
+ f'assistant: "I\'ll use the {agent_name} agent to implement a secure authentication system for your API."',
1084
+ "<commentary>",
1085
+ "The engineer agent is ideal for code implementation tasks because it specializes in writing production-quality code, following best practices, and creating well-architected solutions.",
1086
+ "</commentary>",
1087
+ "</example>",
1088
+ ],
1089
+ "ops": [
1090
+ "<example>",
1091
+ "Context: When you need to deploy or manage infrastructure.",
1092
+ 'user: "I need to deploy my application to the cloud"',
1093
+ f'assistant: "I\'ll use the {agent_name} agent to set up and deploy your application infrastructure."',
1094
+ "<commentary>",
1095
+ "The ops agent excels at infrastructure management and deployment automation, ensuring reliable and scalable production systems.",
1096
+ "</commentary>",
1097
+ "</example>",
1098
+ ],
1099
+ "qa": [
1100
+ "<example>",
1101
+ "Context: When you need to test or validate functionality.",
1102
+ 'user: "I need to write tests for my new feature"',
1103
+ f'assistant: "I\'ll use the {agent_name} agent to create comprehensive tests for your feature."',
1104
+ "<commentary>",
1105
+ "The QA agent specializes in comprehensive testing strategies, quality assurance validation, and creating robust test suites that ensure code reliability.",
1106
+ "</commentary>",
1107
+ "</example>",
1108
+ ],
1109
+ "research": [
1110
+ "<example>",
1111
+ "Context: When you need to investigate or analyze existing codebases.",
1112
+ 'user: "I need to understand how the authentication system works in this project"',
1113
+ f'assistant: "I\'ll use the {agent_name} agent to analyze the codebase and explain the authentication implementation."',
1114
+ "<commentary>",
1115
+ "The research agent is perfect for code exploration and analysis tasks, providing thorough investigation of existing systems while maintaining memory efficiency.",
1116
+ "</commentary>",
1117
+ "</example>",
1118
+ ],
1119
+ "security": [
1120
+ "<example>",
1121
+ "Context: When you need to review code for security vulnerabilities.",
1122
+ 'user: "I need a security review of my authentication implementation"',
1123
+ f'assistant: "I\'ll use the {agent_name} agent to conduct a thorough security analysis of your authentication code."',
1124
+ "<commentary>",
1125
+ "The security agent specializes in identifying security risks, vulnerability assessment, and ensuring applications meet security standards and best practices.",
1126
+ "</commentary>",
1127
+ "</example>",
1128
+ ],
1129
+ "documentation": [
1130
+ "<example>",
1131
+ "Context: When you need to create or update technical documentation.",
1132
+ 'user: "I need to document this new API endpoint"',
1133
+ f'assistant: "I\'ll use the {agent_name} agent to create comprehensive API documentation."',
1134
+ "<commentary>",
1135
+ "The documentation agent excels at creating clear, comprehensive technical documentation including API docs, user guides, and technical specifications.",
1136
+ "</commentary>",
1137
+ "</example>",
1138
+ ],
1139
+ }
1140
+
1141
+ return type_examples.get(
1142
+ agent_type,
1143
+ [
1144
+ "<example>",
1145
+ f"Context: When you need specialized assistance from the {agent_name} agent.",
1146
+ f'user: "I need help with {agent_name.replace("-", " ")} tasks"',
1147
+ f'assistant: "I\'ll use the {agent_name} agent to provide specialized assistance."',
1148
+ "<commentary>",
1149
+ f"This agent provides targeted expertise for {agent_name.replace('-', ' ')} related tasks and follows established best practices.",
1150
+ "</commentary>",
1151
+ "</example>",
1152
+ ],
1153
+ )
1154
+
1155
+ def _indent_multiline_text(self, text: str, spaces: int) -> str:
1156
+ """
1157
+ Indent multiline text with specified number of spaces.
1158
+
1159
+ Args:
1160
+ text: Text to indent
1161
+ spaces: Number of spaces for indentation
1162
+
1163
+ Returns:
1164
+ Indented text
1165
+ """
1166
+ if not text:
1167
+ return ""
1168
+
1169
+ indent = " " * spaces
1170
+ lines = text.split("\n")
1171
+ indented_lines = []
1172
+
1173
+ for line in lines:
1174
+ if line.strip(): # Non-empty lines get indented
1175
+ indented_lines.append(indent + line)
1176
+ else: # Empty lines stay empty
1177
+ indented_lines.append("")
1178
+
1179
+ return "\n".join(indented_lines)
1180
+
1181
+ def _format_description_for_yaml(self, description: str) -> str:
1182
+ """Format description as a single-line YAML string with escaped newlines.
1183
+
1184
+ Args:
1185
+ description: Multi-line description text
1186
+
1187
+ Returns:
1188
+ Single-line YAML-formatted string with \n escapes
1189
+ """
1190
+ if not description:
1191
+ return '""'
1192
+
1193
+ # The description already contains actual newlines, we need to escape them
1194
+ # Replace actual newlines with \n escape sequence
1195
+ escaped = description.replace("\n", "\\n")
1196
+
1197
+ # Escape any quotes in the description
1198
+ escaped = escaped.replace('"', '\\"')
1199
+
1200
+ # Return as quoted string
1201
+ return f'"{escaped}"'
1202
+
1203
+ def _convert_instructions_dict_to_markdown(self, instructions_dict: dict) -> str:
1204
+ """Convert complex instructions dictionary to markdown format.
1205
+
1206
+ Args:
1207
+ instructions_dict: Dictionary containing structured instructions
1208
+
1209
+ Returns:
1210
+ Formatted markdown string representing the instructions
1211
+ """
1212
+ if not instructions_dict:
1213
+ return "# Agent Instructions\n\nThis agent provides specialized assistance."
1214
+
1215
+ markdown_parts = []
1216
+
1217
+ # Add primary role
1218
+ if "primary_role" in instructions_dict:
1219
+ markdown_parts.extend(["# Role", "", instructions_dict["primary_role"], ""])
1220
+
1221
+ # Add core identity
1222
+ if "core_identity" in instructions_dict:
1223
+ markdown_parts.extend(
1224
+ ["## Core Identity", "", instructions_dict["core_identity"], ""]
1225
+ )
1226
+
1227
+ # Add responsibilities
1228
+ if "responsibilities" in instructions_dict:
1229
+ markdown_parts.extend(["## Responsibilities", ""])
1230
+
1231
+ responsibilities = instructions_dict["responsibilities"]
1232
+ if isinstance(responsibilities, list):
1233
+ for resp in responsibilities:
1234
+ if isinstance(resp, dict):
1235
+ area = resp.get("area", "Unknown Area")
1236
+ tasks = resp.get("tasks", [])
1237
+
1238
+ markdown_parts.extend([f"### {area}", ""])
1239
+
1240
+ if isinstance(tasks, list):
1241
+ for task in tasks:
1242
+ markdown_parts.append(f"- {task}")
1243
+
1244
+ markdown_parts.append("")
1245
+ else:
1246
+ markdown_parts.append(f"- {resp}")
1247
+
1248
+ markdown_parts.append("")
1249
+
1250
+ # Add analytical framework
1251
+ if "analytical_framework" in instructions_dict:
1252
+ framework = instructions_dict["analytical_framework"]
1253
+ if isinstance(framework, dict):
1254
+ markdown_parts.extend(["## Analytical Framework", ""])
1255
+
1256
+ for framework_area, framework_data in framework.items():
1257
+ markdown_parts.extend(
1258
+ [f"### {framework_area.replace('_', ' ').title()}", ""]
1259
+ )
1260
+
1261
+ if isinstance(framework_data, dict):
1262
+ for category, items in framework_data.items():
1263
+ markdown_parts.extend(
1264
+ [f"#### {category.replace('_', ' ').title()}", ""]
1265
+ )
1266
+
1267
+ if isinstance(items, list):
1268
+ for item in items:
1269
+ markdown_parts.append(f"- {item}")
1270
+ elif isinstance(items, str):
1271
+ markdown_parts.append(items)
1272
+
1273
+ markdown_parts.append("")
1274
+ elif isinstance(framework_data, list):
1275
+ for item in framework_data:
1276
+ markdown_parts.append(f"- {item}")
1277
+ markdown_parts.append("")
1278
+
1279
+ # Add methodologies
1280
+ if "methodologies" in instructions_dict:
1281
+ methodologies = instructions_dict["methodologies"]
1282
+ if isinstance(methodologies, dict):
1283
+ markdown_parts.extend(["## Methodologies", ""])
1284
+
1285
+ for method_name, method_data in methodologies.items():
1286
+ markdown_parts.extend(
1287
+ [f"### {method_name.replace('_', ' ').title()}", ""]
1288
+ )
1289
+
1290
+ if isinstance(method_data, dict):
1291
+ for key, value in method_data.items():
1292
+ if isinstance(value, list):
1293
+ markdown_parts.extend(
1294
+ [f"#### {key.replace('_', ' ').title()}", ""]
1295
+ )
1296
+ for item in value:
1297
+ markdown_parts.append(f"- {item}")
1298
+ markdown_parts.append("")
1299
+ elif isinstance(value, str):
1300
+ markdown_parts.extend(
1301
+ [
1302
+ f"**{key.replace('_', ' ').title()}**: {value}",
1303
+ "",
1304
+ ]
1305
+ )
1306
+
1307
+ # Add quality standards
1308
+ if "quality_standards" in instructions_dict:
1309
+ standards = instructions_dict["quality_standards"]
1310
+ if isinstance(standards, dict):
1311
+ markdown_parts.extend(["## Quality Standards", ""])
1312
+
1313
+ for standard_area, standard_items in standards.items():
1314
+ markdown_parts.extend(
1315
+ [f"### {standard_area.replace('_', ' ').title()}", ""]
1316
+ )
1317
+
1318
+ if isinstance(standard_items, list):
1319
+ for item in standard_items:
1320
+ markdown_parts.append(f"- {item}")
1321
+ elif isinstance(standard_items, str):
1322
+ markdown_parts.append(standard_items)
1323
+
1324
+ markdown_parts.append("")
1325
+
1326
+ # Add communication style
1327
+ if "communication_style" in instructions_dict:
1328
+ comm_style = instructions_dict["communication_style"]
1329
+ if isinstance(comm_style, dict):
1330
+ markdown_parts.extend(["## Communication Style", ""])
1331
+
1332
+ for style_area, style_items in comm_style.items():
1333
+ markdown_parts.extend(
1334
+ [f"### {style_area.replace('_', ' ').title()}", ""]
1335
+ )
1336
+
1337
+ if isinstance(style_items, list):
1338
+ for item in style_items:
1339
+ markdown_parts.append(f"- {item}")
1340
+ elif isinstance(style_items, str):
1341
+ markdown_parts.append(style_items)
1342
+
1343
+ markdown_parts.append("")
1344
+
1345
+ # If no specific sections were found, convert as generic dict
1346
+ if not markdown_parts:
1347
+ markdown_parts = ["# Agent Instructions", ""]
1348
+ for key, value in instructions_dict.items():
1349
+ key_title = key.replace("_", " ").title()
1350
+ if isinstance(value, str):
1351
+ markdown_parts.extend([f"## {key_title}", "", value, ""])
1352
+ elif isinstance(value, list):
1353
+ markdown_parts.extend([f"## {key_title}", ""])
1354
+ for item in value:
1355
+ markdown_parts.append(f"- {item}")
1356
+ markdown_parts.append("")
1357
+ elif isinstance(value, dict):
1358
+ markdown_parts.extend([f"## {key_title}", ""])
1359
+ # Simple dict formatting
1360
+ for subkey, subvalue in value.items():
1361
+ if isinstance(subvalue, str):
1362
+ markdown_parts.extend(
1363
+ [
1364
+ f"**{subkey.replace('_', ' ').title()}**: {subvalue}",
1365
+ "",
1366
+ ]
1367
+ )
1368
+
1369
+ return "\n".join(markdown_parts).strip()