claude-mpm 5.4.55__py3-none-any.whl → 5.6.1__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 (401) hide show
  1. claude_mpm/VERSION +1 -1
  2. claude_mpm/agents/CLAUDE_MPM_OUTPUT_STYLE.md +66 -241
  3. claude_mpm/agents/CLAUDE_MPM_RESEARCH_OUTPUT_STYLE.md +413 -0
  4. claude_mpm/agents/CLAUDE_MPM_TEACHER_OUTPUT_STYLE.md +109 -1925
  5. claude_mpm/agents/PM_INSTRUCTIONS.md +111 -686
  6. claude_mpm/agents/WORKFLOW.md +2 -0
  7. claude_mpm/agents/templates/circuit-breakers.md +26 -17
  8. claude_mpm/cli/__init__.py +5 -1
  9. claude_mpm/cli/commands/agents.py +2 -4
  10. claude_mpm/cli/commands/agents_reconcile.py +197 -0
  11. claude_mpm/cli/commands/autotodos.py +566 -0
  12. claude_mpm/cli/commands/commander.py +46 -0
  13. claude_mpm/cli/commands/configure.py +620 -21
  14. claude_mpm/cli/commands/hook_errors.py +60 -60
  15. claude_mpm/cli/commands/monitor.py +2 -2
  16. claude_mpm/cli/commands/mpm_init/core.py +2 -2
  17. claude_mpm/cli/commands/run.py +35 -3
  18. claude_mpm/cli/commands/skills.py +166 -14
  19. claude_mpm/cli/executor.py +120 -16
  20. claude_mpm/cli/interactive/__init__.py +10 -0
  21. claude_mpm/cli/interactive/agent_wizard.py +30 -50
  22. claude_mpm/cli/interactive/questionary_styles.py +65 -0
  23. claude_mpm/cli/interactive/skill_selector.py +481 -0
  24. claude_mpm/cli/parsers/base_parser.py +76 -1
  25. claude_mpm/cli/parsers/commander_parser.py +83 -0
  26. claude_mpm/cli/parsers/run_parser.py +10 -0
  27. claude_mpm/cli/startup.py +276 -403
  28. claude_mpm/cli/startup_display.py +72 -5
  29. claude_mpm/cli/startup_logging.py +2 -2
  30. claude_mpm/cli/utils.py +7 -3
  31. claude_mpm/commander/__init__.py +72 -0
  32. claude_mpm/commander/adapters/__init__.py +31 -0
  33. claude_mpm/commander/adapters/base.py +191 -0
  34. claude_mpm/commander/adapters/claude_code.py +361 -0
  35. claude_mpm/commander/adapters/communication.py +366 -0
  36. claude_mpm/commander/api/__init__.py +16 -0
  37. claude_mpm/commander/api/app.py +105 -0
  38. claude_mpm/commander/api/errors.py +112 -0
  39. claude_mpm/commander/api/routes/__init__.py +8 -0
  40. claude_mpm/commander/api/routes/events.py +184 -0
  41. claude_mpm/commander/api/routes/inbox.py +171 -0
  42. claude_mpm/commander/api/routes/messages.py +148 -0
  43. claude_mpm/commander/api/routes/projects.py +271 -0
  44. claude_mpm/commander/api/routes/sessions.py +215 -0
  45. claude_mpm/commander/api/routes/work.py +260 -0
  46. claude_mpm/commander/api/schemas.py +182 -0
  47. claude_mpm/commander/chat/__init__.py +7 -0
  48. claude_mpm/commander/chat/cli.py +107 -0
  49. claude_mpm/commander/chat/commands.py +96 -0
  50. claude_mpm/commander/chat/repl.py +310 -0
  51. claude_mpm/commander/config.py +49 -0
  52. claude_mpm/commander/config_loader.py +115 -0
  53. claude_mpm/commander/daemon.py +398 -0
  54. claude_mpm/commander/events/__init__.py +26 -0
  55. claude_mpm/commander/events/manager.py +332 -0
  56. claude_mpm/commander/frameworks/__init__.py +12 -0
  57. claude_mpm/commander/frameworks/base.py +143 -0
  58. claude_mpm/commander/frameworks/claude_code.py +58 -0
  59. claude_mpm/commander/frameworks/mpm.py +62 -0
  60. claude_mpm/commander/inbox/__init__.py +16 -0
  61. claude_mpm/commander/inbox/dedup.py +128 -0
  62. claude_mpm/commander/inbox/inbox.py +224 -0
  63. claude_mpm/commander/inbox/models.py +70 -0
  64. claude_mpm/commander/instance_manager.py +337 -0
  65. claude_mpm/commander/llm/__init__.py +6 -0
  66. claude_mpm/commander/llm/openrouter_client.py +167 -0
  67. claude_mpm/commander/llm/summarizer.py +70 -0
  68. claude_mpm/commander/models/__init__.py +18 -0
  69. claude_mpm/commander/models/events.py +121 -0
  70. claude_mpm/commander/models/project.py +162 -0
  71. claude_mpm/commander/models/work.py +214 -0
  72. claude_mpm/commander/parsing/__init__.py +20 -0
  73. claude_mpm/commander/parsing/extractor.py +132 -0
  74. claude_mpm/commander/parsing/output_parser.py +270 -0
  75. claude_mpm/commander/parsing/patterns.py +100 -0
  76. claude_mpm/commander/persistence/__init__.py +11 -0
  77. claude_mpm/commander/persistence/event_store.py +274 -0
  78. claude_mpm/commander/persistence/state_store.py +309 -0
  79. claude_mpm/commander/persistence/work_store.py +164 -0
  80. claude_mpm/commander/polling/__init__.py +13 -0
  81. claude_mpm/commander/polling/event_detector.py +104 -0
  82. claude_mpm/commander/polling/output_buffer.py +49 -0
  83. claude_mpm/commander/polling/output_poller.py +153 -0
  84. claude_mpm/commander/project_session.py +268 -0
  85. claude_mpm/commander/proxy/__init__.py +12 -0
  86. claude_mpm/commander/proxy/formatter.py +89 -0
  87. claude_mpm/commander/proxy/output_handler.py +191 -0
  88. claude_mpm/commander/proxy/relay.py +155 -0
  89. claude_mpm/commander/registry.py +404 -0
  90. claude_mpm/commander/runtime/__init__.py +10 -0
  91. claude_mpm/commander/runtime/executor.py +191 -0
  92. claude_mpm/commander/runtime/monitor.py +316 -0
  93. claude_mpm/commander/session/__init__.py +6 -0
  94. claude_mpm/commander/session/context.py +81 -0
  95. claude_mpm/commander/session/manager.py +59 -0
  96. claude_mpm/commander/tmux_orchestrator.py +361 -0
  97. claude_mpm/commander/web/__init__.py +1 -0
  98. claude_mpm/commander/work/__init__.py +30 -0
  99. claude_mpm/commander/work/executor.py +189 -0
  100. claude_mpm/commander/work/queue.py +405 -0
  101. claude_mpm/commander/workflow/__init__.py +27 -0
  102. claude_mpm/commander/workflow/event_handler.py +219 -0
  103. claude_mpm/commander/workflow/notifier.py +146 -0
  104. claude_mpm/commands/mpm-config.md +8 -0
  105. claude_mpm/commands/mpm-doctor.md +8 -0
  106. claude_mpm/commands/mpm-help.md +8 -0
  107. claude_mpm/commands/mpm-init.md +8 -0
  108. claude_mpm/commands/mpm-monitor.md +8 -0
  109. claude_mpm/commands/mpm-organize.md +8 -0
  110. claude_mpm/commands/mpm-postmortem.md +8 -0
  111. claude_mpm/commands/mpm-session-resume.md +9 -1
  112. claude_mpm/commands/mpm-status.md +8 -0
  113. claude_mpm/commands/mpm-ticket-view.md +8 -0
  114. claude_mpm/commands/mpm-version.md +8 -0
  115. claude_mpm/commands/mpm.md +8 -0
  116. claude_mpm/config/agent_presets.py +8 -7
  117. claude_mpm/constants.py +1 -0
  118. claude_mpm/core/claude_runner.py +2 -2
  119. claude_mpm/core/config.py +5 -0
  120. claude_mpm/core/hook_manager.py +51 -3
  121. claude_mpm/core/interactive_session.py +7 -7
  122. claude_mpm/core/logger.py +10 -7
  123. claude_mpm/core/logging_utils.py +4 -2
  124. claude_mpm/core/output_style_manager.py +31 -13
  125. claude_mpm/core/unified_config.py +54 -8
  126. claude_mpm/core/unified_paths.py +30 -13
  127. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/0.C33zOoyM.css +1 -0
  128. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/2.CW1J-YuA.css +1 -0
  129. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{Cs_tUR18.js → 1WZnGYqX.js} +1 -1
  130. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{CDuw-vjf.js → 67pF3qNn.js} +1 -1
  131. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{bTOqqlTd.js → 6RxdMKe4.js} +1 -1
  132. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{DwBR2MJi.js → 8cZrfX0h.js} +1 -1
  133. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{ZGh7QtNv.js → 9a6T2nm-.js} +1 -1
  134. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{D9lljYKQ.js → B443AUzu.js} +1 -1
  135. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{RJiighC3.js → B8AwtY2H.js} +1 -1
  136. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{uuIeMWc-.js → BF15LAsF.js} +1 -1
  137. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{D3k0OPJN.js → BRcwIQNr.js} +1 -1
  138. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{CyWMqx4W.js → BV6nKitt.js} +1 -1
  139. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{CiIAseT4.js → BViJ8lZt.js} +5 -5
  140. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{CBBdVcY8.js → BcQ-Q0FE.js} +1 -1
  141. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{BovzEFCE.js → Bpyvgze_.js} +1 -1
  142. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BzTRqg-z.js +1 -0
  143. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C0Fr8dve.js +1 -0
  144. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{eNVUfhuA.js → C3rbW_a-.js} +1 -1
  145. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{GYwsonyD.js → C8WYN38h.js} +1 -1
  146. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{BIF9m_hv.js → C9I8FlXH.js} +1 -1
  147. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{B0uc0UOD.js → CIQcWgO2.js} +3 -3
  148. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{Be7GpZd6.js → CIctN7YN.js} +1 -1
  149. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{Bh0LDWpI.js → CKrS_JZW.js} +2 -2
  150. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{DUrLdbGD.js → CR6P9C4A.js} +1 -1
  151. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{B7xVLGWV.js → CRRR9MD_.js} +1 -1
  152. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CRcR2DqT.js +334 -0
  153. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{Dhb8PKl3.js → CSXtMOf0.js} +1 -1
  154. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{BPYeabCQ.js → CT-sbxSk.js} +1 -1
  155. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{sQeU3Y1z.js → CWm6DJsp.js} +1 -1
  156. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{CnA0NrzZ.js → CpqQ1Kzn.js} +1 -1
  157. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{C4B-KCzX.js → D2nGpDRe.js} +1 -1
  158. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{DGkLK5U1.js → D9iCMida.js} +1 -1
  159. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{BofRWZRR.js → D9ykgMoY.js} +1 -1
  160. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{DmxopI1J.js → DL2Ldur1.js} +1 -1
  161. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{C30mlcqg.js → DPfltzjH.js} +1 -1
  162. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{Vzk33B_K.js → DR8nis88.js} +2 -2
  163. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{DI7hHRFL.js → DUliQN2b.js} +1 -1
  164. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{C4JcI4KD.js → DXlhR01x.js} +1 -1
  165. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{bT1r9zLR.js → D_lyTybS.js} +1 -1
  166. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{DZX00Y4g.js → DngoTTgh.js} +1 -1
  167. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{CzZX-COe.js → DqkmHtDC.js} +1 -1
  168. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{B7RN905-.js → DsDh8EYs.js} +1 -1
  169. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{DLVjFsZ3.js → DypDmXgd.js} +1 -1
  170. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{iEWssX7S.js → IPYC-LnN.js} +1 -1
  171. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/JTLiF7dt.js +24 -0
  172. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{DaimHw_p.js → JpevfAFt.js} +1 -1
  173. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{DY1XQ8fi.js → R8CEIRAd.js} +1 -1
  174. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{Dle-35c7.js → Zxy7qc-l.js} +2 -2
  175. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/q9Hm6zAU.js +1 -0
  176. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{C_Usid8X.js → qtd3IeO4.js} +2 -2
  177. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{CzeYkLYB.js → ulBFON_C.js} +2 -2
  178. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{Cfqx1Qun.js → wQVh1CoA.js} +1 -1
  179. claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/{app.D6-I5TpK.js → app.Dr7t0z2J.js} +2 -2
  180. claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/start.BGhZHUS3.js +1 -0
  181. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/{0.m1gL8KXf.js → 0.RgBboRvH.js} +1 -1
  182. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/{1.CgNOuw-d.js → 1.DG-KkbDf.js} +1 -1
  183. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/2.D_jnf-x6.js +1 -0
  184. claude_mpm/dashboard/static/svelte-build/_app/version.json +1 -1
  185. claude_mpm/dashboard/static/svelte-build/index.html +9 -9
  186. claude_mpm/experimental/cli_enhancements.py +2 -1
  187. claude_mpm/hooks/claude_hooks/INTEGRATION_EXAMPLE.md +243 -0
  188. claude_mpm/hooks/claude_hooks/README_AUTO_PAUSE.md +403 -0
  189. claude_mpm/hooks/claude_hooks/auto_pause_handler.py +486 -0
  190. claude_mpm/hooks/claude_hooks/event_handlers.py +250 -11
  191. claude_mpm/hooks/claude_hooks/hook_handler.py +106 -89
  192. claude_mpm/hooks/claude_hooks/hook_wrapper.sh +6 -11
  193. claude_mpm/hooks/claude_hooks/installer.py +69 -5
  194. claude_mpm/hooks/claude_hooks/response_tracking.py +3 -1
  195. claude_mpm/hooks/claude_hooks/services/connection_manager.py +20 -0
  196. claude_mpm/hooks/claude_hooks/services/connection_manager_http.py +14 -77
  197. claude_mpm/hooks/claude_hooks/services/subagent_processor.py +30 -6
  198. claude_mpm/hooks/session_resume_hook.py +85 -1
  199. claude_mpm/init.py +1 -1
  200. claude_mpm/scripts/claude-hook-handler.sh +36 -10
  201. claude_mpm/scripts/start_activity_logging.py +0 -0
  202. claude_mpm/services/agents/agent_recommendation_service.py +8 -8
  203. claude_mpm/services/agents/cache_git_manager.py +1 -1
  204. claude_mpm/services/agents/deployment/agent_template_builder.py +8 -0
  205. claude_mpm/services/agents/deployment/deployment_reconciler.py +577 -0
  206. claude_mpm/services/agents/deployment/remote_agent_discovery_service.py +3 -0
  207. claude_mpm/services/agents/deployment/startup_reconciliation.py +138 -0
  208. claude_mpm/services/agents/loading/framework_agent_loader.py +75 -2
  209. claude_mpm/services/agents/sources/git_source_sync_service.py +7 -4
  210. claude_mpm/services/agents/startup_sync.py +5 -2
  211. claude_mpm/services/cli/__init__.py +3 -0
  212. claude_mpm/services/cli/incremental_pause_manager.py +561 -0
  213. claude_mpm/services/cli/session_resume_helper.py +10 -2
  214. claude_mpm/services/delegation_detector.py +175 -0
  215. claude_mpm/services/diagnostics/checks/agent_sources_check.py +30 -0
  216. claude_mpm/services/diagnostics/checks/configuration_check.py +24 -0
  217. claude_mpm/services/diagnostics/checks/installation_check.py +22 -0
  218. claude_mpm/services/diagnostics/checks/mcp_services_check.py +23 -0
  219. claude_mpm/services/diagnostics/doctor_reporter.py +31 -1
  220. claude_mpm/services/diagnostics/models.py +14 -1
  221. claude_mpm/services/event_log.py +325 -0
  222. claude_mpm/services/infrastructure/__init__.py +4 -0
  223. claude_mpm/services/infrastructure/context_usage_tracker.py +291 -0
  224. claude_mpm/services/infrastructure/resume_log_generator.py +24 -5
  225. claude_mpm/services/monitor/daemon_manager.py +15 -4
  226. claude_mpm/services/monitor/management/lifecycle.py +8 -2
  227. claude_mpm/services/monitor/server.py +106 -16
  228. claude_mpm/services/pm_skills_deployer.py +261 -85
  229. claude_mpm/services/skills/git_skill_source_manager.py +75 -10
  230. claude_mpm/services/skills/selective_skill_deployer.py +177 -80
  231. claude_mpm/services/skills/skill_discovery_service.py +57 -3
  232. claude_mpm/services/socketio/handlers/hook.py +14 -7
  233. claude_mpm/services/socketio/server/main.py +12 -4
  234. claude_mpm/skills/bundled/collaboration/brainstorming/SKILL.md +79 -0
  235. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/SKILL.md +178 -0
  236. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/agent-prompts.md +577 -0
  237. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/coordination-patterns.md +467 -0
  238. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/examples.md +537 -0
  239. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/troubleshooting.md +730 -0
  240. claude_mpm/skills/bundled/collaboration/git-worktrees.md +317 -0
  241. claude_mpm/skills/bundled/collaboration/requesting-code-review/SKILL.md +112 -0
  242. claude_mpm/skills/bundled/collaboration/requesting-code-review/references/code-reviewer-template.md +146 -0
  243. claude_mpm/skills/bundled/collaboration/requesting-code-review/references/review-examples.md +412 -0
  244. claude_mpm/skills/bundled/collaboration/stacked-prs.md +251 -0
  245. claude_mpm/skills/bundled/collaboration/writing-plans/SKILL.md +81 -0
  246. claude_mpm/skills/bundled/collaboration/writing-plans/references/best-practices.md +362 -0
  247. claude_mpm/skills/bundled/collaboration/writing-plans/references/plan-structure-templates.md +312 -0
  248. claude_mpm/skills/bundled/debugging/root-cause-tracing/SKILL.md +152 -0
  249. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/advanced-techniques.md +668 -0
  250. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/examples.md +587 -0
  251. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/integration.md +438 -0
  252. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/tracing-techniques.md +391 -0
  253. claude_mpm/skills/bundled/debugging/systematic-debugging/CREATION-LOG.md +119 -0
  254. claude_mpm/skills/bundled/debugging/systematic-debugging/SKILL.md +148 -0
  255. claude_mpm/skills/bundled/debugging/systematic-debugging/references/anti-patterns.md +483 -0
  256. claude_mpm/skills/bundled/debugging/systematic-debugging/references/examples.md +452 -0
  257. claude_mpm/skills/bundled/debugging/systematic-debugging/references/troubleshooting.md +449 -0
  258. claude_mpm/skills/bundled/debugging/systematic-debugging/references/workflow.md +411 -0
  259. claude_mpm/skills/bundled/debugging/systematic-debugging/test-academic.md +14 -0
  260. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-1.md +58 -0
  261. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-2.md +68 -0
  262. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-3.md +69 -0
  263. claude_mpm/skills/bundled/debugging/verification-before-completion/SKILL.md +131 -0
  264. claude_mpm/skills/bundled/debugging/verification-before-completion/references/gate-function.md +325 -0
  265. claude_mpm/skills/bundled/debugging/verification-before-completion/references/integration-and-workflows.md +490 -0
  266. claude_mpm/skills/bundled/debugging/verification-before-completion/references/red-flags-and-failures.md +425 -0
  267. claude_mpm/skills/bundled/debugging/verification-before-completion/references/verification-patterns.md +499 -0
  268. claude_mpm/skills/bundled/infrastructure/env-manager/INTEGRATION.md +611 -0
  269. claude_mpm/skills/bundled/infrastructure/env-manager/README.md +596 -0
  270. claude_mpm/skills/bundled/infrastructure/env-manager/SKILL.md +260 -0
  271. claude_mpm/skills/bundled/infrastructure/env-manager/examples/nextjs-env-structure.md +315 -0
  272. claude_mpm/skills/bundled/infrastructure/env-manager/references/frameworks.md +436 -0
  273. claude_mpm/skills/bundled/infrastructure/env-manager/references/security.md +433 -0
  274. claude_mpm/skills/bundled/infrastructure/env-manager/references/synchronization.md +452 -0
  275. claude_mpm/skills/bundled/infrastructure/env-manager/references/troubleshooting.md +404 -0
  276. claude_mpm/skills/bundled/infrastructure/env-manager/references/validation.md +420 -0
  277. claude_mpm/skills/bundled/main/artifacts-builder/SKILL.md +86 -0
  278. claude_mpm/skills/bundled/main/internal-comms/SKILL.md +43 -0
  279. claude_mpm/skills/bundled/main/internal-comms/examples/3p-updates.md +47 -0
  280. claude_mpm/skills/bundled/main/internal-comms/examples/company-newsletter.md +65 -0
  281. claude_mpm/skills/bundled/main/internal-comms/examples/faq-answers.md +30 -0
  282. claude_mpm/skills/bundled/main/internal-comms/examples/general-comms.md +16 -0
  283. claude_mpm/skills/bundled/main/mcp-builder/SKILL.md +160 -0
  284. claude_mpm/skills/bundled/main/mcp-builder/reference/design_principles.md +412 -0
  285. claude_mpm/skills/bundled/main/mcp-builder/reference/evaluation.md +602 -0
  286. claude_mpm/skills/bundled/main/mcp-builder/reference/mcp_best_practices.md +915 -0
  287. claude_mpm/skills/bundled/main/mcp-builder/reference/node_mcp_server.md +916 -0
  288. claude_mpm/skills/bundled/main/mcp-builder/reference/python_mcp_server.md +752 -0
  289. claude_mpm/skills/bundled/main/mcp-builder/reference/workflow.md +1237 -0
  290. claude_mpm/skills/bundled/main/skill-creator/SKILL.md +189 -0
  291. claude_mpm/skills/bundled/main/skill-creator/references/best-practices.md +500 -0
  292. claude_mpm/skills/bundled/main/skill-creator/references/creation-workflow.md +464 -0
  293. claude_mpm/skills/bundled/main/skill-creator/references/examples.md +619 -0
  294. claude_mpm/skills/bundled/main/skill-creator/references/progressive-disclosure.md +437 -0
  295. claude_mpm/skills/bundled/main/skill-creator/references/skill-structure.md +231 -0
  296. claude_mpm/skills/bundled/php/espocrm-development/SKILL.md +170 -0
  297. claude_mpm/skills/bundled/php/espocrm-development/references/architecture.md +602 -0
  298. claude_mpm/skills/bundled/php/espocrm-development/references/common-tasks.md +821 -0
  299. claude_mpm/skills/bundled/php/espocrm-development/references/development-workflow.md +742 -0
  300. claude_mpm/skills/bundled/php/espocrm-development/references/frontend-customization.md +726 -0
  301. claude_mpm/skills/bundled/php/espocrm-development/references/hooks-and-services.md +764 -0
  302. claude_mpm/skills/bundled/php/espocrm-development/references/testing-debugging.md +831 -0
  303. claude_mpm/skills/bundled/pm/mpm/SKILL.md +38 -0
  304. claude_mpm/skills/bundled/pm/mpm-agent-update-workflow/SKILL.md +75 -0
  305. claude_mpm/skills/bundled/pm/mpm-bug-reporting/SKILL.md +248 -0
  306. claude_mpm/skills/bundled/pm/mpm-circuit-breaker-enforcement/SKILL.md +476 -0
  307. claude_mpm/skills/bundled/pm/mpm-config/SKILL.md +29 -0
  308. claude_mpm/skills/bundled/pm/mpm-delegation-patterns/SKILL.md +167 -0
  309. claude_mpm/skills/bundled/pm/mpm-doctor/SKILL.md +53 -0
  310. claude_mpm/skills/bundled/pm/mpm-git-file-tracking/SKILL.md +113 -0
  311. claude_mpm/skills/bundled/pm/mpm-help/SKILL.md +35 -0
  312. claude_mpm/skills/bundled/pm/mpm-init/SKILL.md +125 -0
  313. claude_mpm/skills/bundled/pm/mpm-monitor/SKILL.md +32 -0
  314. claude_mpm/skills/bundled/pm/mpm-organize/SKILL.md +121 -0
  315. claude_mpm/skills/bundled/pm/mpm-postmortem/SKILL.md +22 -0
  316. claude_mpm/skills/bundled/pm/mpm-pr-workflow/SKILL.md +124 -0
  317. claude_mpm/skills/bundled/pm/mpm-session-management/SKILL.md +312 -0
  318. claude_mpm/skills/bundled/pm/mpm-session-resume/SKILL.md +31 -0
  319. claude_mpm/skills/bundled/pm/mpm-status/SKILL.md +37 -0
  320. claude_mpm/skills/bundled/pm/mpm-teaching-mode/SKILL.md +657 -0
  321. claude_mpm/skills/bundled/pm/mpm-ticket-view/SKILL.md +110 -0
  322. claude_mpm/skills/bundled/pm/mpm-ticketing-integration/SKILL.md +154 -0
  323. claude_mpm/skills/bundled/pm/mpm-tool-usage-guide/SKILL.md +386 -0
  324. claude_mpm/skills/bundled/pm/mpm-verification-protocols/SKILL.md +198 -0
  325. claude_mpm/skills/bundled/pm/mpm-version/SKILL.md +21 -0
  326. claude_mpm/skills/bundled/react/flexlayout-react.md +742 -0
  327. claude_mpm/skills/bundled/rust/desktop-applications/SKILL.md +226 -0
  328. claude_mpm/skills/bundled/rust/desktop-applications/references/architecture-patterns.md +901 -0
  329. claude_mpm/skills/bundled/rust/desktop-applications/references/native-gui-frameworks.md +901 -0
  330. claude_mpm/skills/bundled/rust/desktop-applications/references/platform-integration.md +775 -0
  331. claude_mpm/skills/bundled/rust/desktop-applications/references/state-management.md +937 -0
  332. claude_mpm/skills/bundled/rust/desktop-applications/references/tauri-framework.md +770 -0
  333. claude_mpm/skills/bundled/rust/desktop-applications/references/testing-deployment.md +961 -0
  334. claude_mpm/skills/bundled/tauri/tauri-async-patterns.md +495 -0
  335. claude_mpm/skills/bundled/tauri/tauri-build-deploy.md +599 -0
  336. claude_mpm/skills/bundled/tauri/tauri-command-patterns.md +535 -0
  337. claude_mpm/skills/bundled/tauri/tauri-error-handling.md +613 -0
  338. claude_mpm/skills/bundled/tauri/tauri-event-system.md +648 -0
  339. claude_mpm/skills/bundled/tauri/tauri-file-system.md +673 -0
  340. claude_mpm/skills/bundled/tauri/tauri-frontend-integration.md +767 -0
  341. claude_mpm/skills/bundled/tauri/tauri-performance.md +669 -0
  342. claude_mpm/skills/bundled/tauri/tauri-state-management.md +573 -0
  343. claude_mpm/skills/bundled/tauri/tauri-testing.md +384 -0
  344. claude_mpm/skills/bundled/tauri/tauri-window-management.md +628 -0
  345. claude_mpm/skills/bundled/testing/condition-based-waiting/SKILL.md +119 -0
  346. claude_mpm/skills/bundled/testing/condition-based-waiting/references/patterns-and-implementation.md +253 -0
  347. claude_mpm/skills/bundled/testing/test-driven-development/SKILL.md +145 -0
  348. claude_mpm/skills/bundled/testing/test-driven-development/references/anti-patterns.md +543 -0
  349. claude_mpm/skills/bundled/testing/test-driven-development/references/examples.md +741 -0
  350. claude_mpm/skills/bundled/testing/test-driven-development/references/integration.md +470 -0
  351. claude_mpm/skills/bundled/testing/test-driven-development/references/philosophy.md +458 -0
  352. claude_mpm/skills/bundled/testing/test-driven-development/references/workflow.md +639 -0
  353. claude_mpm/skills/bundled/testing/test-quality-inspector/SKILL.md +458 -0
  354. claude_mpm/skills/bundled/testing/test-quality-inspector/examples/example-inspection-report.md +411 -0
  355. claude_mpm/skills/bundled/testing/test-quality-inspector/references/assertion-quality.md +317 -0
  356. claude_mpm/skills/bundled/testing/test-quality-inspector/references/inspection-checklist.md +270 -0
  357. claude_mpm/skills/bundled/testing/test-quality-inspector/references/red-flags.md +436 -0
  358. claude_mpm/skills/bundled/testing/testing-anti-patterns/SKILL.md +140 -0
  359. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/completeness-anti-patterns.md +572 -0
  360. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/core-anti-patterns.md +411 -0
  361. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/detection-guide.md +569 -0
  362. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/tdd-connection.md +695 -0
  363. claude_mpm/skills/bundled/testing/webapp-testing/SKILL.md +184 -0
  364. claude_mpm/skills/bundled/testing/webapp-testing/decision-tree.md +459 -0
  365. claude_mpm/skills/bundled/testing/webapp-testing/playwright-patterns.md +479 -0
  366. claude_mpm/skills/bundled/testing/webapp-testing/reconnaissance-pattern.md +687 -0
  367. claude_mpm/skills/bundled/testing/webapp-testing/server-management.md +758 -0
  368. claude_mpm/skills/bundled/testing/webapp-testing/troubleshooting.md +868 -0
  369. claude_mpm/skills/skill_manager.py +4 -4
  370. claude_mpm/utils/agent_dependency_loader.py +103 -4
  371. claude_mpm/utils/robust_installer.py +45 -24
  372. claude_mpm-5.6.1.dist-info/METADATA +391 -0
  373. {claude_mpm-5.4.55.dist-info → claude_mpm-5.6.1.dist-info}/RECORD +377 -166
  374. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/0.DWzvg0-y.css +0 -1
  375. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/2.ThTw9_ym.css +0 -1
  376. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/4TdZjIqw.js +0 -1
  377. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/5shd3_w0.js +0 -24
  378. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BKjSRqUr.js +0 -1
  379. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Da0KfYnO.js +0 -1
  380. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Dfy6j1xT.js +0 -323
  381. claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/start.NWzMBYRp.js +0 -1
  382. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/2.C0GcWctS.js +0 -1
  383. claude_mpm/hooks/claude_hooks/__pycache__/__init__.cpython-311.pyc +0 -0
  384. claude_mpm/hooks/claude_hooks/__pycache__/correlation_manager.cpython-311.pyc +0 -0
  385. claude_mpm/hooks/claude_hooks/__pycache__/event_handlers.cpython-311.pyc +0 -0
  386. claude_mpm/hooks/claude_hooks/__pycache__/hook_handler.cpython-311.pyc +0 -0
  387. claude_mpm/hooks/claude_hooks/__pycache__/installer.cpython-311.pyc +0 -0
  388. claude_mpm/hooks/claude_hooks/__pycache__/memory_integration.cpython-311.pyc +0 -0
  389. claude_mpm/hooks/claude_hooks/__pycache__/response_tracking.cpython-311.pyc +0 -0
  390. claude_mpm/hooks/claude_hooks/__pycache__/tool_analysis.cpython-311.pyc +0 -0
  391. claude_mpm/hooks/claude_hooks/services/__pycache__/__init__.cpython-311.pyc +0 -0
  392. claude_mpm/hooks/claude_hooks/services/__pycache__/connection_manager_http.cpython-311.pyc +0 -0
  393. claude_mpm/hooks/claude_hooks/services/__pycache__/duplicate_detector.cpython-311.pyc +0 -0
  394. claude_mpm/hooks/claude_hooks/services/__pycache__/state_manager.cpython-311.pyc +0 -0
  395. claude_mpm/hooks/claude_hooks/services/__pycache__/subagent_processor.cpython-311.pyc +0 -0
  396. claude_mpm-5.4.55.dist-info/METADATA +0 -999
  397. {claude_mpm-5.4.55.dist-info → claude_mpm-5.6.1.dist-info}/WHEEL +0 -0
  398. {claude_mpm-5.4.55.dist-info → claude_mpm-5.6.1.dist-info}/entry_points.txt +0 -0
  399. {claude_mpm-5.4.55.dist-info → claude_mpm-5.6.1.dist-info}/licenses/LICENSE +0 -0
  400. {claude_mpm-5.4.55.dist-info → claude_mpm-5.6.1.dist-info}/licenses/LICENSE-FAQ.md +0 -0
  401. {claude_mpm-5.4.55.dist-info → claude_mpm-5.6.1.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,742 @@
1
+ # Development Workflow Reference
2
+
3
+ ## Extension Development Setup
4
+
5
+ ### Using ext-template
6
+
7
+ The official ext-template provides the recommended structure for EspoCRM extensions.
8
+
9
+ ```bash
10
+ # Clone the template
11
+ git clone https://github.com/espocrm/ext-template.git my-extension
12
+ cd my-extension
13
+
14
+ # Install dependencies
15
+ composer install
16
+ npm install
17
+ ```
18
+
19
+ ### Extension Directory Structure (EspoCRM 7.4+)
20
+
21
+ ```
22
+ my-extension/
23
+ ├── src/
24
+ │ ├── files/
25
+ │ │ └── custom/Espo/Modules/MyModule/
26
+ │ │ ├── Resources/
27
+ │ │ │ ├── metadata/
28
+ │ │ │ │ ├── entityDefs/
29
+ │ │ │ │ ├── clientDefs/
30
+ │ │ │ │ └── scopes/
31
+ │ │ │ └── layouts/
32
+ │ │ ├── Services/
33
+ │ │ ├── Controllers/
34
+ │ │ ├── Repositories/
35
+ │ │ ├── Hooks/
36
+ │ │ └── Entities/
37
+ │ └── scripts/
38
+ ├── tests/
39
+ ├── package.json
40
+ └── manifest.json
41
+ ```
42
+
43
+ ### Manifest File
44
+
45
+ ```json
46
+ {
47
+ "name": "My Extension",
48
+ "version": "1.0.0",
49
+ "acceptableVersions": [">=7.4.0"],
50
+ "author": "Your Name",
51
+ "description": "Extension description",
52
+ "license": "MIT",
53
+ "releaseDate": "2024-01-01",
54
+ "skipBackup": true
55
+ }
56
+ ```
57
+
58
+ ### Building Extension
59
+
60
+ ```bash
61
+ # Build installable package
62
+ npm run build
63
+
64
+ # Output: build/MyExtension-1.0.0.zip
65
+ ```
66
+
67
+ ## Creating Custom Entities
68
+
69
+ ### Step 1: Entity Definition
70
+
71
+ Create `src/files/custom/Espo/Modules/MyModule/Resources/metadata/entityDefs/MyEntity.json`:
72
+
73
+ ```json
74
+ {
75
+ "fields": {
76
+ "name": {
77
+ "type": "varchar",
78
+ "required": true,
79
+ "maxLength": 255,
80
+ "trim": true,
81
+ "view": "views/fields/varchar"
82
+ },
83
+ "description": {
84
+ "type": "text",
85
+ "rows": 4
86
+ },
87
+ "status": {
88
+ "type": "enum",
89
+ "options": ["New", "In Progress", "Complete", "Cancelled"],
90
+ "default": "New",
91
+ "required": true,
92
+ "audited": true,
93
+ "isSorted": true
94
+ },
95
+ "priority": {
96
+ "type": "enum",
97
+ "options": ["Low", "Normal", "High", "Urgent"],
98
+ "default": "Normal",
99
+ "audited": true
100
+ },
101
+ "dueDate": {
102
+ "type": "date",
103
+ "audited": true
104
+ },
105
+ "assignedUser": {
106
+ "type": "link"
107
+ },
108
+ "account": {
109
+ "type": "link"
110
+ },
111
+ "contacts": {
112
+ "type": "linkMultiple"
113
+ }
114
+ },
115
+ "links": {
116
+ "assignedUser": {
117
+ "type": "belongsTo",
118
+ "entity": "User",
119
+ "foreign": "myEntities"
120
+ },
121
+ "account": {
122
+ "type": "belongsTo",
123
+ "entity": "Account",
124
+ "foreign": "myEntities"
125
+ },
126
+ "contacts": {
127
+ "type": "hasMany",
128
+ "entity": "Contact",
129
+ "foreign": "myEntities",
130
+ "layoutRelationshipsDisabled": true
131
+ },
132
+ "teams": {
133
+ "type": "hasMany",
134
+ "entity": "Team",
135
+ "relationName": "EntityTeam",
136
+ "layoutRelationshipsDisabled": true
137
+ }
138
+ },
139
+ "collection": {
140
+ "orderBy": "createdAt",
141
+ "order": "desc"
142
+ },
143
+ "indexes": {
144
+ "name": {
145
+ "columns": ["name", "deleted"]
146
+ },
147
+ "assignedUser": {
148
+ "columns": ["assignedUserId", "deleted"]
149
+ }
150
+ }
151
+ }
152
+ ```
153
+
154
+ ### Step 2: Scope Definition
155
+
156
+ Create `src/files/custom/Espo/Modules/MyModule/Resources/metadata/scopes/MyEntity.json`:
157
+
158
+ ```json
159
+ {
160
+ "entity": true,
161
+ "object": true,
162
+ "layouts": true,
163
+ "tab": true,
164
+ "acl": true,
165
+ "aclActionList": [
166
+ "create",
167
+ "read",
168
+ "edit",
169
+ "delete",
170
+ "stream"
171
+ ],
172
+ "aclLevelList": [
173
+ "all",
174
+ "team",
175
+ "own",
176
+ "no"
177
+ ],
178
+ "aclPortal": true,
179
+ "aclPortalLevelList": [
180
+ "all",
181
+ "account",
182
+ "contact",
183
+ "own",
184
+ "no"
185
+ ],
186
+ "customizable": true,
187
+ "type": "Base",
188
+ "module": "MyModule",
189
+ "stream": true,
190
+ "activities": true,
191
+ "historyDisabled": false,
192
+ "importable": true,
193
+ "notifications": true,
194
+ "activityStatusList": ["Planned", "Held", "Not Held"]
195
+ }
196
+ ```
197
+
198
+ ### Step 3: Client Definitions
199
+
200
+ Create `src/files/custom/Espo/Modules/MyModule/Resources/metadata/clientDefs/MyEntity.json`:
201
+
202
+ ```json
203
+ {
204
+ "controller": "controllers/record",
205
+ "iconClass": "fas fa-tasks",
206
+ "color": "#6FA8D6",
207
+ "createDisabled": false,
208
+ "dynamicLogic": {
209
+ "fields": {
210
+ "dueDate": {
211
+ "required": {
212
+ "conditionGroup": [
213
+ {
214
+ "type": "equals",
215
+ "attribute": "status",
216
+ "value": "In Progress"
217
+ }
218
+ ]
219
+ }
220
+ }
221
+ }
222
+ },
223
+ "filterList": [
224
+ "active",
225
+ "completed"
226
+ ],
227
+ "boolFilterList": [
228
+ "onlyMy"
229
+ ],
230
+ "defaultFilterData": {
231
+ "primary": "active"
232
+ },
233
+ "sidePanels": {
234
+ "detail": [
235
+ {
236
+ "name": "activities",
237
+ "label": "Activities",
238
+ "view": "crm:views/record/panels/activities",
239
+ "aclScope": "Activities"
240
+ }
241
+ ]
242
+ }
243
+ }
244
+ ```
245
+
246
+ ### Step 4: Language Translations
247
+
248
+ Create `src/files/custom/Espo/Modules/MyModule/Resources/i18n/en_US/MyEntity.json`:
249
+
250
+ ```json
251
+ {
252
+ "fields": {
253
+ "name": "Name",
254
+ "description": "Description",
255
+ "status": "Status",
256
+ "priority": "Priority",
257
+ "dueDate": "Due Date",
258
+ "assignedUser": "Assigned To",
259
+ "account": "Account",
260
+ "contacts": "Contacts"
261
+ },
262
+ "links": {
263
+ "assignedUser": "Assigned To",
264
+ "account": "Account",
265
+ "contacts": "Contacts"
266
+ },
267
+ "options": {
268
+ "status": {
269
+ "New": "New",
270
+ "In Progress": "In Progress",
271
+ "Complete": "Complete",
272
+ "Cancelled": "Cancelled"
273
+ },
274
+ "priority": {
275
+ "Low": "Low",
276
+ "Normal": "Normal",
277
+ "High": "High",
278
+ "Urgent": "Urgent"
279
+ }
280
+ },
281
+ "labels": {
282
+ "Create MyEntity": "Create MyEntity"
283
+ },
284
+ "presetFilters": {
285
+ "active": "Active",
286
+ "completed": "Completed"
287
+ },
288
+ "boolFilters": {
289
+ "onlyMy": "Only My"
290
+ }
291
+ }
292
+ ```
293
+
294
+ ### Step 5: Service Layer
295
+
296
+ Create `src/files/custom/Espo/Modules/MyModule/Services/MyEntity.php`:
297
+
298
+ ```php
299
+ <?php
300
+ namespace Espo\Modules\MyModule\Services;
301
+
302
+ use Espo\Services\Record;
303
+ use Espo\ORM\Entity;
304
+
305
+ class MyEntity extends Record
306
+ {
307
+ protected function beforeCreateEntity(Entity $entity, array $data): void
308
+ {
309
+ parent::beforeCreateEntity($entity, $data);
310
+
311
+ // Set default assigned user to creator if not specified
312
+ if (!$entity->get('assignedUserId')) {
313
+ $entity->set('assignedUserId', $this->user->getId());
314
+ }
315
+ }
316
+
317
+ protected function beforeUpdateEntity(Entity $entity, array $data): void
318
+ {
319
+ parent::beforeUpdateEntity($entity, $data);
320
+
321
+ // Auto-complete when status set to Complete
322
+ if ($entity->isAttributeChanged('status') && $entity->get('status') === 'Complete') {
323
+ $entity->set('completedAt', date('Y-m-d H:i:s'));
324
+ }
325
+ }
326
+ }
327
+ ```
328
+
329
+ ### Step 6: Rebuild Cache
330
+
331
+ ```bash
332
+ bin/command rebuild
333
+ ```
334
+
335
+ ## Creating Custom Fields
336
+
337
+ ### Custom Field Type Definition
338
+
339
+ Create `src/files/custom/Espo/Modules/MyModule/Resources/metadata/fields/myFieldType.json`:
340
+
341
+ ```json
342
+ {
343
+ "params": [
344
+ {
345
+ "name": "required",
346
+ "type": "bool",
347
+ "default": false
348
+ },
349
+ {
350
+ "name": "maxLength",
351
+ "type": "int"
352
+ },
353
+ {
354
+ "name": "customParam",
355
+ "type": "varchar"
356
+ }
357
+ ],
358
+ "view": "custom:views/fields/my-field-type",
359
+ "personalData": false
360
+ }
361
+ ```
362
+
363
+ ### Custom Field Backend
364
+
365
+ Create `src/files/custom/Espo/Modules/MyModule/Classes/FieldType/MyFieldTypeType.php`:
366
+
367
+ ```php
368
+ <?php
369
+ namespace Espo\Modules\MyModule\Classes\FieldType;
370
+
371
+ use Espo\ORM\Entity;
372
+ use Espo\ORM\Type\AttributeType;
373
+ use Espo\Core\Field\FieldType;
374
+
375
+ class MyFieldTypeType implements FieldType
376
+ {
377
+ public function getAttributeParamList(): array
378
+ {
379
+ return [
380
+ AttributeType::VARCHAR,
381
+ ];
382
+ }
383
+
384
+ public function getActualAttributeParamList(Entity $entity, string $field): array
385
+ {
386
+ return [
387
+ AttributeType::VARCHAR,
388
+ ];
389
+ }
390
+ }
391
+ ```
392
+
393
+ ### Custom Field Frontend
394
+
395
+ Create `client/custom/src/views/fields/my-field-type.js`:
396
+
397
+ ```javascript
398
+ define('custom:views/fields/my-field-type', ['views/fields/varchar'], function (Dep) {
399
+
400
+ return Dep.extend({
401
+
402
+ setup: function () {
403
+ Dep.prototype.setup.call(this);
404
+
405
+ // Custom setup logic
406
+ this.customParam = this.params.customParam || '';
407
+ },
408
+
409
+ afterRender: function () {
410
+ Dep.prototype.afterRender.call(this);
411
+
412
+ // Custom rendering logic
413
+ },
414
+
415
+ validateRequired: function () {
416
+ if (this.params.required) {
417
+ if (!this.model.get(this.name)) {
418
+ var msg = this.translate('fieldIsRequired', 'messages')
419
+ .replace('{field}', this.getLabelText());
420
+ this.showValidationMessage(msg);
421
+ return true;
422
+ }
423
+ }
424
+ },
425
+
426
+ fetch: function () {
427
+ var data = {};
428
+ data[this.name] = this.$element.val() || null;
429
+ return data;
430
+ }
431
+ });
432
+ });
433
+ ```
434
+
435
+ ## Custom API Endpoints
436
+
437
+ ### Step 1: Define Route
438
+
439
+ Create `src/files/custom/Espo/Modules/MyModule/Resources/metadata/api.json`:
440
+
441
+ ```json
442
+ {
443
+ "routes": [
444
+ {
445
+ "route": "/MyEntity/:id/customAction",
446
+ "method": "post",
447
+ "controller": "MyModule:MyEntity",
448
+ "action": "customAction"
449
+ }
450
+ ]
451
+ }
452
+ ```
453
+
454
+ ### Step 2: Create Controller
455
+
456
+ Create `src/files/custom/Espo/Modules/MyModule/Controllers/MyEntity.php`:
457
+
458
+ ```php
459
+ <?php
460
+ namespace Espo\Modules\MyModule\Controllers;
461
+
462
+ use Espo\Core\Controllers\Record;
463
+ use Espo\Core\Api\Request;
464
+ use Espo\Core\Api\Response;
465
+ use Espo\Core\Exceptions\BadRequest;
466
+ use Espo\Core\Exceptions\Forbidden;
467
+ use stdClass;
468
+
469
+ class MyEntity extends Record
470
+ {
471
+ public function postActionCustomAction(Request $request, Response $response): stdClass
472
+ {
473
+ $id = $request->getRouteParam('id');
474
+
475
+ if (!$id) {
476
+ throw new BadRequest();
477
+ }
478
+
479
+ $data = $request->getParsedBody();
480
+
481
+ if (!$this->acl->check($this->name, 'edit')) {
482
+ throw new Forbidden();
483
+ }
484
+
485
+ // Delegate to service layer
486
+ $service = $this->getRecordService();
487
+ $result = $service->customAction($id, $data);
488
+
489
+ return $result->getValueMap();
490
+ }
491
+ }
492
+ ```
493
+
494
+ ### Step 3: Implement Service Method
495
+
496
+ ```php
497
+ <?php
498
+ namespace Espo\Modules\MyModule\Services;
499
+
500
+ use Espo\Services\Record;
501
+ use Espo\Core\Exceptions\NotFound;
502
+ use stdClass;
503
+
504
+ class MyEntity extends Record
505
+ {
506
+ public function customAction(string $id, stdClass $data): object
507
+ {
508
+ $entity = $this->getEntity($id);
509
+
510
+ if (!$entity) {
511
+ throw new NotFound();
512
+ }
513
+
514
+ // Business logic
515
+ $entity->set('status', $data->status ?? 'In Progress');
516
+ $entity->set('processedAt', date('Y-m-d H:i:s'));
517
+
518
+ $this->entityManager->saveEntity($entity);
519
+
520
+ return $entity;
521
+ }
522
+ }
523
+ ```
524
+
525
+ ### Step 4: Call from Frontend
526
+
527
+ ```javascript
528
+ this.ajaxPostRequest('MyEntity/' + id + '/customAction', {
529
+ status: 'Complete'
530
+ }).then(response => {
531
+ console.log('Action completed', response);
532
+ this.model.set(response);
533
+ });
534
+ ```
535
+
536
+ ## Custom Repositories
537
+
538
+ ### Creating Custom Repository
539
+
540
+ Create `src/files/custom/Espo/Modules/MyModule/Repositories/MyEntity.php`:
541
+
542
+ ```php
543
+ <?php
544
+ namespace Espo\Modules\MyModule\Repositories;
545
+
546
+ use Espo\Core\Repositories\Database;
547
+ use Espo\ORM\Entity;
548
+
549
+ class MyEntity extends Database
550
+ {
551
+ protected function beforeSave(Entity $entity, array $options = []): void
552
+ {
553
+ parent::beforeSave($entity, $options);
554
+
555
+ // Repository-level validation or data transformation
556
+ if ($entity->isNew()) {
557
+ $entity->set('customIdentifier', $this->generateIdentifier());
558
+ }
559
+ }
560
+
561
+ private function generateIdentifier(): string
562
+ {
563
+ // Generate unique identifier
564
+ $prefix = 'ME-';
565
+ $number = $this->getNewNumber();
566
+ return $prefix . str_pad($number, 6, '0', STR_PAD_LEFT);
567
+ }
568
+
569
+ private function getNewNumber(): int
570
+ {
571
+ $query = $this->entityManager
572
+ ->getQueryBuilder()
573
+ ->select()
574
+ ->from('MyEntity')
575
+ ->select('COUNT(*) as count')
576
+ ->build();
577
+
578
+ $sth = $this->entityManager->getQueryExecutor()->execute($query);
579
+ $row = $sth->fetch();
580
+
581
+ return ($row['count'] ?? 0) + 1;
582
+ }
583
+
584
+ public function findActive(): \Espo\ORM\Collection
585
+ {
586
+ return $this->where([
587
+ 'status!=' => ['Complete', 'Cancelled']
588
+ ])->find();
589
+ }
590
+ }
591
+ ```
592
+
593
+ ## Layouts
594
+
595
+ ### List Layout
596
+
597
+ Create `src/files/custom/Espo/Modules/MyModule/Resources/layouts/MyEntity/list.json`:
598
+
599
+ ```json
600
+ [
601
+ {
602
+ "name": "name",
603
+ "width": "30"
604
+ },
605
+ {
606
+ "name": "status",
607
+ "width": "15"
608
+ },
609
+ {
610
+ "name": "priority",
611
+ "width": "15"
612
+ },
613
+ {
614
+ "name": "assignedUser",
615
+ "width": "15"
616
+ },
617
+ {
618
+ "name": "dueDate",
619
+ "width": "15"
620
+ },
621
+ {
622
+ "name": "createdAt",
623
+ "width": "10"
624
+ }
625
+ ]
626
+ ```
627
+
628
+ ### Detail Layout
629
+
630
+ Create `src/files/custom/Espo/Modules/MyModule/Resources/layouts/MyEntity/detail.json`:
631
+
632
+ ```json
633
+ [
634
+ {
635
+ "label": "Overview",
636
+ "rows": [
637
+ [
638
+ {"name": "name"},
639
+ {"name": "status"}
640
+ ],
641
+ [
642
+ {"name": "assignedUser"},
643
+ {"name": "priority"}
644
+ ],
645
+ [
646
+ {"name": "account"},
647
+ {"name": "dueDate"}
648
+ ],
649
+ [
650
+ {"name": "description", "fullWidth": true}
651
+ ]
652
+ ]
653
+ },
654
+ {
655
+ "label": "Contacts",
656
+ "rows": [
657
+ [
658
+ {"name": "contacts", "fullWidth": true}
659
+ ]
660
+ ]
661
+ }
662
+ ]
663
+ ```
664
+
665
+ ## Development Best Practices
666
+
667
+ ### Cache Rebuild Workflow
668
+
669
+ ```bash
670
+ # After any metadata changes
671
+ bin/command rebuild
672
+
673
+ # Clear cache only (faster, but may miss some changes)
674
+ bin/command clear-cache
675
+
676
+ # Hard rebuild (if issues persist)
677
+ rm -rf data/cache/*
678
+ bin/command rebuild
679
+ ```
680
+
681
+ ### Testing Extension Installation
682
+
683
+ ```bash
684
+ # Build extension
685
+ npm run build
686
+
687
+ # Install in test EspoCRM instance
688
+ # Administration > Extensions > Upload
689
+ # Upload build/MyExtension-1.0.0.zip
690
+
691
+ # After changes, rebuild extension and reinstall
692
+ npm run build
693
+ # Uninstall old version via Administration > Extensions
694
+ # Install new version
695
+ ```
696
+
697
+ ### Version Compatibility
698
+
699
+ ```php
700
+ // Check EspoCRM version in code
701
+ $version = $this->config->get('version');
702
+
703
+ if (version_compare($version, '8.0.0', '>=')) {
704
+ // EspoCRM 8.0+ features
705
+ }
706
+
707
+ // Use version-specific metadata
708
+ // For EspoCRM 7.x
709
+ custom/Espo/Modules/MyModule/Resources/metadata/entityDefs/MyEntity.json
710
+
711
+ // For EspoCRM 8.x+
712
+ custom/Espo/Modules/MyModule/Resources/metadata/entityDefs/MyEntity/MyEntity.json
713
+ ```
714
+
715
+ ### Debugging Development Issues
716
+
717
+ ```bash
718
+ # Enable debug mode
719
+ # data/config.php
720
+ 'logger' => [
721
+ 'level' => 'DEBUG',
722
+ ],
723
+
724
+ # Check logs
725
+ tail -f data/logs/espo-$(date +%Y-%m-%d).log
726
+
727
+ # Check for PHP errors
728
+ tail -f /var/log/apache2/error.log # or nginx error log
729
+ ```
730
+
731
+ ### Module Dependencies
732
+
733
+ ```json
734
+ // In manifest.json
735
+ {
736
+ "dependencies": {
737
+ "Advanced Pack": {
738
+ "version": ">=2.14.0"
739
+ }
740
+ }
741
+ }
742
+ ```