claude-mpm 5.4.22__py3-none-any.whl → 5.6.34__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.

Potentially problematic release.


This version of claude-mpm might be problematic. Click here for more details.

Files changed (487) hide show
  1. claude_mpm/VERSION +1 -1
  2. claude_mpm/agents/BASE_AGENT.md +164 -0
  3. claude_mpm/agents/BASE_ENGINEER.md +658 -0
  4. claude_mpm/agents/CLAUDE_MPM_OUTPUT_STYLE.md +66 -241
  5. claude_mpm/agents/CLAUDE_MPM_RESEARCH_OUTPUT_STYLE.md +413 -0
  6. claude_mpm/agents/CLAUDE_MPM_TEACHER_OUTPUT_STYLE.md +109 -1925
  7. claude_mpm/agents/MEMORY.md +1 -1
  8. claude_mpm/agents/PM_INSTRUCTIONS.md +374 -1257
  9. claude_mpm/agents/WORKFLOW.md +6 -253
  10. claude_mpm/agents/agent_loader.py +1 -1
  11. claude_mpm/agents/base_agent.json +31 -0
  12. claude_mpm/agents/frontmatter_validator.py +2 -2
  13. claude_mpm/agents/templates/circuit-breakers.md +26 -17
  14. claude_mpm/cli/__init__.py +5 -1
  15. claude_mpm/cli/commands/agent_state_manager.py +10 -10
  16. claude_mpm/cli/commands/agents.py +11 -13
  17. claude_mpm/cli/commands/agents_reconcile.py +197 -0
  18. claude_mpm/cli/commands/auto_configure.py +4 -4
  19. claude_mpm/cli/commands/autotodos.py +566 -0
  20. claude_mpm/cli/commands/commander.py +216 -0
  21. claude_mpm/cli/commands/configure.py +621 -22
  22. claude_mpm/cli/commands/configure_agent_display.py +12 -0
  23. claude_mpm/cli/commands/hook_errors.py +60 -60
  24. claude_mpm/cli/commands/monitor.py +2 -2
  25. claude_mpm/cli/commands/mpm_init/core.py +72 -0
  26. claude_mpm/cli/commands/postmortem.py +1 -1
  27. claude_mpm/cli/commands/profile.py +276 -0
  28. claude_mpm/cli/commands/run.py +35 -3
  29. claude_mpm/cli/commands/skill_source.py +51 -2
  30. claude_mpm/cli/commands/skills.py +182 -32
  31. claude_mpm/cli/executor.py +130 -16
  32. claude_mpm/cli/interactive/__init__.py +10 -0
  33. claude_mpm/cli/interactive/agent_wizard.py +32 -52
  34. claude_mpm/cli/interactive/questionary_styles.py +65 -0
  35. claude_mpm/cli/interactive/skill_selector.py +481 -0
  36. claude_mpm/cli/parsers/base_parser.py +83 -1
  37. claude_mpm/cli/parsers/commander_parser.py +116 -0
  38. claude_mpm/cli/parsers/profile_parser.py +147 -0
  39. claude_mpm/cli/parsers/run_parser.py +10 -0
  40. claude_mpm/cli/parsers/skill_source_parser.py +4 -0
  41. claude_mpm/cli/parsers/skills_parser.py +2 -3
  42. claude_mpm/cli/startup.py +690 -386
  43. claude_mpm/cli/startup_display.py +74 -6
  44. claude_mpm/cli/startup_logging.py +2 -2
  45. claude_mpm/cli/utils.py +7 -3
  46. claude_mpm/commander/__init__.py +78 -0
  47. claude_mpm/commander/adapters/__init__.py +60 -0
  48. claude_mpm/commander/adapters/auggie.py +260 -0
  49. claude_mpm/commander/adapters/base.py +288 -0
  50. claude_mpm/commander/adapters/claude_code.py +392 -0
  51. claude_mpm/commander/adapters/codex.py +237 -0
  52. claude_mpm/commander/adapters/communication.py +366 -0
  53. claude_mpm/commander/adapters/example_usage.py +310 -0
  54. claude_mpm/commander/adapters/mpm.py +389 -0
  55. claude_mpm/commander/adapters/registry.py +204 -0
  56. claude_mpm/commander/api/__init__.py +16 -0
  57. claude_mpm/commander/api/app.py +121 -0
  58. claude_mpm/commander/api/errors.py +133 -0
  59. claude_mpm/commander/api/routes/__init__.py +8 -0
  60. claude_mpm/commander/api/routes/events.py +184 -0
  61. claude_mpm/commander/api/routes/inbox.py +171 -0
  62. claude_mpm/commander/api/routes/messages.py +148 -0
  63. claude_mpm/commander/api/routes/projects.py +271 -0
  64. claude_mpm/commander/api/routes/sessions.py +226 -0
  65. claude_mpm/commander/api/routes/work.py +296 -0
  66. claude_mpm/commander/api/schemas.py +186 -0
  67. claude_mpm/commander/chat/__init__.py +7 -0
  68. claude_mpm/commander/chat/cli.py +146 -0
  69. claude_mpm/commander/chat/commands.py +96 -0
  70. claude_mpm/commander/chat/repl.py +310 -0
  71. claude_mpm/commander/config.py +51 -0
  72. claude_mpm/commander/config_loader.py +115 -0
  73. claude_mpm/commander/core/__init__.py +10 -0
  74. claude_mpm/commander/core/block_manager.py +325 -0
  75. claude_mpm/commander/core/response_manager.py +323 -0
  76. claude_mpm/commander/daemon.py +603 -0
  77. claude_mpm/commander/env_loader.py +59 -0
  78. claude_mpm/commander/events/__init__.py +26 -0
  79. claude_mpm/commander/events/manager.py +332 -0
  80. claude_mpm/commander/frameworks/__init__.py +12 -0
  81. claude_mpm/commander/frameworks/base.py +146 -0
  82. claude_mpm/commander/frameworks/claude_code.py +58 -0
  83. claude_mpm/commander/frameworks/mpm.py +62 -0
  84. claude_mpm/commander/inbox/__init__.py +16 -0
  85. claude_mpm/commander/inbox/dedup.py +128 -0
  86. claude_mpm/commander/inbox/inbox.py +224 -0
  87. claude_mpm/commander/inbox/models.py +70 -0
  88. claude_mpm/commander/instance_manager.py +450 -0
  89. claude_mpm/commander/llm/__init__.py +6 -0
  90. claude_mpm/commander/llm/openrouter_client.py +167 -0
  91. claude_mpm/commander/llm/summarizer.py +70 -0
  92. claude_mpm/commander/memory/__init__.py +45 -0
  93. claude_mpm/commander/memory/compression.py +347 -0
  94. claude_mpm/commander/memory/embeddings.py +230 -0
  95. claude_mpm/commander/memory/entities.py +310 -0
  96. claude_mpm/commander/memory/example_usage.py +290 -0
  97. claude_mpm/commander/memory/integration.py +325 -0
  98. claude_mpm/commander/memory/search.py +381 -0
  99. claude_mpm/commander/memory/store.py +657 -0
  100. claude_mpm/commander/models/__init__.py +18 -0
  101. claude_mpm/commander/models/events.py +121 -0
  102. claude_mpm/commander/models/project.py +162 -0
  103. claude_mpm/commander/models/work.py +214 -0
  104. claude_mpm/commander/parsing/__init__.py +20 -0
  105. claude_mpm/commander/parsing/extractor.py +132 -0
  106. claude_mpm/commander/parsing/output_parser.py +270 -0
  107. claude_mpm/commander/parsing/patterns.py +100 -0
  108. claude_mpm/commander/persistence/__init__.py +11 -0
  109. claude_mpm/commander/persistence/event_store.py +274 -0
  110. claude_mpm/commander/persistence/state_store.py +309 -0
  111. claude_mpm/commander/persistence/work_store.py +164 -0
  112. claude_mpm/commander/polling/__init__.py +13 -0
  113. claude_mpm/commander/polling/event_detector.py +104 -0
  114. claude_mpm/commander/polling/output_buffer.py +49 -0
  115. claude_mpm/commander/polling/output_poller.py +153 -0
  116. claude_mpm/commander/project_session.py +268 -0
  117. claude_mpm/commander/proxy/__init__.py +12 -0
  118. claude_mpm/commander/proxy/formatter.py +89 -0
  119. claude_mpm/commander/proxy/output_handler.py +191 -0
  120. claude_mpm/commander/proxy/relay.py +155 -0
  121. claude_mpm/commander/registry.py +410 -0
  122. claude_mpm/commander/runtime/__init__.py +10 -0
  123. claude_mpm/commander/runtime/executor.py +191 -0
  124. claude_mpm/commander/runtime/monitor.py +346 -0
  125. claude_mpm/commander/session/__init__.py +6 -0
  126. claude_mpm/commander/session/context.py +81 -0
  127. claude_mpm/commander/session/manager.py +59 -0
  128. claude_mpm/commander/tmux_orchestrator.py +361 -0
  129. claude_mpm/commander/web/__init__.py +1 -0
  130. claude_mpm/commander/work/__init__.py +30 -0
  131. claude_mpm/commander/work/executor.py +207 -0
  132. claude_mpm/commander/work/queue.py +405 -0
  133. claude_mpm/commander/workflow/__init__.py +27 -0
  134. claude_mpm/commander/workflow/event_handler.py +241 -0
  135. claude_mpm/commander/workflow/notifier.py +146 -0
  136. claude_mpm/commands/mpm-config.md +20 -249
  137. claude_mpm/commands/mpm-doctor.md +16 -21
  138. claude_mpm/commands/mpm-help.md +12 -205
  139. claude_mpm/commands/mpm-init.md +88 -506
  140. claude_mpm/commands/mpm-monitor.md +22 -401
  141. claude_mpm/commands/mpm-organize.md +70 -442
  142. claude_mpm/commands/mpm-postmortem.md +13 -107
  143. claude_mpm/commands/mpm-session-resume.md +20 -363
  144. claude_mpm/commands/mpm-status.md +13 -69
  145. claude_mpm/commands/mpm-ticket-view.md +60 -495
  146. claude_mpm/commands/mpm-version.md +13 -107
  147. claude_mpm/commands/mpm.md +8 -0
  148. claude_mpm/config/agent_presets.py +8 -7
  149. claude_mpm/config/skill_sources.py +16 -0
  150. claude_mpm/constants.py +1 -0
  151. claude_mpm/core/claude_runner.py +154 -2
  152. claude_mpm/core/config.py +37 -26
  153. claude_mpm/core/config_constants.py +74 -9
  154. claude_mpm/core/constants.py +56 -12
  155. claude_mpm/core/framework/loaders/agent_loader.py +1 -1
  156. claude_mpm/core/framework/loaders/instruction_loader.py +52 -11
  157. claude_mpm/core/hook_manager.py +51 -3
  158. claude_mpm/core/interactive_session.py +12 -11
  159. claude_mpm/core/logger.py +26 -9
  160. claude_mpm/core/logging_utils.py +39 -13
  161. claude_mpm/core/network_config.py +148 -0
  162. claude_mpm/core/oneshot_session.py +7 -6
  163. claude_mpm/core/optimized_startup.py +61 -0
  164. claude_mpm/core/output_style_manager.py +66 -18
  165. claude_mpm/core/shared/config_loader.py +3 -1
  166. claude_mpm/core/socketio_pool.py +47 -15
  167. claude_mpm/core/unified_agent_registry.py +1 -1
  168. claude_mpm/core/unified_config.py +54 -8
  169. claude_mpm/core/unified_paths.py +95 -90
  170. claude_mpm/dashboard/static/svelte-build/_app/env.js +1 -0
  171. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/0.C33zOoyM.css +1 -0
  172. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/2.CW1J-YuA.css +1 -0
  173. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/1WZnGYqX.js +24 -0
  174. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/67pF3qNn.js +1 -0
  175. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/6RxdMKe4.js +1 -0
  176. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/8cZrfX0h.js +60 -0
  177. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/9a6T2nm-.js +7 -0
  178. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/B443AUzu.js +1 -0
  179. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/B8AwtY2H.js +1 -0
  180. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BF15LAsF.js +1 -0
  181. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BQaXIfA_.js +331 -0
  182. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BRcwIQNr.js +4 -0
  183. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BSNlmTZj.js +1 -0
  184. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BV6nKitt.js +43 -0
  185. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BViJ8lZt.js +128 -0
  186. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BcQ-Q0FE.js +1 -0
  187. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Bpyvgze_.js +30 -0
  188. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BzTRqg-z.js +1 -0
  189. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C0Fr8dve.js +1 -0
  190. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C3rbW_a-.js +1 -0
  191. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C8WYN38h.js +1 -0
  192. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C9I8FlXH.js +61 -0
  193. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CIQcWgO2.js +36 -0
  194. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CIctN7YN.js +7 -0
  195. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CKrS_JZW.js +145 -0
  196. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CR6P9C4A.js +89 -0
  197. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CRRR9MD_.js +2 -0
  198. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CRcR2DqT.js +334 -0
  199. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CSXtMOf0.js +1 -0
  200. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CT-sbxSk.js +1 -0
  201. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CWm6DJsp.js +1 -0
  202. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CmKTTxBW.js +1 -0
  203. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CpqQ1Kzn.js +1 -0
  204. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Cu_Erd72.js +261 -0
  205. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/D2nGpDRe.js +1 -0
  206. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/D9iCMida.js +267 -0
  207. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/D9ykgMoY.js +10 -0
  208. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DL2Ldur1.js +1 -0
  209. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DPfltzjH.js +165 -0
  210. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DR8nis88.js +2 -0
  211. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DUliQN2b.js +1 -0
  212. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DVp1hx9R.js +1 -0
  213. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DXlhR01x.js +122 -0
  214. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/D_lyTybS.js +1 -0
  215. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DngoTTgh.js +1 -0
  216. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DqkmHtDC.js +220 -0
  217. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DsDh8EYs.js +1 -0
  218. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DypDmXgd.js +139 -0
  219. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Gi6I4Gst.js +1 -0
  220. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/IPYC-LnN.js +162 -0
  221. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/JTLiF7dt.js +24 -0
  222. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/JpevfAFt.js +68 -0
  223. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/NqQ1dWOy.js +1 -0
  224. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/R8CEIRAd.js +2 -0
  225. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Zxy7qc-l.js +64 -0
  226. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/q9Hm6zAU.js +1 -0
  227. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/qtd3IeO4.js +15 -0
  228. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/ulBFON_C.js +65 -0
  229. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/wQVh1CoA.js +10 -0
  230. claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/app.Dr7t0z2J.js +2 -0
  231. claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/start.BGhZHUS3.js +1 -0
  232. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/0.RgBboRvH.js +1 -0
  233. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/1.DG-KkbDf.js +1 -0
  234. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/2.D_jnf-x6.js +1 -0
  235. claude_mpm/dashboard/static/svelte-build/_app/version.json +1 -0
  236. claude_mpm/dashboard/static/svelte-build/favicon.svg +7 -0
  237. claude_mpm/dashboard/static/svelte-build/index.html +36 -0
  238. claude_mpm/dashboard-svelte/node_modules/katex/src/fonts/generate_fonts.py +58 -0
  239. claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/extract_tfms.py +114 -0
  240. claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/extract_ttfs.py +122 -0
  241. claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/format_json.py +28 -0
  242. claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/parse_tfm.py +211 -0
  243. claude_mpm/experimental/cli_enhancements.py +2 -1
  244. claude_mpm/hooks/claude_hooks/INTEGRATION_EXAMPLE.md +243 -0
  245. claude_mpm/hooks/claude_hooks/README_AUTO_PAUSE.md +403 -0
  246. claude_mpm/hooks/claude_hooks/__pycache__/__init__.cpython-311.pyc +0 -0
  247. claude_mpm/hooks/claude_hooks/__pycache__/auto_pause_handler.cpython-311.pyc +0 -0
  248. claude_mpm/hooks/claude_hooks/__pycache__/correlation_manager.cpython-311.pyc +0 -0
  249. claude_mpm/hooks/claude_hooks/__pycache__/event_handlers.cpython-311.pyc +0 -0
  250. claude_mpm/hooks/claude_hooks/__pycache__/hook_handler.cpython-311.pyc +0 -0
  251. claude_mpm/hooks/claude_hooks/__pycache__/memory_integration.cpython-311.pyc +0 -0
  252. claude_mpm/hooks/claude_hooks/__pycache__/response_tracking.cpython-311.pyc +0 -0
  253. claude_mpm/hooks/claude_hooks/__pycache__/tool_analysis.cpython-311.pyc +0 -0
  254. claude_mpm/hooks/claude_hooks/auto_pause_handler.py +485 -0
  255. claude_mpm/hooks/claude_hooks/event_handlers.py +527 -136
  256. claude_mpm/hooks/claude_hooks/hook_handler.py +313 -99
  257. claude_mpm/hooks/claude_hooks/hook_wrapper.sh +6 -11
  258. claude_mpm/hooks/claude_hooks/installer.py +206 -36
  259. claude_mpm/hooks/claude_hooks/memory_integration.py +52 -32
  260. claude_mpm/hooks/claude_hooks/response_tracking.py +43 -60
  261. claude_mpm/hooks/claude_hooks/services/__init__.py +21 -0
  262. claude_mpm/hooks/claude_hooks/services/__pycache__/__init__.cpython-311.pyc +0 -0
  263. claude_mpm/hooks/claude_hooks/services/__pycache__/connection_manager_http.cpython-311.pyc +0 -0
  264. claude_mpm/hooks/claude_hooks/services/__pycache__/container.cpython-311.pyc +0 -0
  265. claude_mpm/hooks/claude_hooks/services/__pycache__/duplicate_detector.cpython-311.pyc +0 -0
  266. claude_mpm/hooks/claude_hooks/services/__pycache__/protocols.cpython-311.pyc +0 -0
  267. claude_mpm/hooks/claude_hooks/services/__pycache__/state_manager.cpython-311.pyc +0 -0
  268. claude_mpm/hooks/claude_hooks/services/__pycache__/subagent_processor.cpython-311.pyc +0 -0
  269. claude_mpm/hooks/claude_hooks/services/connection_manager.py +67 -32
  270. claude_mpm/hooks/claude_hooks/services/connection_manager_http.py +38 -105
  271. claude_mpm/hooks/claude_hooks/services/container.py +310 -0
  272. claude_mpm/hooks/claude_hooks/services/protocols.py +328 -0
  273. claude_mpm/hooks/claude_hooks/services/state_manager.py +25 -38
  274. claude_mpm/hooks/claude_hooks/services/subagent_processor.py +75 -77
  275. claude_mpm/hooks/kuzu_memory_hook.py +5 -5
  276. claude_mpm/hooks/session_resume_hook.py +89 -1
  277. claude_mpm/hooks/templates/pre_tool_use_simple.py +6 -6
  278. claude_mpm/hooks/templates/pre_tool_use_template.py +16 -8
  279. claude_mpm/init.py +276 -0
  280. claude_mpm/models/git_repository.py +3 -3
  281. claude_mpm/scripts/claude-hook-handler.sh +46 -19
  282. claude_mpm/services/agents/agent_builder.py +3 -3
  283. claude_mpm/services/agents/agent_recommendation_service.py +8 -8
  284. claude_mpm/services/agents/agent_selection_service.py +2 -2
  285. claude_mpm/services/agents/cache_git_manager.py +7 -7
  286. claude_mpm/services/agents/deployment/agent_deployment.py +29 -7
  287. claude_mpm/services/agents/deployment/agent_discovery_service.py +4 -2
  288. claude_mpm/services/agents/deployment/agent_format_converter.py +25 -13
  289. claude_mpm/services/agents/deployment/agent_template_builder.py +39 -19
  290. claude_mpm/services/agents/deployment/agents_directory_resolver.py +2 -2
  291. claude_mpm/services/agents/deployment/async_agent_deployment.py +31 -27
  292. claude_mpm/services/agents/deployment/deployment_reconciler.py +577 -0
  293. claude_mpm/services/agents/deployment/local_template_deployment.py +3 -1
  294. claude_mpm/services/agents/deployment/multi_source_deployment_service.py +169 -26
  295. claude_mpm/services/agents/deployment/remote_agent_discovery_service.py +101 -75
  296. claude_mpm/services/agents/deployment/startup_reconciliation.py +138 -0
  297. claude_mpm/services/agents/git_source_manager.py +23 -4
  298. claude_mpm/services/agents/loading/framework_agent_loader.py +75 -2
  299. claude_mpm/services/agents/recommender.py +5 -3
  300. claude_mpm/services/agents/single_tier_deployment_service.py +6 -6
  301. claude_mpm/services/agents/sources/git_source_sync_service.py +121 -10
  302. claude_mpm/services/agents/startup_sync.py +27 -4
  303. claude_mpm/services/cli/__init__.py +3 -0
  304. claude_mpm/services/cli/incremental_pause_manager.py +561 -0
  305. claude_mpm/services/cli/session_resume_helper.py +10 -2
  306. claude_mpm/services/command_deployment_service.py +44 -26
  307. claude_mpm/services/delegation_detector.py +175 -0
  308. claude_mpm/services/diagnostics/checks/agent_check.py +2 -2
  309. claude_mpm/services/diagnostics/checks/agent_sources_check.py +31 -1
  310. claude_mpm/services/diagnostics/checks/configuration_check.py +24 -0
  311. claude_mpm/services/diagnostics/checks/installation_check.py +22 -0
  312. claude_mpm/services/diagnostics/checks/mcp_services_check.py +23 -0
  313. claude_mpm/services/diagnostics/doctor_reporter.py +31 -1
  314. claude_mpm/services/diagnostics/models.py +14 -1
  315. claude_mpm/services/event_log.py +325 -0
  316. claude_mpm/services/git/git_operations_service.py +8 -8
  317. claude_mpm/services/hook_installer_service.py +77 -8
  318. claude_mpm/services/infrastructure/__init__.py +4 -0
  319. claude_mpm/services/infrastructure/context_usage_tracker.py +291 -0
  320. claude_mpm/services/infrastructure/resume_log_generator.py +24 -5
  321. claude_mpm/services/monitor/daemon_manager.py +15 -4
  322. claude_mpm/services/monitor/management/lifecycle.py +15 -3
  323. claude_mpm/services/monitor/server.py +571 -11
  324. claude_mpm/services/pm_skills_deployer.py +884 -0
  325. claude_mpm/services/profile_manager.py +337 -0
  326. claude_mpm/services/skills/git_skill_source_manager.py +281 -20
  327. claude_mpm/services/skills/selective_skill_deployer.py +211 -46
  328. claude_mpm/services/skills/skill_discovery_service.py +74 -4
  329. claude_mpm/services/skills_deployer.py +192 -70
  330. claude_mpm/services/socketio/dashboard_server.py +1 -0
  331. claude_mpm/services/socketio/event_normalizer.py +37 -6
  332. claude_mpm/services/socketio/handlers/hook.py +14 -7
  333. claude_mpm/services/socketio/server/core.py +262 -123
  334. claude_mpm/services/socketio/server/main.py +12 -4
  335. claude_mpm/skills/__init__.py +2 -1
  336. claude_mpm/skills/bundled/collaboration/brainstorming/SKILL.md +79 -0
  337. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/SKILL.md +178 -0
  338. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/agent-prompts.md +577 -0
  339. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/coordination-patterns.md +467 -0
  340. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/examples.md +537 -0
  341. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/troubleshooting.md +730 -0
  342. claude_mpm/skills/bundled/collaboration/git-worktrees.md +317 -0
  343. claude_mpm/skills/bundled/collaboration/requesting-code-review/SKILL.md +112 -0
  344. claude_mpm/skills/bundled/collaboration/requesting-code-review/references/code-reviewer-template.md +146 -0
  345. claude_mpm/skills/bundled/collaboration/requesting-code-review/references/review-examples.md +412 -0
  346. claude_mpm/skills/bundled/collaboration/stacked-prs.md +251 -0
  347. claude_mpm/skills/bundled/collaboration/writing-plans/SKILL.md +81 -0
  348. claude_mpm/skills/bundled/collaboration/writing-plans/references/best-practices.md +362 -0
  349. claude_mpm/skills/bundled/collaboration/writing-plans/references/plan-structure-templates.md +312 -0
  350. claude_mpm/skills/bundled/debugging/root-cause-tracing/SKILL.md +152 -0
  351. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/advanced-techniques.md +668 -0
  352. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/examples.md +587 -0
  353. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/integration.md +438 -0
  354. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/tracing-techniques.md +391 -0
  355. claude_mpm/skills/bundled/debugging/systematic-debugging/CREATION-LOG.md +119 -0
  356. claude_mpm/skills/bundled/debugging/systematic-debugging/SKILL.md +148 -0
  357. claude_mpm/skills/bundled/debugging/systematic-debugging/references/anti-patterns.md +483 -0
  358. claude_mpm/skills/bundled/debugging/systematic-debugging/references/examples.md +452 -0
  359. claude_mpm/skills/bundled/debugging/systematic-debugging/references/troubleshooting.md +449 -0
  360. claude_mpm/skills/bundled/debugging/systematic-debugging/references/workflow.md +411 -0
  361. claude_mpm/skills/bundled/debugging/systematic-debugging/test-academic.md +14 -0
  362. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-1.md +58 -0
  363. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-2.md +68 -0
  364. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-3.md +69 -0
  365. claude_mpm/skills/bundled/debugging/verification-before-completion/SKILL.md +131 -0
  366. claude_mpm/skills/bundled/debugging/verification-before-completion/references/gate-function.md +325 -0
  367. claude_mpm/skills/bundled/debugging/verification-before-completion/references/integration-and-workflows.md +490 -0
  368. claude_mpm/skills/bundled/debugging/verification-before-completion/references/red-flags-and-failures.md +425 -0
  369. claude_mpm/skills/bundled/debugging/verification-before-completion/references/verification-patterns.md +499 -0
  370. claude_mpm/skills/bundled/infrastructure/env-manager/INTEGRATION.md +611 -0
  371. claude_mpm/skills/bundled/infrastructure/env-manager/README.md +596 -0
  372. claude_mpm/skills/bundled/infrastructure/env-manager/SKILL.md +260 -0
  373. claude_mpm/skills/bundled/infrastructure/env-manager/examples/nextjs-env-structure.md +315 -0
  374. claude_mpm/skills/bundled/infrastructure/env-manager/references/frameworks.md +436 -0
  375. claude_mpm/skills/bundled/infrastructure/env-manager/references/security.md +433 -0
  376. claude_mpm/skills/bundled/infrastructure/env-manager/references/synchronization.md +452 -0
  377. claude_mpm/skills/bundled/infrastructure/env-manager/references/troubleshooting.md +404 -0
  378. claude_mpm/skills/bundled/infrastructure/env-manager/references/validation.md +420 -0
  379. claude_mpm/skills/bundled/main/artifacts-builder/SKILL.md +86 -0
  380. claude_mpm/skills/bundled/main/internal-comms/SKILL.md +43 -0
  381. claude_mpm/skills/bundled/main/internal-comms/examples/3p-updates.md +47 -0
  382. claude_mpm/skills/bundled/main/internal-comms/examples/company-newsletter.md +65 -0
  383. claude_mpm/skills/bundled/main/internal-comms/examples/faq-answers.md +30 -0
  384. claude_mpm/skills/bundled/main/internal-comms/examples/general-comms.md +16 -0
  385. claude_mpm/skills/bundled/main/mcp-builder/SKILL.md +160 -0
  386. claude_mpm/skills/bundled/main/mcp-builder/reference/design_principles.md +412 -0
  387. claude_mpm/skills/bundled/main/mcp-builder/reference/evaluation.md +602 -0
  388. claude_mpm/skills/bundled/main/mcp-builder/reference/mcp_best_practices.md +915 -0
  389. claude_mpm/skills/bundled/main/mcp-builder/reference/node_mcp_server.md +916 -0
  390. claude_mpm/skills/bundled/main/mcp-builder/reference/python_mcp_server.md +752 -0
  391. claude_mpm/skills/bundled/main/mcp-builder/reference/workflow.md +1237 -0
  392. claude_mpm/skills/bundled/main/skill-creator/SKILL.md +189 -0
  393. claude_mpm/skills/bundled/main/skill-creator/references/best-practices.md +500 -0
  394. claude_mpm/skills/bundled/main/skill-creator/references/creation-workflow.md +464 -0
  395. claude_mpm/skills/bundled/main/skill-creator/references/examples.md +619 -0
  396. claude_mpm/skills/bundled/main/skill-creator/references/progressive-disclosure.md +437 -0
  397. claude_mpm/skills/bundled/main/skill-creator/references/skill-structure.md +231 -0
  398. claude_mpm/skills/bundled/php/espocrm-development/SKILL.md +170 -0
  399. claude_mpm/skills/bundled/php/espocrm-development/references/architecture.md +602 -0
  400. claude_mpm/skills/bundled/php/espocrm-development/references/common-tasks.md +821 -0
  401. claude_mpm/skills/bundled/php/espocrm-development/references/development-workflow.md +742 -0
  402. claude_mpm/skills/bundled/php/espocrm-development/references/frontend-customization.md +726 -0
  403. claude_mpm/skills/bundled/php/espocrm-development/references/hooks-and-services.md +764 -0
  404. claude_mpm/skills/bundled/php/espocrm-development/references/testing-debugging.md +831 -0
  405. claude_mpm/skills/bundled/pm/mpm/SKILL.md +38 -0
  406. claude_mpm/skills/bundled/pm/mpm-agent-update-workflow/SKILL.md +75 -0
  407. claude_mpm/skills/bundled/pm/mpm-bug-reporting/SKILL.md +248 -0
  408. claude_mpm/skills/bundled/pm/mpm-circuit-breaker-enforcement/SKILL.md +476 -0
  409. claude_mpm/skills/bundled/pm/mpm-config/SKILL.md +29 -0
  410. claude_mpm/skills/bundled/pm/mpm-delegation-patterns/SKILL.md +167 -0
  411. claude_mpm/skills/bundled/pm/mpm-doctor/SKILL.md +53 -0
  412. claude_mpm/skills/bundled/pm/mpm-git-file-tracking/SKILL.md +113 -0
  413. claude_mpm/skills/bundled/pm/mpm-help/SKILL.md +35 -0
  414. claude_mpm/skills/bundled/pm/mpm-init/SKILL.md +125 -0
  415. claude_mpm/skills/bundled/pm/mpm-monitor/SKILL.md +32 -0
  416. claude_mpm/skills/bundled/pm/mpm-organize/SKILL.md +121 -0
  417. claude_mpm/skills/bundled/pm/mpm-postmortem/SKILL.md +22 -0
  418. claude_mpm/skills/bundled/pm/mpm-pr-workflow/SKILL.md +124 -0
  419. claude_mpm/skills/bundled/pm/mpm-session-management/SKILL.md +312 -0
  420. claude_mpm/skills/bundled/pm/mpm-session-pause/SKILL.md +170 -0
  421. claude_mpm/skills/bundled/pm/mpm-session-resume/SKILL.md +31 -0
  422. claude_mpm/skills/bundled/pm/mpm-status/SKILL.md +37 -0
  423. claude_mpm/skills/bundled/pm/mpm-teaching-mode/SKILL.md +657 -0
  424. claude_mpm/skills/bundled/pm/mpm-ticket-view/SKILL.md +110 -0
  425. claude_mpm/skills/bundled/pm/mpm-ticketing-integration/SKILL.md +154 -0
  426. claude_mpm/skills/bundled/pm/mpm-tool-usage-guide/SKILL.md +386 -0
  427. claude_mpm/skills/bundled/pm/mpm-verification-protocols/SKILL.md +198 -0
  428. claude_mpm/skills/bundled/pm/mpm-version/SKILL.md +21 -0
  429. claude_mpm/skills/bundled/react/flexlayout-react.md +742 -0
  430. claude_mpm/skills/bundled/rust/desktop-applications/SKILL.md +226 -0
  431. claude_mpm/skills/bundled/rust/desktop-applications/references/architecture-patterns.md +901 -0
  432. claude_mpm/skills/bundled/rust/desktop-applications/references/native-gui-frameworks.md +901 -0
  433. claude_mpm/skills/bundled/rust/desktop-applications/references/platform-integration.md +775 -0
  434. claude_mpm/skills/bundled/rust/desktop-applications/references/state-management.md +937 -0
  435. claude_mpm/skills/bundled/rust/desktop-applications/references/tauri-framework.md +770 -0
  436. claude_mpm/skills/bundled/rust/desktop-applications/references/testing-deployment.md +961 -0
  437. claude_mpm/skills/bundled/security-scanning.md +112 -0
  438. claude_mpm/skills/bundled/tauri/tauri-async-patterns.md +495 -0
  439. claude_mpm/skills/bundled/tauri/tauri-build-deploy.md +599 -0
  440. claude_mpm/skills/bundled/tauri/tauri-command-patterns.md +535 -0
  441. claude_mpm/skills/bundled/tauri/tauri-error-handling.md +613 -0
  442. claude_mpm/skills/bundled/tauri/tauri-event-system.md +648 -0
  443. claude_mpm/skills/bundled/tauri/tauri-file-system.md +673 -0
  444. claude_mpm/skills/bundled/tauri/tauri-frontend-integration.md +767 -0
  445. claude_mpm/skills/bundled/tauri/tauri-performance.md +669 -0
  446. claude_mpm/skills/bundled/tauri/tauri-state-management.md +573 -0
  447. claude_mpm/skills/bundled/tauri/tauri-testing.md +384 -0
  448. claude_mpm/skills/bundled/tauri/tauri-window-management.md +628 -0
  449. claude_mpm/skills/bundled/testing/condition-based-waiting/SKILL.md +119 -0
  450. claude_mpm/skills/bundled/testing/condition-based-waiting/references/patterns-and-implementation.md +253 -0
  451. claude_mpm/skills/bundled/testing/test-driven-development/SKILL.md +145 -0
  452. claude_mpm/skills/bundled/testing/test-driven-development/references/anti-patterns.md +543 -0
  453. claude_mpm/skills/bundled/testing/test-driven-development/references/examples.md +741 -0
  454. claude_mpm/skills/bundled/testing/test-driven-development/references/integration.md +470 -0
  455. claude_mpm/skills/bundled/testing/test-driven-development/references/philosophy.md +458 -0
  456. claude_mpm/skills/bundled/testing/test-driven-development/references/workflow.md +639 -0
  457. claude_mpm/skills/bundled/testing/test-quality-inspector/SKILL.md +458 -0
  458. claude_mpm/skills/bundled/testing/test-quality-inspector/examples/example-inspection-report.md +411 -0
  459. claude_mpm/skills/bundled/testing/test-quality-inspector/references/assertion-quality.md +317 -0
  460. claude_mpm/skills/bundled/testing/test-quality-inspector/references/inspection-checklist.md +270 -0
  461. claude_mpm/skills/bundled/testing/test-quality-inspector/references/red-flags.md +436 -0
  462. claude_mpm/skills/bundled/testing/testing-anti-patterns/SKILL.md +140 -0
  463. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/completeness-anti-patterns.md +572 -0
  464. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/core-anti-patterns.md +411 -0
  465. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/detection-guide.md +569 -0
  466. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/tdd-connection.md +695 -0
  467. claude_mpm/skills/bundled/testing/webapp-testing/SKILL.md +184 -0
  468. claude_mpm/skills/bundled/testing/webapp-testing/decision-tree.md +459 -0
  469. claude_mpm/skills/bundled/testing/webapp-testing/playwright-patterns.md +479 -0
  470. claude_mpm/skills/bundled/testing/webapp-testing/reconnaissance-pattern.md +687 -0
  471. claude_mpm/skills/bundled/testing/webapp-testing/server-management.md +758 -0
  472. claude_mpm/skills/bundled/testing/webapp-testing/troubleshooting.md +868 -0
  473. claude_mpm/skills/registry.py +295 -90
  474. claude_mpm/skills/skill_manager.py +98 -3
  475. claude_mpm/templates/.pre-commit-config.yaml +112 -0
  476. claude_mpm/utils/agent_dependency_loader.py +115 -4
  477. claude_mpm/utils/agent_filters.py +1 -1
  478. claude_mpm/utils/migration.py +4 -4
  479. claude_mpm/utils/robust_installer.py +86 -21
  480. claude_mpm-5.6.34.dist-info/METADATA +393 -0
  481. {claude_mpm-5.4.22.dist-info → claude_mpm-5.6.34.dist-info}/RECORD +486 -145
  482. claude_mpm-5.4.22.dist-info/METADATA +0 -996
  483. {claude_mpm-5.4.22.dist-info → claude_mpm-5.6.34.dist-info}/WHEEL +0 -0
  484. {claude_mpm-5.4.22.dist-info → claude_mpm-5.6.34.dist-info}/entry_points.txt +0 -0
  485. {claude_mpm-5.4.22.dist-info → claude_mpm-5.6.34.dist-info}/licenses/LICENSE +0 -0
  486. {claude_mpm-5.4.22.dist-info → claude_mpm-5.6.34.dist-info}/licenses/LICENSE-FAQ.md +0 -0
  487. {claude_mpm-5.4.22.dist-info → claude_mpm-5.6.34.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,831 @@
1
+ # Testing and Debugging Reference
2
+
3
+ ## Unit Testing
4
+
5
+ EspoCRM supports PHPUnit for testing.
6
+
7
+ ### Test Setup
8
+
9
+ Install PHPUnit in your extension:
10
+
11
+ ```bash
12
+ composer require --dev phpunit/phpunit
13
+ ```
14
+
15
+ Create `phpunit.xml`:
16
+
17
+ ```xml
18
+ <?xml version="1.0" encoding="UTF-8"?>
19
+ <phpunit
20
+ bootstrap="tests/bootstrap.php"
21
+ colors="true"
22
+ convertErrorsToExceptions="true"
23
+ convertNoticesToExceptions="true"
24
+ convertWarningsToExceptions="true"
25
+ stopOnFailure="false">
26
+ <testsuites>
27
+ <testsuite name="Unit Tests">
28
+ <directory>tests/unit</directory>
29
+ </testsuite>
30
+ <testsuite name="Integration Tests">
31
+ <directory>tests/integration</directory>
32
+ </testsuite>
33
+ </testsuites>
34
+ </phpunit>
35
+ ```
36
+
37
+ ### Unit Test Example
38
+
39
+ Create `tests/unit/MyServiceTest.php`:
40
+
41
+ ```php
42
+ <?php
43
+ namespace Espo\Modules\MyModule\Tests\Unit;
44
+
45
+ use PHPUnit\Framework\TestCase;
46
+ use Espo\Modules\MyModule\Services\MyEntity;
47
+ use Espo\ORM\EntityManager;
48
+ use Espo\Core\Acl;
49
+ use Espo\Entities\User;
50
+
51
+ class MyServiceTest extends TestCase
52
+ {
53
+ private MyEntity $service;
54
+ private EntityManager $entityManager;
55
+ private Acl $acl;
56
+ private User $user;
57
+
58
+ protected function setUp(): void
59
+ {
60
+ parent::setUp();
61
+
62
+ // Create mocks
63
+ $this->entityManager = $this->createMock(EntityManager::class);
64
+ $this->acl = $this->createMock(Acl::class);
65
+ $this->user = $this->createMock(User::class);
66
+
67
+ // Create service with mocked dependencies
68
+ $this->service = new MyEntity();
69
+
70
+ // Inject dependencies via reflection (if needed)
71
+ $reflection = new \ReflectionClass($this->service);
72
+ $property = $reflection->getProperty('entityManager');
73
+ $property->setAccessible(true);
74
+ $property->setValue($this->service, $this->entityManager);
75
+ }
76
+
77
+ public function testCalculateTotal(): void
78
+ {
79
+ // Arrange
80
+ $entity = $this->createMock(\Espo\ORM\Entity::class);
81
+
82
+ $entity->method('get')
83
+ ->willReturnMap([
84
+ ['quantity', null, 5],
85
+ ['unitPrice', null, 10.50]
86
+ ]);
87
+
88
+ // Act
89
+ $total = $this->service->calculateTotal($entity);
90
+
91
+ // Assert
92
+ $this->assertEquals(52.50, $total);
93
+ }
94
+
95
+ public function testValidateThrowsExceptionForInvalidData(): void
96
+ {
97
+ // Arrange
98
+ $entity = $this->createMock(\Espo\ORM\Entity::class);
99
+ $entity->method('get')
100
+ ->willReturn(-100);
101
+
102
+ // Assert
103
+ $this->expectException(\Espo\Core\Exceptions\BadRequest::class);
104
+ $this->expectExceptionMessage('Amount cannot be negative');
105
+
106
+ // Act
107
+ $this->service->validate($entity);
108
+ }
109
+
110
+ public function testCreateEntitySetsDefaults(): void
111
+ {
112
+ // Arrange
113
+ $entity = $this->createMock(\Espo\ORM\Entity::class);
114
+
115
+ $this->user->method('getId')
116
+ ->willReturn('user-123');
117
+
118
+ $entity->expects($this->once())
119
+ ->method('set')
120
+ ->with('assignedUserId', 'user-123');
121
+
122
+ // Act
123
+ $this->service->setDefaults($entity);
124
+ }
125
+ }
126
+ ```
127
+
128
+ ### Integration Test Example
129
+
130
+ Create `tests/integration/HookTest.php`:
131
+
132
+ ```php
133
+ <?php
134
+ namespace Espo\Modules\MyModule\Tests\Integration;
135
+
136
+ use PHPUnit\Framework\TestCase;
137
+ use Espo\Core\Container;
138
+ use Espo\ORM\EntityManager;
139
+
140
+ class HookTest extends TestCase
141
+ {
142
+ private Container $container;
143
+ private EntityManager $entityManager;
144
+
145
+ protected function setUp(): void
146
+ {
147
+ parent::setUp();
148
+
149
+ // Load EspoCRM container (requires EspoCRM installation)
150
+ require_once 'bootstrap.php';
151
+
152
+ $this->container = $GLOBALS['container'];
153
+ $this->entityManager = $this->container->get('entityManager');
154
+ }
155
+
156
+ public function testAccountHookUpdatesRelatedContacts(): void
157
+ {
158
+ // Create test account
159
+ $account = $this->entityManager->getNewEntity('Account');
160
+ $account->set('name', 'Test Account');
161
+ $this->entityManager->saveEntity($account);
162
+
163
+ // Create related contact
164
+ $contact = $this->entityManager->getNewEntity('Contact');
165
+ $contact->set([
166
+ 'firstName' => 'John',
167
+ 'lastName' => 'Doe',
168
+ 'accountId' => $account->getId()
169
+ ]);
170
+ $this->entityManager->saveEntity($contact);
171
+
172
+ // Update account status
173
+ $account->set('status', 'Inactive');
174
+ $this->entityManager->saveEntity($account);
175
+
176
+ // Refresh contact
177
+ $contact = $this->entityManager->getEntityById('Contact', $contact->getId());
178
+
179
+ // Assert hook updated contact
180
+ $this->assertEquals('Inactive', $contact->get('accountStatus'));
181
+
182
+ // Cleanup
183
+ $this->entityManager->removeEntity($contact);
184
+ $this->entityManager->removeEntity($account);
185
+ }
186
+ }
187
+ ```
188
+
189
+ ### Running Tests
190
+
191
+ ```bash
192
+ # Run all tests
193
+ vendor/bin/phpunit
194
+
195
+ # Run specific test file
196
+ vendor/bin/phpunit tests/unit/MyServiceTest.php
197
+
198
+ # Run specific test method
199
+ vendor/bin/phpunit --filter testCalculateTotal
200
+
201
+ # Run with coverage
202
+ vendor/bin/phpunit --coverage-html coverage
203
+ ```
204
+
205
+ ## Debugging Techniques
206
+
207
+ ### Enable Debug Mode
208
+
209
+ Edit `data/config.php`:
210
+
211
+ ```php
212
+ <?php
213
+ return [
214
+ // ... other config ...
215
+ 'logger' => [
216
+ 'level' => 'DEBUG', // Change from 'WARNING' to 'DEBUG'
217
+ 'maxFileNumber' => 30,
218
+ ],
219
+ 'isDeveloperMode' => true, // Enable developer mode
220
+ ];
221
+ ```
222
+
223
+ ### Logging
224
+
225
+ ```php
226
+ <?php
227
+ namespace Espo\Modules\MyModule\Services;
228
+
229
+ use Espo\Core\Utils\Log;
230
+
231
+ class MyService
232
+ {
233
+ public function __construct(private Log $log) {}
234
+
235
+ public function processData(array $data): void
236
+ {
237
+ // Debug logging
238
+ $this->log->debug('Processing data', ['data' => $data]);
239
+
240
+ try {
241
+ // ... processing ...
242
+
243
+ $this->log->info('Data processed successfully', ['count' => count($data)]);
244
+ } catch (\Throwable $e) {
245
+ $this->log->error('Failed to process data: ' . $e->getMessage(), [
246
+ 'exception' => $e,
247
+ 'trace' => $e->getTraceAsString()
248
+ ]);
249
+
250
+ throw $e;
251
+ }
252
+ }
253
+
254
+ public function debugQuery(): void
255
+ {
256
+ $query = $this->entityManager
257
+ ->getQueryBuilder()
258
+ ->select()
259
+ ->from('Account')
260
+ ->where(['type' => 'Customer'])
261
+ ->build();
262
+
263
+ // Log the SQL query
264
+ $sql = $this->entityManager->getQueryComposer()->compose($query);
265
+ $this->log->debug('SQL Query', ['sql' => $sql]);
266
+ }
267
+ }
268
+ ```
269
+
270
+ ### Check Logs
271
+
272
+ ```bash
273
+ # Today's log
274
+ tail -f data/logs/espo-$(date +%Y-%m-%d).log
275
+
276
+ # Search logs
277
+ grep "ERROR" data/logs/espo-*.log
278
+
279
+ # Filter by component
280
+ grep "MyService" data/logs/espo-*.log
281
+ ```
282
+
283
+ ### XDebug Setup
284
+
285
+ Install XDebug PHP extension:
286
+
287
+ ```bash
288
+ # Ubuntu/Debian
289
+ sudo apt-get install php-xdebug
290
+
291
+ # macOS (Homebrew)
292
+ brew install php-xdebug
293
+ ```
294
+
295
+ Configure XDebug in `php.ini`:
296
+
297
+ ```ini
298
+ [xdebug]
299
+ zend_extension=xdebug.so
300
+ xdebug.mode=debug
301
+ xdebug.start_with_request=yes
302
+ xdebug.client_host=127.0.0.1
303
+ xdebug.client_port=9003
304
+ xdebug.idekey=PHPSTORM
305
+ ```
306
+
307
+ ### Step Debugging with PhpStorm
308
+
309
+ 1. Set breakpoints in code
310
+ 2. Click "Start Listening for PHP Debug Connections"
311
+ 3. Trigger the code path (API call, scheduled job, etc.)
312
+ 4. PhpStorm will pause at breakpoints
313
+
314
+ ### Debug API Requests
315
+
316
+ ```bash
317
+ # Using curl with verbose output
318
+ curl -v -X GET \
319
+ 'http://espocrm.local/api/v1/Account/123' \
320
+ -H 'X-Api-Key: your-api-key'
321
+
322
+ # Debug POST request
323
+ curl -v -X POST \
324
+ 'http://espocrm.local/api/v1/Account' \
325
+ -H 'Content-Type: application/json' \
326
+ -H 'X-Api-Key: your-api-key' \
327
+ -d '{
328
+ "name": "Test Account",
329
+ "type": "Customer"
330
+ }'
331
+
332
+ # Save response to file for analysis
333
+ curl -X GET \
334
+ 'http://espocrm.local/api/v1/Account' \
335
+ -H 'X-Api-Key: your-api-key' \
336
+ -o response.json
337
+ ```
338
+
339
+ ### Browser DevTools for Frontend Debugging
340
+
341
+ ```javascript
342
+ // Add console logging in custom views
343
+ define('custom:views/my-view', ['view'], function (Dep) {
344
+ return Dep.extend({
345
+ setup: function () {
346
+ Dep.prototype.setup.call(this);
347
+
348
+ console.log('View setup', {
349
+ model: this.model.attributes,
350
+ options: this.options
351
+ });
352
+ },
353
+
354
+ afterRender: function () {
355
+ Dep.prototype.afterRender.call(this);
356
+
357
+ console.log('View rendered', {
358
+ el: this.$el,
359
+ modelId: this.model.id
360
+ });
361
+ }
362
+ });
363
+ });
364
+
365
+ // Debug AJAX requests
366
+ this.ajaxPostRequest('Account/action/myAction', data)
367
+ .then(response => {
368
+ console.log('Response received', response);
369
+ })
370
+ .catch(xhr => {
371
+ console.error('Request failed', {
372
+ status: xhr.status,
373
+ response: xhr.responseJSON,
374
+ error: xhr.responseText
375
+ });
376
+ });
377
+ ```
378
+
379
+ ### Database Debugging
380
+
381
+ ```bash
382
+ # Connect to MySQL
383
+ mysql -u espocrm_user -p espocrm_db
384
+
385
+ # Enable query logging
386
+ SET GLOBAL general_log = 'ON';
387
+ SET GLOBAL log_output = 'TABLE';
388
+
389
+ # View query log
390
+ SELECT * FROM mysql.general_log
391
+ WHERE command_type = 'Query'
392
+ ORDER BY event_time DESC
393
+ LIMIT 100;
394
+
395
+ # Disable query logging
396
+ SET GLOBAL general_log = 'OFF';
397
+ ```
398
+
399
+ ### SQL Query Analysis
400
+
401
+ ```php
402
+ // Log query execution time
403
+ $startTime = microtime(true);
404
+
405
+ $result = $this->entityManager
406
+ ->getRDBRepository('Account')
407
+ ->where(['type' => 'Customer'])
408
+ ->find();
409
+
410
+ $executionTime = microtime(true) - $startTime;
411
+
412
+ $this->log->debug('Query execution time', [
413
+ 'time' => $executionTime,
414
+ 'count' => count($result)
415
+ ]);
416
+
417
+ // Explain query
418
+ $query = $this->entityManager
419
+ ->getQueryBuilder()
420
+ ->select()
421
+ ->from('Account')
422
+ ->where(['type' => 'Customer'])
423
+ ->build();
424
+
425
+ $sql = $this->entityManager->getQueryComposer()->compose($query);
426
+
427
+ $pdo = $this->entityManager->getPDO();
428
+ $sth = $pdo->prepare('EXPLAIN ' . $sql);
429
+ $sth->execute();
430
+
431
+ $explain = $sth->fetchAll(\PDO::FETCH_ASSOC);
432
+ $this->log->debug('Query explain', ['explain' => $explain]);
433
+ ```
434
+
435
+ ## Performance Optimization
436
+
437
+ ### Use STH Collections for Large Datasets
438
+
439
+ ```php
440
+ // ❌ WRONG - Loads all records into memory
441
+ $contacts = $this->entityManager
442
+ ->getRDBRepository('Contact')
443
+ ->find();
444
+
445
+ foreach ($contacts as $contact) {
446
+ // Process
447
+ }
448
+
449
+ // ✅ CORRECT - Streams records one at a time
450
+ $sthCollection = $this->entityManager
451
+ ->getRDBRepository('Contact')
452
+ ->sth()
453
+ ->find();
454
+
455
+ foreach ($sthCollection as $contact) {
456
+ // Process without loading all into memory
457
+ }
458
+ ```
459
+
460
+ ### Batch Operations
461
+
462
+ ```php
463
+ // ❌ WRONG - One query per update
464
+ $accounts = $this->entityManager
465
+ ->getRDBRepository('Account')
466
+ ->where(['type' => 'Customer'])
467
+ ->find();
468
+
469
+ foreach ($accounts as $account) {
470
+ $account->set('updated', true);
471
+ $this->entityManager->saveEntity($account); // Individual query
472
+ }
473
+
474
+ // ✅ CORRECT - Batch update with single query
475
+ $this->entityManager
476
+ ->getQueryBuilder()
477
+ ->update()
478
+ ->in('Account')
479
+ ->set(['updated' => true])
480
+ ->where(['type' => 'Customer'])
481
+ ->build()
482
+ ->execute();
483
+ ```
484
+
485
+ ### Query Optimization
486
+
487
+ ```php
488
+ // ❌ WRONG - N+1 queries
489
+ $opportunities = $this->entityManager
490
+ ->getRDBRepository('Opportunity')
491
+ ->find();
492
+
493
+ foreach ($opportunities as $opportunity) {
494
+ // Each iteration makes a query for account
495
+ $account = $this->entityManager
496
+ ->getEntityById('Account', $opportunity->get('accountId'));
497
+ }
498
+
499
+ // ✅ CORRECT - Single query with JOIN
500
+ $opportunities = $this->entityManager
501
+ ->getRDBRepository('Opportunity')
502
+ ->distinct()
503
+ ->join('account')
504
+ ->select(['*', 'account.name AS accountName'])
505
+ ->find();
506
+
507
+ foreach ($opportunities as $opportunity) {
508
+ // Account name already loaded
509
+ $accountName = $opportunity->get('accountName');
510
+ }
511
+ ```
512
+
513
+ ### Caching
514
+
515
+ ```php
516
+ use Espo\Core\Utils\DataCache;
517
+
518
+ class MyService
519
+ {
520
+ public function __construct(private DataCache $dataCache) {}
521
+
522
+ public function getExpensiveData(): array
523
+ {
524
+ $cacheKey = 'myExpensiveData';
525
+
526
+ // Try to get from cache
527
+ if ($this->dataCache->has($cacheKey)) {
528
+ return $this->dataCache->get($cacheKey);
529
+ }
530
+
531
+ // Calculate if not cached
532
+ $data = $this->performExpensiveCalculation();
533
+
534
+ // Store in cache for 1 hour
535
+ $this->dataCache->store($cacheKey, $data, 3600);
536
+
537
+ return $data;
538
+ }
539
+
540
+ private function performExpensiveCalculation(): array
541
+ {
542
+ // Expensive operation
543
+ return [];
544
+ }
545
+ }
546
+ ```
547
+
548
+ ### Profiling with Xdebug
549
+
550
+ ```php
551
+ // Enable profiling in php.ini
552
+ xdebug.mode=profile
553
+ xdebug.output_dir=/tmp/xdebug
554
+ xdebug.start_with_request=trigger
555
+
556
+ // Trigger with request parameter
557
+ // http://espocrm.local/api/v1/Account?XDEBUG_PROFILE=1
558
+
559
+ // Analyze with tools like:
560
+ // - KCachegrind (Linux)
561
+ // - QCachegrind (macOS/Windows)
562
+ // - PhpStorm built-in profiler
563
+ ```
564
+
565
+ ## Common Pitfalls and Solutions
566
+
567
+ ### Pitfall 1: Infinite Hook Loops
568
+
569
+ **Problem:**
570
+ ```php
571
+ // Hook triggers itself infinitely
572
+ class MyHook implements AfterSave {
573
+ public function afterSave(Entity $entity, array $options): void {
574
+ $entity->set('updated', true);
575
+ $this->entityManager->saveEntity($entity); // Triggers hook again!
576
+ }
577
+ }
578
+ ```
579
+
580
+ **Solution:**
581
+ ```php
582
+ class MyHook implements AfterSave {
583
+ public function afterSave(Entity $entity, array $options): void {
584
+ // Check if already processing
585
+ if (!empty($options['skipHooks'])) {
586
+ return;
587
+ }
588
+
589
+ $entity->set('updated', true);
590
+ $this->entityManager->saveEntity($entity, ['skipHooks' => true]);
591
+ }
592
+ }
593
+ ```
594
+
595
+ ### Pitfall 2: Missing Cache Rebuild
596
+
597
+ **Problem:**
598
+ - Metadata changes not reflected
599
+ - New classes not found
600
+
601
+ **Solution:**
602
+ ```bash
603
+ # Always rebuild after metadata or class changes
604
+ bin/command rebuild
605
+ ```
606
+
607
+ ### Pitfall 3: Type Mismatches
608
+
609
+ **Problem:**
610
+ ```php
611
+ // Method expects string, gets null
612
+ public function process(string $id): void {
613
+ // $id is null, causes error
614
+ }
615
+ ```
616
+
617
+ **Solution:**
618
+ ```php
619
+ // Use nullable types
620
+ public function process(?string $id): void {
621
+ if (!$id) {
622
+ throw new BadRequest('ID is required');
623
+ }
624
+ // Process
625
+ }
626
+ ```
627
+
628
+ ### Pitfall 4: Container Dependency
629
+
630
+ **Problem:**
631
+ ```php
632
+ // Tight coupling to Container
633
+ class MyService {
634
+ public function __construct(private Container $container) {}
635
+ }
636
+ ```
637
+
638
+ **Solution:**
639
+ ```php
640
+ // Inject specific dependencies
641
+ class MyService {
642
+ public function __construct(
643
+ private EntityManager $entityManager,
644
+ private Metadata $metadata
645
+ ) {}
646
+ }
647
+ ```
648
+
649
+ ### Pitfall 5: Not Using Transactions
650
+
651
+ **Problem:**
652
+ ```php
653
+ // Partial updates on error
654
+ $entity1->set('status', 'Processing');
655
+ $this->entityManager->saveEntity($entity1);
656
+
657
+ // Error occurs here
658
+ throw new \Exception('Error');
659
+
660
+ // entity2 never updated, data inconsistent
661
+ $entity2->set('status', 'Complete');
662
+ $this->entityManager->saveEntity($entity2);
663
+ ```
664
+
665
+ **Solution:**
666
+ ```php
667
+ use Espo\ORM\TransactionManager;
668
+
669
+ // Atomic operation
670
+ $this->transactionManager->run(function () {
671
+ $entity1->set('status', 'Processing');
672
+ $this->entityManager->saveEntity($entity1);
673
+
674
+ // If error occurs, both changes rolled back
675
+ if ($error) {
676
+ throw new \Exception('Error');
677
+ }
678
+
679
+ $entity2->set('status', 'Complete');
680
+ $this->entityManager->saveEntity($entity2);
681
+ });
682
+ ```
683
+
684
+ ### Pitfall 6: Direct PDO Usage
685
+
686
+ **Problem:**
687
+ ```php
688
+ // Bypassing ORM
689
+ $pdo = $this->entityManager->getPDO();
690
+ $pdo->query("INSERT INTO account (name) VALUES ('Test')");
691
+ ```
692
+
693
+ **Solution:**
694
+ ```php
695
+ // Use ORM
696
+ $account = $this->entityManager->getNewEntity('Account');
697
+ $account->set('name', 'Test');
698
+ $this->entityManager->saveEntity($account);
699
+ ```
700
+
701
+ ### Pitfall 7: Memory Exhaustion
702
+
703
+ **Problem:**
704
+ ```php
705
+ // Loading 100k records into memory
706
+ $contacts = $this->entityManager
707
+ ->getRDBRepository('Contact')
708
+ ->find(); // Memory exhausted!
709
+ ```
710
+
711
+ **Solution:**
712
+ ```php
713
+ // Use STH collection
714
+ $sthCollection = $this->entityManager
715
+ ->getRDBRepository('Contact')
716
+ ->sth()
717
+ ->find();
718
+
719
+ foreach ($sthCollection as $contact) {
720
+ // Process one at a time
721
+ }
722
+ ```
723
+
724
+ ## Error Handling Best Practices
725
+
726
+ ### Use Specific Exceptions
727
+
728
+ ```php
729
+ use Espo\Core\Exceptions\{
730
+ BadRequest,
731
+ Forbidden,
732
+ NotFound,
733
+ Conflict,
734
+ Error
735
+ };
736
+
737
+ class MyService {
738
+ public function process(string $id): Entity {
739
+ // Validate input
740
+ if (!$id) {
741
+ throw new BadRequest('ID is required');
742
+ }
743
+
744
+ // Check existence
745
+ $entity = $this->entityManager->getEntityById('Account', $id);
746
+ if (!$entity) {
747
+ throw new NotFound('Account not found');
748
+ }
749
+
750
+ // Check permissions
751
+ if (!$this->acl->check($entity, 'edit')) {
752
+ throw new Forbidden('No edit access');
753
+ }
754
+
755
+ // Check business rules
756
+ if ($entity->get('locked')) {
757
+ throw new Conflict('Account is locked');
758
+ }
759
+
760
+ // Process...
761
+ return $entity;
762
+ }
763
+ }
764
+ ```
765
+
766
+ ### Try-Catch Patterns
767
+
768
+ ```php
769
+ // Catch and log
770
+ try {
771
+ $this->performRiskyOperation();
772
+ } catch (\Throwable $e) {
773
+ $this->log->error('Operation failed: ' . $e->getMessage(), [
774
+ 'exception' => $e
775
+ ]);
776
+ throw $e; // Re-throw
777
+ }
778
+
779
+ // Catch and convert
780
+ try {
781
+ $result = $this->externalApi->call();
782
+ } catch (\GuzzleHttp\Exception\RequestException $e) {
783
+ throw new Error('External API failed: ' . $e->getMessage());
784
+ }
785
+
786
+ // Catch specific, let others propagate
787
+ try {
788
+ $this->operation();
789
+ } catch (NotFound $e) {
790
+ // Handle not found specifically
791
+ return $this->createDefault();
792
+ }
793
+ // Other exceptions propagate up
794
+ ```
795
+
796
+ ## Debugging Checklist
797
+
798
+ When something doesn't work:
799
+
800
+ 1. **Check Logs**
801
+ - `data/logs/espo-YYYY-MM-DD.log`
802
+ - PHP error logs
803
+
804
+ 2. **Verify Cache Rebuild**
805
+ - `bin/command rebuild`
806
+
807
+ 3. **Check Permissions**
808
+ - File permissions (files should be writable)
809
+ - ACL permissions (user has access)
810
+
811
+ 4. **Validate Metadata**
812
+ - JSON syntax correct
813
+ - Required fields present
814
+
815
+ 5. **Check Database**
816
+ - Table exists
817
+ - Columns exist
818
+ - Indexes present
819
+
820
+ 6. **Test API Directly**
821
+ - Use curl or Postman
822
+ - Check request/response
823
+
824
+ 7. **Enable Debug Mode**
825
+ - `'logger' => ['level' => 'DEBUG']`
826
+ - `'isDeveloperMode' => true`
827
+
828
+ 8. **Use XDebug**
829
+ - Set breakpoints
830
+ - Step through code
831
+ - Inspect variables