claude-mpm 4.25.10__py3-none-any.whl → 5.1.8__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 (507) hide show
  1. claude_mpm/VERSION +1 -1
  2. claude_mpm/agents/BASE_PM.md +12 -0
  3. claude_mpm/agents/PM_INSTRUCTIONS.md +1055 -2230
  4. claude_mpm/agents/PM_INSTRUCTIONS_TEACH.md +1322 -0
  5. claude_mpm/agents/WORKFLOW.md +4 -4
  6. claude_mpm/agents/__init__.py +6 -0
  7. claude_mpm/agents/agent_loader.py +1 -4
  8. claude_mpm/agents/base_agent_loader.py +10 -35
  9. claude_mpm/agents/templates/{circuit_breakers.md → circuit-breakers.md} +576 -66
  10. claude_mpm/agents/templates/context-management-examples.md +544 -0
  11. claude_mpm/agents/templates/pr-workflow-examples.md +427 -0
  12. claude_mpm/agents/templates/research-gate-examples.md +669 -0
  13. claude_mpm/agents/templates/structured-questions-examples.md +615 -0
  14. claude_mpm/agents/templates/ticket-completeness-examples.md +139 -0
  15. claude_mpm/agents/templates/ticketing-examples.md +277 -0
  16. claude_mpm/cli/__init__.py +28 -3
  17. claude_mpm/cli/commands/__init__.py +2 -0
  18. claude_mpm/cli/commands/agent_source.py +774 -0
  19. claude_mpm/cli/commands/agent_state_manager.py +188 -30
  20. claude_mpm/cli/commands/agents.py +959 -36
  21. claude_mpm/cli/commands/agents_cleanup.py +210 -0
  22. claude_mpm/cli/commands/agents_discover.py +338 -0
  23. claude_mpm/cli/commands/auto_configure.py +537 -239
  24. claude_mpm/cli/commands/config.py +7 -4
  25. claude_mpm/cli/commands/configure.py +924 -45
  26. claude_mpm/cli/commands/configure_navigation.py +63 -46
  27. claude_mpm/cli/commands/doctor.py +10 -2
  28. claude_mpm/cli/commands/local_deploy.py +1 -4
  29. claude_mpm/cli/commands/postmortem.py +401 -0
  30. claude_mpm/cli/commands/run.py +1 -39
  31. claude_mpm/cli/commands/skill_source.py +694 -0
  32. claude_mpm/cli/commands/skills.py +322 -19
  33. claude_mpm/cli/executor.py +22 -3
  34. claude_mpm/cli/interactive/agent_wizard.py +1028 -43
  35. claude_mpm/cli/parsers/agent_source_parser.py +171 -0
  36. claude_mpm/cli/parsers/agents_parser.py +256 -4
  37. claude_mpm/cli/parsers/auto_configure_parser.py +13 -0
  38. claude_mpm/cli/parsers/base_parser.py +25 -0
  39. claude_mpm/cli/parsers/config_parser.py +96 -43
  40. claude_mpm/cli/parsers/skill_source_parser.py +169 -0
  41. claude_mpm/cli/parsers/skills_parser.py +7 -0
  42. claude_mpm/cli/parsers/source_parser.py +138 -0
  43. claude_mpm/cli/startup.py +456 -103
  44. claude_mpm/cli/startup_display.py +4 -4
  45. claude_mpm/commands/{mpm-auto-configure.md → mpm-agents-auto-configure.md} +9 -0
  46. claude_mpm/commands/mpm-agents-detect.md +9 -0
  47. claude_mpm/commands/{mpm-agents.md → mpm-agents-list.md} +9 -0
  48. claude_mpm/commands/mpm-agents-recommend.md +9 -0
  49. claude_mpm/commands/{mpm-config.md → mpm-config-view.md} +9 -0
  50. claude_mpm/commands/mpm-doctor.md +9 -0
  51. claude_mpm/commands/mpm-help.md +14 -2
  52. claude_mpm/commands/mpm-init.md +9 -0
  53. claude_mpm/commands/mpm-monitor.md +9 -0
  54. claude_mpm/commands/mpm-postmortem.md +123 -0
  55. claude_mpm/commands/{mpm-resume.md → mpm-session-resume.md} +9 -0
  56. claude_mpm/commands/mpm-status.md +9 -0
  57. claude_mpm/commands/{mpm-organize.md → mpm-ticket-organize.md} +9 -0
  58. claude_mpm/commands/mpm-ticket-view.md +552 -0
  59. claude_mpm/commands/mpm-version.md +9 -0
  60. claude_mpm/commands/mpm.md +10 -0
  61. claude_mpm/config/agent_presets.py +488 -0
  62. claude_mpm/config/agent_sources.py +325 -0
  63. claude_mpm/config/skill_presets.py +392 -0
  64. claude_mpm/config/skill_sources.py +590 -0
  65. claude_mpm/constants.py +1 -0
  66. claude_mpm/core/claude_runner.py +5 -34
  67. claude_mpm/core/config.py +16 -0
  68. claude_mpm/core/framework/__init__.py +3 -16
  69. claude_mpm/core/framework/loaders/file_loader.py +54 -101
  70. claude_mpm/core/framework/loaders/instruction_loader.py +25 -5
  71. claude_mpm/core/interactive_session.py +83 -7
  72. claude_mpm/core/oneshot_session.py +71 -8
  73. claude_mpm/core/protocols/__init__.py +23 -0
  74. claude_mpm/core/protocols/runner_protocol.py +103 -0
  75. claude_mpm/core/protocols/session_protocol.py +131 -0
  76. claude_mpm/core/shared/singleton_manager.py +11 -4
  77. claude_mpm/core/system_context.py +38 -0
  78. claude_mpm/core/unified_config.py +22 -0
  79. claude_mpm/experimental/cli_enhancements.py +1 -5
  80. claude_mpm/hooks/claude_hooks/__pycache__/__init__.cpython-313.pyc +0 -0
  81. claude_mpm/hooks/claude_hooks/__pycache__/event_handlers.cpython-313.pyc +0 -0
  82. claude_mpm/hooks/claude_hooks/__pycache__/hook_handler.cpython-313.pyc +0 -0
  83. claude_mpm/hooks/claude_hooks/__pycache__/memory_integration.cpython-313.pyc +0 -0
  84. claude_mpm/hooks/claude_hooks/__pycache__/response_tracking.cpython-313.pyc +0 -0
  85. claude_mpm/hooks/claude_hooks/__pycache__/tool_analysis.cpython-313.pyc +0 -0
  86. claude_mpm/hooks/claude_hooks/memory_integration.py +12 -1
  87. claude_mpm/hooks/claude_hooks/services/__pycache__/__init__.cpython-313.pyc +0 -0
  88. claude_mpm/hooks/claude_hooks/services/__pycache__/connection_manager_http.cpython-313.pyc +0 -0
  89. claude_mpm/hooks/claude_hooks/services/__pycache__/duplicate_detector.cpython-313.pyc +0 -0
  90. claude_mpm/hooks/claude_hooks/services/__pycache__/state_manager.cpython-313.pyc +0 -0
  91. claude_mpm/hooks/claude_hooks/services/__pycache__/subagent_processor.cpython-313.pyc +0 -0
  92. claude_mpm/hooks/failure_learning/__init__.py +2 -8
  93. claude_mpm/hooks/failure_learning/failure_detection_hook.py +1 -6
  94. claude_mpm/hooks/failure_learning/fix_detection_hook.py +1 -6
  95. claude_mpm/hooks/failure_learning/learning_extraction_hook.py +1 -6
  96. claude_mpm/hooks/kuzu_response_hook.py +1 -5
  97. claude_mpm/models/git_repository.py +198 -0
  98. claude_mpm/services/agents/agent_builder.py +45 -9
  99. claude_mpm/services/agents/agent_preset_service.py +238 -0
  100. claude_mpm/services/agents/agent_selection_service.py +484 -0
  101. claude_mpm/services/agents/auto_deploy_index_parser.py +569 -0
  102. claude_mpm/services/agents/cache_git_manager.py +621 -0
  103. claude_mpm/services/agents/deployment/agent_deployment.py +126 -2
  104. claude_mpm/services/agents/deployment/agent_discovery_service.py +105 -73
  105. claude_mpm/services/agents/deployment/agent_lifecycle_manager.py +1 -5
  106. claude_mpm/services/agents/deployment/agent_restore_handler.py +1 -4
  107. claude_mpm/services/agents/deployment/agent_template_builder.py +236 -15
  108. claude_mpm/services/agents/deployment/agents_directory_resolver.py +101 -15
  109. claude_mpm/services/agents/deployment/async_agent_deployment.py +2 -1
  110. claude_mpm/services/agents/deployment/multi_source_deployment_service.py +115 -15
  111. claude_mpm/services/agents/deployment/refactored_agent_deployment_service.py +1 -4
  112. claude_mpm/services/agents/deployment/remote_agent_discovery_service.py +363 -0
  113. claude_mpm/services/agents/deployment/single_agent_deployer.py +2 -2
  114. claude_mpm/services/agents/deployment/system_instructions_deployer.py +168 -46
  115. claude_mpm/services/agents/deployment/validation/deployment_validator.py +2 -2
  116. claude_mpm/services/agents/git_source_manager.py +629 -0
  117. claude_mpm/services/agents/loading/framework_agent_loader.py +1 -4
  118. claude_mpm/services/agents/local_template_manager.py +47 -9
  119. claude_mpm/services/agents/single_tier_deployment_service.py +696 -0
  120. claude_mpm/services/agents/sources/__init__.py +13 -0
  121. claude_mpm/services/agents/sources/agent_sync_state.py +516 -0
  122. claude_mpm/services/agents/sources/git_source_sync_service.py +1087 -0
  123. claude_mpm/services/agents/startup_sync.py +239 -0
  124. claude_mpm/services/agents/toolchain_detector.py +474 -0
  125. claude_mpm/services/analysis/__init__.py +25 -0
  126. claude_mpm/services/analysis/postmortem_reporter.py +474 -0
  127. claude_mpm/services/analysis/postmortem_service.py +765 -0
  128. claude_mpm/services/command_deployment_service.py +200 -6
  129. claude_mpm/services/core/base.py +7 -2
  130. claude_mpm/services/core/interfaces/__init__.py +1 -3
  131. claude_mpm/services/core/interfaces/health.py +1 -4
  132. claude_mpm/services/core/models/__init__.py +2 -11
  133. claude_mpm/services/diagnostics/checks/__init__.py +4 -0
  134. claude_mpm/services/diagnostics/checks/agent_sources_check.py +577 -0
  135. claude_mpm/services/diagnostics/checks/mcp_services_check.py +7 -15
  136. claude_mpm/services/diagnostics/checks/skill_sources_check.py +587 -0
  137. claude_mpm/services/diagnostics/diagnostic_runner.py +9 -0
  138. claude_mpm/services/diagnostics/doctor_reporter.py +34 -6
  139. claude_mpm/services/git/__init__.py +21 -0
  140. claude_mpm/services/git/git_operations_service.py +494 -0
  141. claude_mpm/services/github/__init__.py +21 -0
  142. claude_mpm/services/github/github_cli_service.py +397 -0
  143. claude_mpm/services/infrastructure/monitoring/__init__.py +1 -5
  144. claude_mpm/services/infrastructure/monitoring/aggregator.py +1 -6
  145. claude_mpm/services/instructions/__init__.py +9 -0
  146. claude_mpm/services/instructions/instruction_cache_service.py +374 -0
  147. claude_mpm/services/local_ops/__init__.py +3 -13
  148. claude_mpm/services/local_ops/health_checks/__init__.py +1 -3
  149. claude_mpm/services/local_ops/health_manager.py +1 -4
  150. claude_mpm/services/mcp_config_manager.py +75 -145
  151. claude_mpm/services/mcp_gateway/core/process_pool.py +22 -16
  152. claude_mpm/services/mcp_gateway/server/mcp_gateway.py +1 -6
  153. claude_mpm/services/mcp_service_verifier.py +6 -3
  154. claude_mpm/services/monitor/daemon.py +28 -8
  155. claude_mpm/services/monitor/daemon_manager.py +96 -19
  156. claude_mpm/services/pr/__init__.py +14 -0
  157. claude_mpm/services/pr/pr_template_service.py +329 -0
  158. claude_mpm/services/project/project_organizer.py +4 -0
  159. claude_mpm/services/runner_configuration_service.py +16 -3
  160. claude_mpm/services/session_management_service.py +16 -4
  161. claude_mpm/services/skills/__init__.py +18 -0
  162. claude_mpm/services/skills/git_skill_source_manager.py +1169 -0
  163. claude_mpm/services/skills/skill_discovery_service.py +568 -0
  164. claude_mpm/services/socketio/server/core.py +1 -4
  165. claude_mpm/services/socketio/server/main.py +1 -3
  166. claude_mpm/services/unified/deployment_strategies/vercel.py +1 -5
  167. claude_mpm/services/unified/unified_deployment.py +1 -5
  168. claude_mpm/services/visualization/__init__.py +1 -5
  169. claude_mpm/templates/questions/__init__.py +2 -7
  170. claude_mpm/templates/questions/pr_strategy.py +1 -4
  171. claude_mpm/templates/questions/project_init.py +1 -4
  172. claude_mpm/templates/questions/ticket_mgmt.py +1 -4
  173. claude_mpm/utils/agent_dependency_loader.py +77 -10
  174. claude_mpm/utils/agent_filters.py +288 -0
  175. claude_mpm/utils/gitignore.py +3 -0
  176. claude_mpm/utils/migration.py +372 -0
  177. claude_mpm/utils/progress.py +387 -0
  178. {claude_mpm-4.25.10.dist-info → claude_mpm-5.1.8.dist-info}/METADATA +356 -112
  179. {claude_mpm-4.25.10.dist-info → claude_mpm-5.1.8.dist-info}/RECORD +188 -439
  180. claude_mpm/agents/templates/agent-manager.json +0 -273
  181. claude_mpm/agents/templates/agentic-coder-optimizer.json +0 -248
  182. claude_mpm/agents/templates/api_qa.json +0 -183
  183. claude_mpm/agents/templates/clerk-ops.json +0 -235
  184. claude_mpm/agents/templates/code_analyzer.json +0 -101
  185. claude_mpm/agents/templates/content-agent.json +0 -358
  186. claude_mpm/agents/templates/dart_engineer.json +0 -307
  187. claude_mpm/agents/templates/data_engineer.json +0 -225
  188. claude_mpm/agents/templates/documentation.json +0 -238
  189. claude_mpm/agents/templates/engineer.json +0 -210
  190. claude_mpm/agents/templates/gcp_ops_agent.json +0 -253
  191. claude_mpm/agents/templates/golang_engineer.json +0 -270
  192. claude_mpm/agents/templates/imagemagick.json +0 -264
  193. claude_mpm/agents/templates/java_engineer.json +0 -346
  194. claude_mpm/agents/templates/javascript_engineer_agent.json +0 -380
  195. claude_mpm/agents/templates/local_ops_agent.json +0 -1840
  196. claude_mpm/agents/templates/memory_manager.json +0 -158
  197. claude_mpm/agents/templates/nextjs_engineer.json +0 -285
  198. claude_mpm/agents/templates/ops.json +0 -185
  199. claude_mpm/agents/templates/php-engineer.json +0 -287
  200. claude_mpm/agents/templates/product_owner.json +0 -338
  201. claude_mpm/agents/templates/project_organizer.json +0 -144
  202. claude_mpm/agents/templates/prompt-engineer.json +0 -737
  203. claude_mpm/agents/templates/python_engineer.json +0 -387
  204. claude_mpm/agents/templates/qa.json +0 -243
  205. claude_mpm/agents/templates/react_engineer.json +0 -239
  206. claude_mpm/agents/templates/refactoring_engineer.json +0 -276
  207. claude_mpm/agents/templates/research.json +0 -258
  208. claude_mpm/agents/templates/ruby-engineer.json +0 -280
  209. claude_mpm/agents/templates/rust_engineer.json +0 -275
  210. claude_mpm/agents/templates/security.json +0 -202
  211. claude_mpm/agents/templates/svelte-engineer.json +0 -225
  212. claude_mpm/agents/templates/tauri_engineer.json +0 -274
  213. claude_mpm/agents/templates/ticketing.json +0 -181
  214. claude_mpm/agents/templates/typescript_engineer.json +0 -285
  215. claude_mpm/agents/templates/vercel_ops_agent.json +0 -412
  216. claude_mpm/agents/templates/version_control.json +0 -159
  217. claude_mpm/agents/templates/web_qa.json +0 -400
  218. claude_mpm/agents/templates/web_ui.json +0 -189
  219. claude_mpm/cli/README.md +0 -253
  220. claude_mpm/cli/commands/mcp_install_commands.py.backup +0 -284
  221. claude_mpm/cli/commands/mpm_init/README.md +0 -365
  222. claude_mpm/cli_module/refactoring_guide.md +0 -253
  223. claude_mpm/commands/mpm-tickets.md +0 -151
  224. claude_mpm/config/agent_capabilities.yaml +0 -658
  225. claude_mpm/config/async_logging_config.yaml +0 -145
  226. claude_mpm/core/.claude-mpm/logs/hooks_20250730.log +0 -34
  227. claude_mpm/d2/.gitignore +0 -22
  228. claude_mpm/d2/ARCHITECTURE_COMPARISON.md +0 -273
  229. claude_mpm/d2/FLASK_INTEGRATION.md +0 -156
  230. claude_mpm/d2/IMPLEMENTATION_SUMMARY.md +0 -452
  231. claude_mpm/d2/QUICKSTART.md +0 -186
  232. claude_mpm/d2/README.md +0 -232
  233. claude_mpm/d2/STORE_FIX_SUMMARY.md +0 -167
  234. claude_mpm/d2/SVELTE5_STORES_GUIDE.md +0 -180
  235. claude_mpm/d2/TESTING.md +0 -288
  236. claude_mpm/d2/index.html +0 -118
  237. claude_mpm/d2/package.json +0 -19
  238. claude_mpm/d2/src/App.svelte +0 -110
  239. claude_mpm/d2/src/components/Header.svelte +0 -153
  240. claude_mpm/d2/src/components/MainContent.svelte +0 -74
  241. claude_mpm/d2/src/components/Sidebar.svelte +0 -85
  242. claude_mpm/d2/src/components/tabs/EventsTab.svelte +0 -326
  243. claude_mpm/d2/src/lib/socketio.js +0 -144
  244. claude_mpm/d2/src/main.js +0 -7
  245. claude_mpm/d2/src/stores/events.js +0 -114
  246. claude_mpm/d2/src/stores/socket.js +0 -108
  247. claude_mpm/d2/src/stores/theme.js +0 -65
  248. claude_mpm/d2/svelte.config.js +0 -12
  249. claude_mpm/d2/vite.config.js +0 -15
  250. claude_mpm/dashboard/.claude-mpm/memories/README.md +0 -36
  251. claude_mpm/dashboard/BUILD_NUMBER +0 -1
  252. claude_mpm/dashboard/README.md +0 -121
  253. claude_mpm/dashboard/VERSION +0 -1
  254. claude_mpm/dashboard/react/components/DataInspector/DataInspector.module.css +0 -188
  255. claude_mpm/dashboard/react/components/DataInspector/DataInspector.tsx +0 -273
  256. claude_mpm/dashboard/react/components/ErrorBoundary.tsx +0 -75
  257. claude_mpm/dashboard/react/components/EventViewer/EventViewer.module.css +0 -156
  258. claude_mpm/dashboard/react/components/EventViewer/EventViewer.tsx +0 -141
  259. claude_mpm/dashboard/react/components/shared/ConnectionStatus.module.css +0 -38
  260. claude_mpm/dashboard/react/components/shared/ConnectionStatus.tsx +0 -36
  261. claude_mpm/dashboard/react/components/shared/FilterBar.module.css +0 -92
  262. claude_mpm/dashboard/react/components/shared/FilterBar.tsx +0 -89
  263. claude_mpm/dashboard/react/contexts/DashboardContext.tsx +0 -215
  264. claude_mpm/dashboard/react/entries/events.tsx +0 -165
  265. claude_mpm/dashboard/react/hooks/useEvents.ts +0 -191
  266. claude_mpm/dashboard/react/hooks/useSocket.ts +0 -225
  267. claude_mpm/dashboard/static/archive/activity_dashboard_fixed.html +0 -248
  268. claude_mpm/dashboard/static/built/REFACTORING_SUMMARY.md +0 -170
  269. claude_mpm/dashboard/static/built/assets/events.DjpNxWNo.css +0 -1
  270. claude_mpm/dashboard/static/built/components/activity-tree.js +0 -2
  271. claude_mpm/dashboard/static/built/components/activity-tree.js.map +0 -1
  272. claude_mpm/dashboard/static/built/components/agent-hierarchy.js +0 -777
  273. claude_mpm/dashboard/static/built/components/agent-inference.js +0 -2
  274. claude_mpm/dashboard/static/built/components/agent-inference.js.map +0 -1
  275. claude_mpm/dashboard/static/built/components/build-tracker.js +0 -333
  276. claude_mpm/dashboard/static/built/components/code-simple.js +0 -857
  277. claude_mpm/dashboard/static/built/components/code-tree/tree-breadcrumb.js +0 -353
  278. claude_mpm/dashboard/static/built/components/code-tree/tree-constants.js +0 -235
  279. claude_mpm/dashboard/static/built/components/code-tree/tree-search.js +0 -409
  280. claude_mpm/dashboard/static/built/components/code-tree/tree-utils.js +0 -435
  281. claude_mpm/dashboard/static/built/components/code-tree.js +0 -2
  282. claude_mpm/dashboard/static/built/components/code-tree.js.map +0 -1
  283. claude_mpm/dashboard/static/built/components/code-viewer.js +0 -2
  284. claude_mpm/dashboard/static/built/components/code-viewer.js.map +0 -1
  285. claude_mpm/dashboard/static/built/components/connection-debug.js +0 -654
  286. claude_mpm/dashboard/static/built/components/diff-viewer.js +0 -891
  287. claude_mpm/dashboard/static/built/components/event-processor.js +0 -2
  288. claude_mpm/dashboard/static/built/components/event-processor.js.map +0 -1
  289. claude_mpm/dashboard/static/built/components/event-viewer.js +0 -2
  290. claude_mpm/dashboard/static/built/components/event-viewer.js.map +0 -1
  291. claude_mpm/dashboard/static/built/components/export-manager.js +0 -2
  292. claude_mpm/dashboard/static/built/components/export-manager.js.map +0 -1
  293. claude_mpm/dashboard/static/built/components/file-change-tracker.js +0 -443
  294. claude_mpm/dashboard/static/built/components/file-change-viewer.js +0 -690
  295. claude_mpm/dashboard/static/built/components/file-tool-tracker.js +0 -2
  296. claude_mpm/dashboard/static/built/components/file-tool-tracker.js.map +0 -1
  297. claude_mpm/dashboard/static/built/components/file-viewer.js +0 -2
  298. claude_mpm/dashboard/static/built/components/file-viewer.js.map +0 -1
  299. claude_mpm/dashboard/static/built/components/hud-library-loader.js +0 -2
  300. claude_mpm/dashboard/static/built/components/hud-library-loader.js.map +0 -1
  301. claude_mpm/dashboard/static/built/components/hud-manager.js +0 -2
  302. claude_mpm/dashboard/static/built/components/hud-manager.js.map +0 -1
  303. claude_mpm/dashboard/static/built/components/hud-visualizer.js +0 -2
  304. claude_mpm/dashboard/static/built/components/hud-visualizer.js.map +0 -1
  305. claude_mpm/dashboard/static/built/components/module-viewer.js +0 -2
  306. claude_mpm/dashboard/static/built/components/module-viewer.js.map +0 -1
  307. claude_mpm/dashboard/static/built/components/nav-bar.js +0 -145
  308. claude_mpm/dashboard/static/built/components/page-structure.js +0 -429
  309. claude_mpm/dashboard/static/built/components/session-manager.js +0 -2
  310. claude_mpm/dashboard/static/built/components/session-manager.js.map +0 -1
  311. claude_mpm/dashboard/static/built/components/socket-manager.js +0 -2
  312. claude_mpm/dashboard/static/built/components/socket-manager.js.map +0 -1
  313. claude_mpm/dashboard/static/built/components/ui-state-manager.js +0 -2
  314. claude_mpm/dashboard/static/built/components/ui-state-manager.js.map +0 -1
  315. claude_mpm/dashboard/static/built/components/unified-data-viewer.js +0 -2
  316. claude_mpm/dashboard/static/built/components/unified-data-viewer.js.map +0 -1
  317. claude_mpm/dashboard/static/built/components/working-directory.js +0 -2
  318. claude_mpm/dashboard/static/built/components/working-directory.js.map +0 -1
  319. claude_mpm/dashboard/static/built/connection-manager.js +0 -536
  320. claude_mpm/dashboard/static/built/dashboard.js +0 -2
  321. claude_mpm/dashboard/static/built/dashboard.js.map +0 -1
  322. claude_mpm/dashboard/static/built/extension-error-handler.js +0 -164
  323. claude_mpm/dashboard/static/built/react/events.js +0 -30
  324. claude_mpm/dashboard/static/built/react/events.js.map +0 -1
  325. claude_mpm/dashboard/static/built/shared/dom-helpers.js +0 -396
  326. claude_mpm/dashboard/static/built/shared/event-bus.js +0 -330
  327. claude_mpm/dashboard/static/built/shared/event-filter-service.js +0 -540
  328. claude_mpm/dashboard/static/built/shared/logger.js +0 -385
  329. claude_mpm/dashboard/static/built/shared/page-structure.js +0 -249
  330. claude_mpm/dashboard/static/built/shared/tooltip-service.js +0 -253
  331. claude_mpm/dashboard/static/built/socket-client.js +0 -2
  332. claude_mpm/dashboard/static/built/socket-client.js.map +0 -1
  333. claude_mpm/dashboard/static/built/tab-isolation-fix.js +0 -185
  334. claude_mpm/dashboard/static/events.html +0 -607
  335. claude_mpm/dashboard/static/index.html +0 -635
  336. claude_mpm/dashboard/static/js/REFACTORING_SUMMARY.md +0 -170
  337. claude_mpm/dashboard/static/js/shared/dom-helpers.js +0 -396
  338. claude_mpm/dashboard/static/js/shared/event-bus.js +0 -330
  339. claude_mpm/dashboard/static/js/shared/logger.js +0 -385
  340. claude_mpm/dashboard/static/js/shared/tooltip-service.js +0 -253
  341. claude_mpm/dashboard/static/js/stores/dashboard-store.js +0 -562
  342. claude_mpm/dashboard/static/legacy/activity.html +0 -736
  343. claude_mpm/dashboard/static/legacy/agents.html +0 -786
  344. claude_mpm/dashboard/static/legacy/files.html +0 -747
  345. claude_mpm/dashboard/static/legacy/tools.html +0 -831
  346. claude_mpm/dashboard/static/monitors.html +0 -431
  347. claude_mpm/dashboard/static/navigation-test-results.md +0 -118
  348. claude_mpm/dashboard/static/production/events.html +0 -659
  349. claude_mpm/dashboard/static/production/main.html +0 -698
  350. claude_mpm/dashboard/static/production/monitors.html +0 -483
  351. claude_mpm/dashboard/static/test-archive/dashboard.html +0 -635
  352. claude_mpm/dashboard/static/test-archive/debug-events.html +0 -147
  353. claude_mpm/dashboard/static/test-archive/test-navigation.html +0 -256
  354. claude_mpm/dashboard/static/test-archive/test-react-exports.html +0 -180
  355. claude_mpm/dashboard/templates/.claude-mpm/memories/README.md +0 -36
  356. claude_mpm/dashboard/templates/.claude-mpm/memories/engineer_agent.md +0 -39
  357. claude_mpm/dashboard/templates/.claude-mpm/memories/version_control_agent.md +0 -38
  358. claude_mpm/hooks/README.md +0 -143
  359. claude_mpm/hooks/templates/README.md +0 -180
  360. claude_mpm/hooks/templates/settings.json.example +0 -147
  361. claude_mpm/schemas/agent_schema.json +0 -596
  362. claude_mpm/schemas/frontmatter_schema.json +0 -165
  363. claude_mpm/services/event_bus/README.md +0 -244
  364. claude_mpm/services/events/README.md +0 -303
  365. claude_mpm/services/framework_claude_md_generator/README.md +0 -119
  366. claude_mpm/services/mcp_gateway/README.md +0 -185
  367. claude_mpm/services/socketio/handlers/connection.py.backup +0 -217
  368. claude_mpm/services/socketio/handlers/hook.py.backup +0 -154
  369. claude_mpm/services/static/.gitkeep +0 -2
  370. claude_mpm/services/version_control/VERSION +0 -1
  371. claude_mpm/skills/bundled/.gitkeep +0 -2
  372. claude_mpm/skills/bundled/collaboration/brainstorming/SKILL.md +0 -79
  373. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/SKILL.md +0 -178
  374. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/agent-prompts.md +0 -577
  375. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/coordination-patterns.md +0 -467
  376. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/examples.md +0 -537
  377. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/troubleshooting.md +0 -730
  378. claude_mpm/skills/bundled/collaboration/git-worktrees.md +0 -317
  379. claude_mpm/skills/bundled/collaboration/requesting-code-review/SKILL.md +0 -112
  380. claude_mpm/skills/bundled/collaboration/requesting-code-review/references/code-reviewer-template.md +0 -146
  381. claude_mpm/skills/bundled/collaboration/requesting-code-review/references/review-examples.md +0 -412
  382. claude_mpm/skills/bundled/collaboration/stacked-prs.md +0 -251
  383. claude_mpm/skills/bundled/collaboration/writing-plans/SKILL.md +0 -81
  384. claude_mpm/skills/bundled/collaboration/writing-plans/references/best-practices.md +0 -362
  385. claude_mpm/skills/bundled/collaboration/writing-plans/references/plan-structure-templates.md +0 -312
  386. claude_mpm/skills/bundled/debugging/root-cause-tracing/SKILL.md +0 -152
  387. claude_mpm/skills/bundled/debugging/root-cause-tracing/find-polluter.sh +0 -63
  388. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/advanced-techniques.md +0 -668
  389. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/examples.md +0 -587
  390. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/integration.md +0 -438
  391. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/tracing-techniques.md +0 -391
  392. claude_mpm/skills/bundled/debugging/systematic-debugging/CREATION-LOG.md +0 -119
  393. claude_mpm/skills/bundled/debugging/systematic-debugging/SKILL.md +0 -148
  394. claude_mpm/skills/bundled/debugging/systematic-debugging/references/anti-patterns.md +0 -483
  395. claude_mpm/skills/bundled/debugging/systematic-debugging/references/examples.md +0 -452
  396. claude_mpm/skills/bundled/debugging/systematic-debugging/references/troubleshooting.md +0 -449
  397. claude_mpm/skills/bundled/debugging/systematic-debugging/references/workflow.md +0 -411
  398. claude_mpm/skills/bundled/debugging/systematic-debugging/test-academic.md +0 -14
  399. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-1.md +0 -58
  400. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-2.md +0 -68
  401. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-3.md +0 -69
  402. claude_mpm/skills/bundled/debugging/verification-before-completion/SKILL.md +0 -131
  403. claude_mpm/skills/bundled/debugging/verification-before-completion/references/gate-function.md +0 -325
  404. claude_mpm/skills/bundled/debugging/verification-before-completion/references/integration-and-workflows.md +0 -490
  405. claude_mpm/skills/bundled/debugging/verification-before-completion/references/red-flags-and-failures.md +0 -425
  406. claude_mpm/skills/bundled/debugging/verification-before-completion/references/verification-patterns.md +0 -499
  407. claude_mpm/skills/bundled/infrastructure/env-manager/INTEGRATION.md +0 -611
  408. claude_mpm/skills/bundled/infrastructure/env-manager/README.md +0 -596
  409. claude_mpm/skills/bundled/infrastructure/env-manager/SKILL.md +0 -260
  410. claude_mpm/skills/bundled/infrastructure/env-manager/examples/nextjs-env-structure.md +0 -315
  411. claude_mpm/skills/bundled/infrastructure/env-manager/references/frameworks.md +0 -436
  412. claude_mpm/skills/bundled/infrastructure/env-manager/references/security.md +0 -433
  413. claude_mpm/skills/bundled/infrastructure/env-manager/references/synchronization.md +0 -452
  414. claude_mpm/skills/bundled/infrastructure/env-manager/references/troubleshooting.md +0 -404
  415. claude_mpm/skills/bundled/infrastructure/env-manager/references/validation.md +0 -420
  416. claude_mpm/skills/bundled/main/artifacts-builder/LICENSE.txt +0 -202
  417. claude_mpm/skills/bundled/main/artifacts-builder/SKILL.md +0 -86
  418. claude_mpm/skills/bundled/main/artifacts-builder/scripts/bundle-artifact.sh +0 -54
  419. claude_mpm/skills/bundled/main/artifacts-builder/scripts/init-artifact.sh +0 -322
  420. claude_mpm/skills/bundled/main/artifacts-builder/scripts/shadcn-components.tar.gz +0 -0
  421. claude_mpm/skills/bundled/main/internal-comms/LICENSE.txt +0 -202
  422. claude_mpm/skills/bundled/main/internal-comms/SKILL.md +0 -43
  423. claude_mpm/skills/bundled/main/internal-comms/examples/3p-updates.md +0 -47
  424. claude_mpm/skills/bundled/main/internal-comms/examples/company-newsletter.md +0 -65
  425. claude_mpm/skills/bundled/main/internal-comms/examples/faq-answers.md +0 -30
  426. claude_mpm/skills/bundled/main/internal-comms/examples/general-comms.md +0 -16
  427. claude_mpm/skills/bundled/main/mcp-builder/LICENSE.txt +0 -202
  428. claude_mpm/skills/bundled/main/mcp-builder/SKILL.md +0 -160
  429. claude_mpm/skills/bundled/main/mcp-builder/reference/design_principles.md +0 -412
  430. claude_mpm/skills/bundled/main/mcp-builder/reference/evaluation.md +0 -602
  431. claude_mpm/skills/bundled/main/mcp-builder/reference/mcp_best_practices.md +0 -915
  432. claude_mpm/skills/bundled/main/mcp-builder/reference/node_mcp_server.md +0 -916
  433. claude_mpm/skills/bundled/main/mcp-builder/reference/python_mcp_server.md +0 -752
  434. claude_mpm/skills/bundled/main/mcp-builder/reference/workflow.md +0 -1237
  435. claude_mpm/skills/bundled/main/mcp-builder/scripts/example_evaluation.xml +0 -22
  436. claude_mpm/skills/bundled/main/mcp-builder/scripts/requirements.txt +0 -2
  437. claude_mpm/skills/bundled/main/skill-creator/LICENSE.txt +0 -202
  438. claude_mpm/skills/bundled/main/skill-creator/SKILL.md +0 -189
  439. claude_mpm/skills/bundled/main/skill-creator/references/best-practices.md +0 -500
  440. claude_mpm/skills/bundled/main/skill-creator/references/creation-workflow.md +0 -464
  441. claude_mpm/skills/bundled/main/skill-creator/references/examples.md +0 -619
  442. claude_mpm/skills/bundled/main/skill-creator/references/progressive-disclosure.md +0 -437
  443. claude_mpm/skills/bundled/main/skill-creator/references/skill-structure.md +0 -231
  444. claude_mpm/skills/bundled/php/espocrm-development/SKILL.md +0 -170
  445. claude_mpm/skills/bundled/php/espocrm-development/references/architecture.md +0 -602
  446. claude_mpm/skills/bundled/php/espocrm-development/references/common-tasks.md +0 -821
  447. claude_mpm/skills/bundled/php/espocrm-development/references/development-workflow.md +0 -742
  448. claude_mpm/skills/bundled/php/espocrm-development/references/frontend-customization.md +0 -726
  449. claude_mpm/skills/bundled/php/espocrm-development/references/hooks-and-services.md +0 -764
  450. claude_mpm/skills/bundled/php/espocrm-development/references/testing-debugging.md +0 -831
  451. claude_mpm/skills/bundled/react/flexlayout-react.md +0 -742
  452. claude_mpm/skills/bundled/rust/desktop-applications/SKILL.md +0 -226
  453. claude_mpm/skills/bundled/rust/desktop-applications/references/architecture-patterns.md +0 -901
  454. claude_mpm/skills/bundled/rust/desktop-applications/references/native-gui-frameworks.md +0 -901
  455. claude_mpm/skills/bundled/rust/desktop-applications/references/platform-integration.md +0 -775
  456. claude_mpm/skills/bundled/rust/desktop-applications/references/state-management.md +0 -937
  457. claude_mpm/skills/bundled/rust/desktop-applications/references/tauri-framework.md +0 -770
  458. claude_mpm/skills/bundled/rust/desktop-applications/references/testing-deployment.md +0 -961
  459. claude_mpm/skills/bundled/tauri/tauri-async-patterns.md +0 -495
  460. claude_mpm/skills/bundled/tauri/tauri-build-deploy.md +0 -599
  461. claude_mpm/skills/bundled/tauri/tauri-command-patterns.md +0 -535
  462. claude_mpm/skills/bundled/tauri/tauri-error-handling.md +0 -613
  463. claude_mpm/skills/bundled/tauri/tauri-event-system.md +0 -648
  464. claude_mpm/skills/bundled/tauri/tauri-file-system.md +0 -673
  465. claude_mpm/skills/bundled/tauri/tauri-frontend-integration.md +0 -767
  466. claude_mpm/skills/bundled/tauri/tauri-performance.md +0 -669
  467. claude_mpm/skills/bundled/tauri/tauri-state-management.md +0 -573
  468. claude_mpm/skills/bundled/tauri/tauri-testing.md +0 -384
  469. claude_mpm/skills/bundled/tauri/tauri-window-management.md +0 -628
  470. claude_mpm/skills/bundled/testing/condition-based-waiting/SKILL.md +0 -119
  471. claude_mpm/skills/bundled/testing/condition-based-waiting/example.ts +0 -158
  472. claude_mpm/skills/bundled/testing/condition-based-waiting/references/patterns-and-implementation.md +0 -253
  473. claude_mpm/skills/bundled/testing/test-driven-development/SKILL.md +0 -145
  474. claude_mpm/skills/bundled/testing/test-driven-development/references/anti-patterns.md +0 -543
  475. claude_mpm/skills/bundled/testing/test-driven-development/references/examples.md +0 -741
  476. claude_mpm/skills/bundled/testing/test-driven-development/references/integration.md +0 -470
  477. claude_mpm/skills/bundled/testing/test-driven-development/references/philosophy.md +0 -458
  478. claude_mpm/skills/bundled/testing/test-driven-development/references/workflow.md +0 -639
  479. claude_mpm/skills/bundled/testing/test-quality-inspector/SKILL.md +0 -458
  480. claude_mpm/skills/bundled/testing/test-quality-inspector/examples/example-inspection-report.md +0 -411
  481. claude_mpm/skills/bundled/testing/test-quality-inspector/references/assertion-quality.md +0 -317
  482. claude_mpm/skills/bundled/testing/test-quality-inspector/references/inspection-checklist.md +0 -270
  483. claude_mpm/skills/bundled/testing/test-quality-inspector/references/red-flags.md +0 -436
  484. claude_mpm/skills/bundled/testing/testing-anti-patterns/SKILL.md +0 -140
  485. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/completeness-anti-patterns.md +0 -572
  486. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/core-anti-patterns.md +0 -411
  487. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/detection-guide.md +0 -569
  488. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/tdd-connection.md +0 -695
  489. claude_mpm/skills/bundled/testing/webapp-testing/LICENSE.txt +0 -202
  490. claude_mpm/skills/bundled/testing/webapp-testing/SKILL.md +0 -184
  491. claude_mpm/skills/bundled/testing/webapp-testing/decision-tree.md +0 -459
  492. claude_mpm/skills/bundled/testing/webapp-testing/playwright-patterns.md +0 -479
  493. claude_mpm/skills/bundled/testing/webapp-testing/reconnaissance-pattern.md +0 -687
  494. claude_mpm/skills/bundled/testing/webapp-testing/server-management.md +0 -758
  495. claude_mpm/skills/bundled/testing/webapp-testing/troubleshooting.md +0 -868
  496. claude_mpm/templates/questions/EXAMPLES.md +0 -501
  497. claude_mpm/tools/README_SOCKETIO_DEBUG.md +0 -224
  498. claude_mpm/tools/code_tree_analyzer/README.md +0 -64
  499. /claude_mpm/agents/templates/{git_file_tracking.md → git-file-tracking.md} +0 -0
  500. /claude_mpm/agents/templates/{pm_examples.md → pm-examples.md} +0 -0
  501. /claude_mpm/agents/templates/{pm_red_flags.md → pm-red-flags.md} +0 -0
  502. /claude_mpm/agents/templates/{response_format.md → response-format.md} +0 -0
  503. /claude_mpm/agents/templates/{validation_templates.md → validation-templates.md} +0 -0
  504. {claude_mpm-4.25.10.dist-info → claude_mpm-5.1.8.dist-info}/WHEEL +0 -0
  505. {claude_mpm-4.25.10.dist-info → claude_mpm-5.1.8.dist-info}/entry_points.txt +0 -0
  506. {claude_mpm-4.25.10.dist-info → claude_mpm-5.1.8.dist-info}/licenses/LICENSE +0 -0
  507. {claude_mpm-4.25.10.dist-info → claude_mpm-5.1.8.dist-info}/top_level.txt +0 -0
@@ -1,764 +0,0 @@
1
- # Hooks and Services Reference
2
-
3
- ## Service Layer Pattern
4
-
5
- The service layer is where ALL business logic belongs in EspoCRM. Never put business logic in hooks, controllers, or repositories.
6
-
7
- ### Service Layer Hierarchy
8
-
9
- ```
10
- Base Record Service (Espo\Services\Record)
11
-
12
- Your Custom Service (extends Record)
13
-
14
- Business Logic Methods
15
- ```
16
-
17
- ### Creating a Service
18
-
19
- ```php
20
- <?php
21
- namespace Espo\Modules\MyModule\Services;
22
-
23
- use Espo\Services\Record;
24
- use Espo\ORM\Entity;
25
- use Espo\Core\Exceptions\{BadRequest, Forbidden, NotFound};
26
- use Espo\Core\Mail\EmailSender;
27
- use Espo\Core\Utils\DateTime as DateTimeUtil;
28
-
29
- class Opportunity extends Record
30
- {
31
- public function __construct(
32
- private EmailSender $emailSender,
33
- private DateTimeUtil $dateTime
34
- ) {
35
- parent::__construct();
36
- }
37
-
38
- // Override create flow
39
- protected function beforeCreateEntity(Entity $entity, array $data): void
40
- {
41
- parent::beforeCreateEntity($entity, $data);
42
-
43
- // Business logic: Set expected close date to 30 days from now if not provided
44
- if (!$entity->get('closeDate')) {
45
- $closeDate = $this->dateTime->getDateTime()
46
- ->modify('+30 days')
47
- ->format('Y-m-d');
48
- $entity->set('closeDate', $closeDate);
49
- }
50
-
51
- // Business logic: Auto-assign to manager for large opportunities
52
- if ($entity->get('amount') > 100000 && !$entity->get('assignedUserId')) {
53
- $managerId = $this->getManagerForLargeDeals();
54
- $entity->set('assignedUserId', $managerId);
55
- }
56
- }
57
-
58
- // Override update flow
59
- protected function beforeUpdateEntity(Entity $entity, array $data): void
60
- {
61
- parent::beforeUpdateEntity($entity, $data);
62
-
63
- // Business logic: Track stage changes
64
- if ($entity->isAttributeChanged('stage')) {
65
- $this->trackStageChange($entity);
66
- }
67
- }
68
-
69
- // Custom business logic method
70
- public function markAsWon(string $id): Entity
71
- {
72
- $entity = $this->getEntity($id);
73
-
74
- if (!$entity) {
75
- throw new NotFound();
76
- }
77
-
78
- if (!$this->acl->check($entity, 'edit')) {
79
- throw new Forbidden();
80
- }
81
-
82
- // Validate can be marked as won
83
- if (!$this->canMarkAsWon($entity)) {
84
- throw new BadRequest('Opportunity cannot be marked as won in current state');
85
- }
86
-
87
- // Update entity
88
- $entity->set([
89
- 'stage' => 'Closed Won',
90
- 'probability' => 100,
91
- 'closeDate' => date('Y-m-d')
92
- ]);
93
-
94
- $this->entityManager->saveEntity($entity);
95
-
96
- // Additional business logic
97
- $this->createWinNotification($entity);
98
- $this->updateAccountRevenue($entity);
99
-
100
- return $entity;
101
- }
102
-
103
- private function canMarkAsWon(Entity $opportunity): bool
104
- {
105
- // Business rules for winning
106
- $stage = $opportunity->get('stage');
107
- $allowedStages = ['Proposal', 'Negotiation'];
108
-
109
- return in_array($stage, $allowedStages);
110
- }
111
-
112
- private function createWinNotification(Entity $opportunity): void
113
- {
114
- $assignedUserId = $opportunity->get('assignedUserId');
115
-
116
- if (!$assignedUserId) {
117
- return;
118
- }
119
-
120
- // Send email notification
121
- $emailSender = $this->emailSender->create();
122
-
123
- $emailSender
124
- ->withSubject('Opportunity Won: ' . $opportunity->get('name'))
125
- ->withBody('Congratulations! Opportunity has been marked as won.')
126
- ->withToUserIdList([$assignedUserId])
127
- ->send();
128
- }
129
-
130
- private function updateAccountRevenue(Entity $opportunity): void
131
- {
132
- $accountId = $opportunity->get('accountId');
133
-
134
- if (!$accountId) {
135
- return;
136
- }
137
-
138
- $account = $this->entityManager->getEntityById('Account', $accountId);
139
-
140
- if (!$account) {
141
- return;
142
- }
143
-
144
- // Calculate total won opportunities
145
- $totalRevenue = $this->entityManager
146
- ->getRDBRepository('Opportunity')
147
- ->where([
148
- 'accountId' => $accountId,
149
- 'stage' => 'Closed Won'
150
- ])
151
- ->select(['SUM:amount'])
152
- ->findOne()
153
- ->get('SUM:amount') ?? 0;
154
-
155
- $account->set('totalRevenue', $totalRevenue);
156
- $this->entityManager->saveEntity($account);
157
- }
158
-
159
- private function trackStageChange(Entity $opportunity): void
160
- {
161
- $note = $this->entityManager->getNewEntity('Note');
162
- $note->set([
163
- 'type' => 'Update',
164
- 'parentType' => 'Opportunity',
165
- 'parentId' => $opportunity->getId(),
166
- 'data' => [
167
- 'fields' => ['stage'],
168
- 'attributes' => [
169
- 'stage' => [
170
- 'was' => $opportunity->getFetched('stage'),
171
- 'became' => $opportunity->get('stage')
172
- ]
173
- ]
174
- ]
175
- ]);
176
-
177
- $this->entityManager->saveEntity($note);
178
- }
179
-
180
- private function getManagerForLargeDeals(): ?string
181
- {
182
- // Get sales manager role user
183
- $manager = $this->entityManager
184
- ->getRDBRepository('User')
185
- ->join('teams')
186
- ->where([
187
- 'teams.name' => 'Sales Management',
188
- 'isActive' => true
189
- ])
190
- ->findOne();
191
-
192
- return $manager?->getId();
193
- }
194
- }
195
- ```
196
-
197
- ### Record Service Hook Points
198
-
199
- Override these methods to inject custom logic into the standard CRUD flow:
200
-
201
- ```php
202
- // Before operations
203
- protected function beforeCreateEntity(Entity $entity, array $data): void
204
- protected function beforeUpdateEntity(Entity $entity, array $data): void
205
- protected function beforeDeleteEntity(Entity $entity): void
206
-
207
- // After operations
208
- protected function afterCreateEntity(Entity $entity, array $data): void
209
- protected function afterUpdateEntity(Entity $entity, array $data): void
210
- protected function afterDeleteEntity(Entity $entity): void
211
-
212
- // Link operations
213
- protected function beforeLink(Entity $entity, string $link, Entity $foreign): void
214
- protected function afterLink(Entity $entity, string $link, Entity $foreign): void
215
- protected function beforeUnlink(Entity $entity, string $link, Entity $foreign): void
216
- protected function afterUnlink(Entity $entity, string $link, Entity $foreign): void
217
- ```
218
-
219
- ## Hook System
220
-
221
- Hooks are for **validation and side effects ONLY**, not business logic. Business logic belongs in Services.
222
-
223
- ### The 7 Hook Types
224
-
225
- #### 1. BeforeSave - Validation
226
-
227
- ```php
228
- <?php
229
- namespace Espo\Modules\MyModule\Hooks\Account;
230
-
231
- use Espo\ORM\Entity;
232
- use Espo\Core\Hook\Hook\BeforeSave;
233
- use Espo\Core\Exceptions\BadRequest;
234
-
235
- class ValidateData implements BeforeSave
236
- {
237
- public function beforeSave(Entity $entity, array $options): void
238
- {
239
- // Validation: Check phone number format
240
- if ($entity->isAttributeChanged('phoneNumber')) {
241
- $phone = $entity->get('phoneNumber');
242
-
243
- if ($phone && !$this->isValidPhone($phone)) {
244
- throw new BadRequest('Invalid phone number format');
245
- }
246
- }
247
-
248
- // Validation: Ensure website starts with https
249
- if ($entity->isAttributeChanged('website')) {
250
- $website = $entity->get('website');
251
-
252
- if ($website && !str_starts_with($website, 'https://')) {
253
- $entity->set('website', 'https://' . ltrim($website, 'http://'));
254
- }
255
- }
256
-
257
- // Validation: Business rule
258
- if ($entity->get('type') === 'Customer' && !$entity->get('industry')) {
259
- throw new BadRequest('Industry is required for Customer accounts');
260
- }
261
- }
262
-
263
- private function isValidPhone(string $phone): bool
264
- {
265
- return preg_match('/^\+?[0-9\s\-\(\)]+$/', $phone);
266
- }
267
- }
268
- ```
269
-
270
- #### 2. AfterSave - Side Effects
271
-
272
- ```php
273
- <?php
274
- namespace Espo\Modules\MyModule\Hooks\Opportunity;
275
-
276
- use Espo\ORM\Entity;
277
- use Espo\Core\Hook\Hook\AfterSave;
278
- use Espo\ORM\EntityManager;
279
-
280
- class UpdateAccountStats implements AfterSave
281
- {
282
- public function __construct(private EntityManager $entityManager) {}
283
-
284
- public function afterSave(Entity $entity, array $options): void
285
- {
286
- // Side effect: Update account statistics when opportunity stage changes
287
- if ($entity->isAttributeChanged('stage')) {
288
- $accountId = $entity->get('accountId');
289
-
290
- if ($accountId) {
291
- $this->updateAccountOpportunityStats($accountId);
292
- }
293
- }
294
-
295
- // Side effect: Create activity when opportunity is won
296
- if ($entity->isAttributeChanged('stage') && $entity->get('stage') === 'Closed Won') {
297
- $this->createWonActivity($entity);
298
- }
299
- }
300
-
301
- private function updateAccountOpportunityStats(string $accountId): void
302
- {
303
- $account = $this->entityManager->getEntityById('Account', $accountId);
304
-
305
- if (!$account) {
306
- return;
307
- }
308
-
309
- // Count opportunities
310
- $openCount = $this->entityManager
311
- ->getRDBRepository('Opportunity')
312
- ->where([
313
- 'accountId' => $accountId,
314
- 'stage!=' => ['Closed Won', 'Closed Lost']
315
- ])
316
- ->count();
317
-
318
- $account->set('openOpportunitiesCount', $openCount);
319
- $this->entityManager->saveEntity($account);
320
- }
321
-
322
- private function createWonActivity(Entity $opportunity): void
323
- {
324
- $meeting = $this->entityManager->getNewEntity('Meeting');
325
- $meeting->set([
326
- 'name' => 'Follow-up: ' . $opportunity->get('name'),
327
- 'parentType' => 'Opportunity',
328
- 'parentId' => $opportunity->getId(),
329
- 'assignedUserId' => $opportunity->get('assignedUserId'),
330
- 'status' => 'Planned',
331
- 'dateStart' => date('Y-m-d H:i:s', strtotime('+1 week'))
332
- ]);
333
-
334
- $this->entityManager->saveEntity($meeting);
335
- }
336
- }
337
- ```
338
-
339
- #### 3. BeforeRemove - Pre-deletion Validation
340
-
341
- ```php
342
- <?php
343
- namespace Espo\Modules\MyModule\Hooks\Account;
344
-
345
- use Espo\ORM\Entity;
346
- use Espo\Core\Hook\Hook\BeforeRemove;
347
- use Espo\Core\Exceptions\Forbidden;
348
- use Espo\ORM\EntityManager;
349
-
350
- class PreventDeletionWithOpenOpportunities implements BeforeRemove
351
- {
352
- public function __construct(private EntityManager $entityManager) {}
353
-
354
- public function beforeRemove(Entity $entity, array $options): void
355
- {
356
- // Validation: Prevent deletion if account has open opportunities
357
- $hasOpenOpportunities = $this->entityManager
358
- ->getRDBRepository('Opportunity')
359
- ->where([
360
- 'accountId' => $entity->getId(),
361
- 'stage!=' => ['Closed Won', 'Closed Lost']
362
- ])
363
- ->count() > 0;
364
-
365
- if ($hasOpenOpportunities) {
366
- throw new Forbidden('Cannot delete account with open opportunities');
367
- }
368
- }
369
- }
370
- ```
371
-
372
- #### 4. AfterRemove - Post-deletion Cleanup
373
-
374
- ```php
375
- <?php
376
- namespace Espo\Modules\MyModule\Hooks\Account;
377
-
378
- use Espo\ORM\Entity;
379
- use Espo\Core\Hook\Hook\AfterRemove;
380
- use Espo\ORM\EntityManager;
381
-
382
- class CleanupRelatedData implements AfterRemove
383
- {
384
- public function __construct(private EntityManager $entityManager) {}
385
-
386
- public function afterRemove(Entity $entity, array $options): void
387
- {
388
- // Cleanup: Remove orphaned custom records
389
- $customRecords = $this->entityManager
390
- ->getRDBRepository('CustomEntity')
391
- ->where(['accountId' => $entity->getId()])
392
- ->find();
393
-
394
- foreach ($customRecords as $record) {
395
- $this->entityManager->removeEntity($record);
396
- }
397
- }
398
- }
399
- ```
400
-
401
- #### 5. AfterRelate - React to Relationship Creation
402
-
403
- ```php
404
- <?php
405
- namespace Espo\Modules\MyModule\Hooks\Contact;
406
-
407
- use Espo\ORM\Entity;
408
- use Espo\Core\Hook\Hook\AfterRelate;
409
- use Espo\ORM\EntityManager;
410
-
411
- class UpdateAccountContacts implements AfterRelate
412
- {
413
- public function __construct(private EntityManager $entityManager) {}
414
-
415
- public function afterRelate(
416
- Entity $entity,
417
- string $relationName,
418
- Entity $foreign,
419
- ?array $columnData,
420
- array $options
421
- ): void {
422
- // React to contact being linked to account
423
- if ($relationName === 'account') {
424
- $account = $foreign;
425
-
426
- // Update account's primary contact if not set
427
- if (!$account->get('primaryContactId')) {
428
- $account->set('primaryContactId', $entity->getId());
429
- $this->entityManager->saveEntity($account);
430
- }
431
- }
432
- }
433
- }
434
- ```
435
-
436
- #### 6. AfterUnrelate - React to Relationship Removal
437
-
438
- ```php
439
- <?php
440
- namespace Espo\Modules\MyModule\Hooks\Contact;
441
-
442
- use Espo\ORM\Entity;
443
- use Espo\Core\Hook\Hook\AfterUnrelate;
444
- use Espo\ORM\EntityManager;
445
-
446
- class UpdateAccountOnUnlink implements AfterUnrelate
447
- {
448
- public function __construct(private EntityManager $entityManager) {}
449
-
450
- public function afterUnrelate(
451
- Entity $entity,
452
- string $relationName,
453
- Entity $foreign,
454
- array $options
455
- ): void {
456
- // React to contact being unlinked from account
457
- if ($relationName === 'account') {
458
- $account = $foreign;
459
-
460
- // Clear primary contact if it was this contact
461
- if ($account->get('primaryContactId') === $entity->getId()) {
462
- $account->set('primaryContactId', null);
463
- $this->entityManager->saveEntity($account);
464
- }
465
- }
466
- }
467
- }
468
- ```
469
-
470
- #### 7. AfterMassRelate - React to Bulk Relationship Operations
471
-
472
- ```php
473
- <?php
474
- namespace Espo\Modules\MyModule\Hooks\Contact;
475
-
476
- use Espo\ORM\Entity;
477
- use Espo\Core\Hook\Hook\AfterMassRelate;
478
- use Espo\ORM\EntityManager;
479
-
480
- class RecalculateAccountStats implements AfterMassRelate
481
- {
482
- public function __construct(private EntityManager $entityManager) {}
483
-
484
- public function afterMassRelate(
485
- Entity $entity,
486
- string $relationName,
487
- array $params,
488
- array $options
489
- ): void {
490
- // React to mass relate operation
491
- if ($relationName === 'accounts') {
492
- // Recalculate statistics for all affected accounts
493
- $this->recalculateStats($entity->getId());
494
- }
495
- }
496
-
497
- private function recalculateStats(string $contactId): void
498
- {
499
- // Implementation
500
- }
501
- }
502
- ```
503
-
504
- ## Dependency Injection in Hooks
505
-
506
- ### Constructor Injection (CORRECT)
507
-
508
- ```php
509
- <?php
510
- namespace Espo\Modules\MyModule\Hooks\Account;
511
-
512
- use Espo\ORM\Entity;
513
- use Espo\Core\Hook\Hook\BeforeSave;
514
- use Espo\ORM\EntityManager;
515
- use Espo\Core\Utils\Metadata;
516
- use Espo\Core\Mail\EmailSender;
517
- use Espo\Core\ServiceFactory;
518
-
519
- class MyHook implements BeforeSave
520
- {
521
- public function __construct(
522
- private EntityManager $entityManager,
523
- private Metadata $metadata,
524
- private EmailSender $emailSender,
525
- private ServiceFactory $serviceFactory
526
- ) {}
527
-
528
- public function beforeSave(Entity $entity, array $options): void
529
- {
530
- // Use injected dependencies
531
- $config = $this->metadata->get(['app', 'myConfig']);
532
-
533
- // Access services
534
- $service = $this->serviceFactory->create('Account');
535
- }
536
- }
537
- ```
538
-
539
- ### NEVER Pass Container
540
-
541
- ```php
542
- // ❌ WRONG - Never inject Container
543
- use Espo\Core\Container;
544
-
545
- class BadHook implements BeforeSave
546
- {
547
- public function __construct(private Container $container) {}
548
- }
549
-
550
- // ✅ CORRECT - Inject specific dependencies
551
- use Espo\ORM\EntityManager;
552
-
553
- class GoodHook implements BeforeSave
554
- {
555
- public function __construct(private EntityManager $entityManager) {}
556
- }
557
- ```
558
-
559
- ## Transaction Handling
560
-
561
- Use TransactionManager for operations that must be atomic:
562
-
563
- ```php
564
- <?php
565
- namespace Espo\Modules\MyModule\Services;
566
-
567
- use Espo\Services\Record;
568
- use Espo\ORM\TransactionManager;
569
-
570
- class MyEntity extends Record
571
- {
572
- public function __construct(private TransactionManager $transactionManager)
573
- {
574
- parent::__construct();
575
- }
576
-
577
- public function complexOperation(string $id): void
578
- {
579
- $this->transactionManager->run(function () use ($id) {
580
- // All operations in this closure are transactional
581
- $entity = $this->getEntity($id);
582
- $entity->set('status', 'Processing');
583
- $this->entityManager->saveEntity($entity);
584
-
585
- // Related operation
586
- $relatedEntity = $this->entityManager->getNewEntity('RelatedEntity');
587
- $relatedEntity->set('parentId', $id);
588
- $this->entityManager->saveEntity($relatedEntity);
589
-
590
- // If any exception is thrown, ALL changes are rolled back
591
- });
592
- }
593
- }
594
- ```
595
-
596
- ## Formula Scripts - Declarative Logic
597
-
598
- Use Formula scripts for simple field calculations instead of hooks:
599
-
600
- ### Formula in Entity Metadata
601
-
602
- ```json
603
- {
604
- "fields": {
605
- "totalPrice": {
606
- "type": "currency",
607
- "formula": "quantity * unitPrice"
608
- },
609
- "fullName": {
610
- "type": "varchar",
611
- "formula": "string\\concatenate(firstName, ' ', lastName)"
612
- },
613
- "daysUntilDue": {
614
- "type": "int",
615
- "formula": "datetime\\diff(datetime\\today(), dueDate, 'days')"
616
- }
617
- }
618
- }
619
- ```
620
-
621
- ### Formula vs. Hooks Decision Matrix
622
-
623
- | Use Case | Solution | Reason |
624
- |----------|----------|--------|
625
- | Calculate field from other fields | Formula | Simple, declarative |
626
- | String concatenation | Formula | Built-in functions |
627
- | Conditional field values | Formula | If/then logic available |
628
- | Date calculations | Formula | Date functions available |
629
- | Complex business rules | Service | Needs multiple entities |
630
- | External API calls | Service | Needs async/error handling |
631
- | Multi-entity operations | Service | Transaction support |
632
- | Validation requiring DB query | Hook (BeforeSave) | Needs EntityManager |
633
-
634
- ### Formula Functions Reference
635
-
636
- **String Functions:**
637
- ```javascript
638
- string\concatenate(firstName, ' ', lastName)
639
- string\substring(name, 0, 10)
640
- string\length(description)
641
- string\trim(input)
642
- string\lowerCase(email)
643
- string\upperCase(name)
644
- ```
645
-
646
- **Date Functions:**
647
- ```javascript
648
- datetime\today()
649
- datetime\now()
650
- datetime\diff(date1, date2, 'days')
651
- datetime\addDays(dateStart, 5)
652
- datetime\month(dateStart)
653
- datetime\year(dateStart)
654
- ```
655
-
656
- **Numeric Functions:**
657
- ```javascript
658
- number\round(amount, 2)
659
- number\floor(value)
660
- number\ceil(value)
661
- number\abs(value)
662
- ```
663
-
664
- **Conditional Logic:**
665
- ```javascript
666
- ifThen(
667
- status == 'Complete',
668
- 100,
669
- probability
670
- )
671
-
672
- ifThenElse(
673
- amount > 10000,
674
- 'High',
675
- 'Normal'
676
- )
677
- ```
678
-
679
- ## Best Practices
680
-
681
- ### 1. Business Logic Placement
682
-
683
- ```php
684
- // ✅ CORRECT - Business logic in Service
685
- class OpportunityService extends Record {
686
- public function calculateCommission(string $id): float {
687
- $opportunity = $this->getEntity($id);
688
- $amount = $opportunity->get('amount');
689
- $stage = $opportunity->get('stage');
690
-
691
- return $this->getCommissionCalculator()->calculate($amount, $stage);
692
- }
693
- }
694
-
695
- // ❌ WRONG - Business logic in Hook
696
- class OpportunityHook implements AfterSave {
697
- public function afterSave(Entity $entity, array $options): void {
698
- // Don't put complex business logic here
699
- $commission = $this->calculateCommission($entity);
700
- }
701
- }
702
- ```
703
-
704
- ### 2. Hook Complexity
705
-
706
- ```php
707
- // ✅ CORRECT - Simple side effects in hooks
708
- class NotificationHook implements AfterSave {
709
- public function afterSave(Entity $entity, array $options): void {
710
- if ($entity->isAttributeChanged('assignedUserId')) {
711
- $this->sendNotification($entity);
712
- }
713
- }
714
- }
715
-
716
- // ❌ WRONG - Complex logic in hooks
717
- class ComplexHook implements AfterSave {
718
- public function afterSave(Entity $entity, array $options): void {
719
- // Too much logic - belongs in service
720
- $this->updateRelatedRecords($entity);
721
- $this->recalculateMetrics($entity);
722
- $this->syncWithExternalSystem($entity);
723
- $this->generateReports($entity);
724
- }
725
- }
726
- ```
727
-
728
- ### 3. Avoid Recursive Hook Calls
729
-
730
- ```php
731
- // ✅ CORRECT - Prevent recursive calls
732
- class MyHook implements AfterSave {
733
- public function afterSave(Entity $entity, array $options): void {
734
- // Check if this is a programmatic save to avoid recursion
735
- if (!empty($options['silent'])) {
736
- return;
737
- }
738
-
739
- // Make changes to related entity
740
- $related = $this->entityManager->getEntityById('Related', $entity->get('relatedId'));
741
- $related->set('updated', true);
742
-
743
- // Save with 'silent' option to prevent triggering hooks
744
- $this->entityManager->saveEntity($related, ['silent' => true]);
745
- }
746
- }
747
- ```
748
-
749
- ### 4. Type Safety
750
-
751
- ```php
752
- // ✅ CORRECT - Strict types
753
- declare(strict_types=1);
754
-
755
- class MyHook implements BeforeSave {
756
- public function beforeSave(Entity $entity, array $options): void {
757
- $value = $entity->get('amount');
758
-
759
- if (!is_numeric($value)) {
760
- throw new BadRequest('Amount must be numeric');
761
- }
762
- }
763
- }
764
- ```