claude-mpm 4.21.3__py3-none-any.whl → 5.1.9__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 (517) hide show
  1. claude_mpm/VERSION +1 -1
  2. claude_mpm/agents/BASE_PM.md +12 -0
  3. claude_mpm/agents/{OUTPUT_STYLE.md → CLAUDE_MPM_OUTPUT_STYLE.md} +3 -48
  4. claude_mpm/agents/CLAUDE_MPM_TEACHER_OUTPUT_STYLE.md +2002 -0
  5. claude_mpm/agents/PM_INSTRUCTIONS.md +1239 -674
  6. claude_mpm/agents/WORKFLOW.md +75 -2
  7. claude_mpm/agents/__init__.py +6 -0
  8. claude_mpm/agents/agent_loader.py +1 -4
  9. claude_mpm/agents/base_agent.json +6 -3
  10. claude_mpm/agents/base_agent_loader.py +10 -35
  11. claude_mpm/agents/frontmatter_validator.py +69 -1
  12. claude_mpm/agents/templates/circuit-breakers.md +1254 -0
  13. claude_mpm/agents/templates/context-management-examples.md +544 -0
  14. claude_mpm/agents/templates/{pm_red_flags.md → pm-red-flags.md} +89 -19
  15. claude_mpm/agents/templates/pr-workflow-examples.md +427 -0
  16. claude_mpm/agents/templates/research-gate-examples.md +669 -0
  17. claude_mpm/agents/templates/structured-questions-examples.md +615 -0
  18. claude_mpm/agents/templates/ticket-completeness-examples.md +139 -0
  19. claude_mpm/agents/templates/ticketing-examples.md +277 -0
  20. claude_mpm/cli/__init__.py +37 -2
  21. claude_mpm/cli/commands/__init__.py +2 -0
  22. claude_mpm/cli/commands/agent_source.py +774 -0
  23. claude_mpm/cli/commands/agent_state_manager.py +188 -30
  24. claude_mpm/cli/commands/agents.py +1128 -36
  25. claude_mpm/cli/commands/agents_cleanup.py +210 -0
  26. claude_mpm/cli/commands/agents_discover.py +338 -0
  27. claude_mpm/cli/commands/aggregate.py +1 -1
  28. claude_mpm/cli/commands/analyze.py +3 -3
  29. claude_mpm/cli/commands/auto_configure.py +537 -239
  30. claude_mpm/cli/commands/cleanup.py +1 -1
  31. claude_mpm/cli/commands/config.py +7 -4
  32. claude_mpm/cli/commands/configure.py +935 -45
  33. claude_mpm/cli/commands/configure_agent_display.py +4 -4
  34. claude_mpm/cli/commands/configure_navigation.py +63 -46
  35. claude_mpm/cli/commands/debug.py +12 -12
  36. claude_mpm/cli/commands/doctor.py +10 -2
  37. claude_mpm/cli/commands/hook_errors.py +277 -0
  38. claude_mpm/cli/commands/local_deploy.py +1 -4
  39. claude_mpm/cli/commands/mcp_install_commands.py +1 -1
  40. claude_mpm/cli/commands/mpm_init/core.py +50 -2
  41. claude_mpm/cli/commands/mpm_init/git_activity.py +10 -10
  42. claude_mpm/cli/commands/mpm_init/prompts.py +6 -6
  43. claude_mpm/cli/commands/postmortem.py +401 -0
  44. claude_mpm/cli/commands/run.py +125 -167
  45. claude_mpm/cli/commands/skill_source.py +694 -0
  46. claude_mpm/cli/commands/skills.py +757 -20
  47. claude_mpm/cli/executor.py +78 -3
  48. claude_mpm/cli/interactive/agent_wizard.py +1032 -47
  49. claude_mpm/cli/parsers/agent_source_parser.py +171 -0
  50. claude_mpm/cli/parsers/agents_parser.py +310 -4
  51. claude_mpm/cli/parsers/auto_configure_parser.py +13 -0
  52. claude_mpm/cli/parsers/base_parser.py +53 -0
  53. claude_mpm/cli/parsers/config_parser.py +96 -43
  54. claude_mpm/cli/parsers/skill_source_parser.py +169 -0
  55. claude_mpm/cli/parsers/skills_parser.py +145 -0
  56. claude_mpm/cli/parsers/source_parser.py +138 -0
  57. claude_mpm/cli/startup.py +564 -108
  58. claude_mpm/cli/startup_display.py +480 -0
  59. claude_mpm/cli/utils.py +1 -1
  60. claude_mpm/cli_module/commands.py +1 -1
  61. claude_mpm/commands/{mpm-auto-configure.md → mpm-agents-auto-configure.md} +9 -0
  62. claude_mpm/commands/mpm-agents-detect.md +9 -0
  63. claude_mpm/commands/{mpm-agents.md → mpm-agents-list.md} +9 -0
  64. claude_mpm/commands/mpm-agents-recommend.md +9 -0
  65. claude_mpm/commands/{mpm-config.md → mpm-config-view.md} +9 -0
  66. claude_mpm/commands/mpm-doctor.md +9 -0
  67. claude_mpm/commands/mpm-help.md +14 -2
  68. claude_mpm/commands/mpm-init.md +27 -2
  69. claude_mpm/commands/mpm-monitor.md +9 -0
  70. claude_mpm/commands/mpm-postmortem.md +123 -0
  71. claude_mpm/commands/{mpm-resume.md → mpm-session-resume.md} +9 -0
  72. claude_mpm/commands/mpm-status.md +9 -0
  73. claude_mpm/commands/{mpm-organize.md → mpm-ticket-organize.md} +9 -0
  74. claude_mpm/commands/mpm-ticket-view.md +552 -0
  75. claude_mpm/commands/mpm-version.md +9 -0
  76. claude_mpm/commands/mpm.md +10 -0
  77. claude_mpm/config/agent_presets.py +488 -0
  78. claude_mpm/config/agent_sources.py +325 -0
  79. claude_mpm/config/skill_presets.py +392 -0
  80. claude_mpm/config/skill_sources.py +590 -0
  81. claude_mpm/constants.py +13 -0
  82. claude_mpm/core/api_validator.py +1 -1
  83. claude_mpm/core/claude_runner.py +19 -35
  84. claude_mpm/core/config.py +24 -0
  85. claude_mpm/core/constants.py +1 -1
  86. claude_mpm/core/framework/__init__.py +3 -16
  87. claude_mpm/core/framework/loaders/file_loader.py +54 -101
  88. claude_mpm/core/framework/loaders/instruction_loader.py +25 -5
  89. claude_mpm/core/framework/processors/metadata_processor.py +1 -1
  90. claude_mpm/core/hook_error_memory.py +381 -0
  91. claude_mpm/core/hook_manager.py +41 -2
  92. claude_mpm/core/interactive_session.py +131 -10
  93. claude_mpm/core/logger.py +3 -1
  94. claude_mpm/core/oneshot_session.py +110 -8
  95. claude_mpm/core/output_style_manager.py +173 -43
  96. claude_mpm/core/protocols/__init__.py +23 -0
  97. claude_mpm/core/protocols/runner_protocol.py +103 -0
  98. claude_mpm/core/protocols/session_protocol.py +131 -0
  99. claude_mpm/core/shared/singleton_manager.py +11 -4
  100. claude_mpm/core/system_context.py +38 -0
  101. claude_mpm/core/unified_agent_registry.py +129 -1
  102. claude_mpm/core/unified_config.py +22 -0
  103. claude_mpm/dashboard/static/css/activity.css +69 -69
  104. claude_mpm/dashboard/static/css/connection-status.css +10 -10
  105. claude_mpm/dashboard/static/css/dashboard.css +15 -15
  106. claude_mpm/dashboard/static/js/components/activity-tree.js +178 -178
  107. claude_mpm/dashboard/static/js/components/agent-hierarchy.js +101 -101
  108. claude_mpm/dashboard/static/js/components/agent-inference.js +31 -31
  109. claude_mpm/dashboard/static/js/components/build-tracker.js +59 -59
  110. claude_mpm/dashboard/static/js/components/code-simple.js +107 -107
  111. claude_mpm/dashboard/static/js/components/connection-debug.js +101 -101
  112. claude_mpm/dashboard/static/js/components/diff-viewer.js +113 -113
  113. claude_mpm/dashboard/static/js/components/event-viewer.js +12 -12
  114. claude_mpm/dashboard/static/js/components/file-change-tracker.js +57 -57
  115. claude_mpm/dashboard/static/js/components/file-change-viewer.js +74 -74
  116. claude_mpm/dashboard/static/js/components/file-tool-tracker.js +6 -6
  117. claude_mpm/dashboard/static/js/components/file-viewer.js +42 -42
  118. claude_mpm/dashboard/static/js/components/module-viewer.js +27 -27
  119. claude_mpm/dashboard/static/js/components/session-manager.js +14 -14
  120. claude_mpm/dashboard/static/js/components/socket-manager.js +1 -1
  121. claude_mpm/dashboard/static/js/components/ui-state-manager.js +14 -14
  122. claude_mpm/dashboard/static/js/components/unified-data-viewer.js +110 -110
  123. claude_mpm/dashboard/static/js/components/working-directory.js +8 -8
  124. claude_mpm/dashboard/static/js/connection-manager.js +76 -76
  125. claude_mpm/dashboard/static/js/dashboard.js +76 -58
  126. claude_mpm/dashboard/static/js/extension-error-handler.js +22 -22
  127. claude_mpm/dashboard/static/js/socket-client.js +138 -121
  128. claude_mpm/dashboard/templates/code_simple.html +23 -23
  129. claude_mpm/dashboard/templates/index.html +18 -18
  130. claude_mpm/experimental/cli_enhancements.py +1 -5
  131. claude_mpm/hooks/claude_hooks/event_handlers.py +3 -1
  132. claude_mpm/hooks/claude_hooks/hook_handler.py +24 -7
  133. claude_mpm/hooks/claude_hooks/installer.py +45 -0
  134. claude_mpm/hooks/claude_hooks/memory_integration.py +12 -1
  135. claude_mpm/hooks/failure_learning/__init__.py +2 -8
  136. claude_mpm/hooks/failure_learning/failure_detection_hook.py +1 -6
  137. claude_mpm/hooks/failure_learning/fix_detection_hook.py +1 -6
  138. claude_mpm/hooks/failure_learning/learning_extraction_hook.py +1 -6
  139. claude_mpm/hooks/kuzu_response_hook.py +1 -5
  140. claude_mpm/hooks/templates/pre_tool_use_simple.py +78 -0
  141. claude_mpm/hooks/templates/pre_tool_use_template.py +323 -0
  142. claude_mpm/models/agent_definition.py +7 -0
  143. claude_mpm/models/git_repository.py +198 -0
  144. claude_mpm/scripts/claude-hook-handler.sh +3 -3
  145. claude_mpm/scripts/start_activity_logging.py +3 -1
  146. claude_mpm/services/agents/agent_builder.py +45 -9
  147. claude_mpm/services/agents/agent_preset_service.py +238 -0
  148. claude_mpm/services/agents/agent_selection_service.py +484 -0
  149. claude_mpm/services/agents/auto_deploy_index_parser.py +569 -0
  150. claude_mpm/services/agents/cache_git_manager.py +621 -0
  151. claude_mpm/services/agents/deployment/agent_deployment.py +126 -2
  152. claude_mpm/services/agents/deployment/agent_discovery_service.py +105 -73
  153. claude_mpm/services/agents/deployment/agent_format_converter.py +1 -1
  154. claude_mpm/services/agents/deployment/agent_lifecycle_manager.py +1 -5
  155. claude_mpm/services/agents/deployment/agent_metrics_collector.py +3 -3
  156. claude_mpm/services/agents/deployment/agent_restore_handler.py +1 -4
  157. claude_mpm/services/agents/deployment/agent_template_builder.py +236 -15
  158. claude_mpm/services/agents/deployment/agents_directory_resolver.py +101 -15
  159. claude_mpm/services/agents/deployment/async_agent_deployment.py +2 -1
  160. claude_mpm/services/agents/deployment/facade/deployment_facade.py +3 -3
  161. claude_mpm/services/agents/deployment/multi_source_deployment_service.py +225 -18
  162. claude_mpm/services/agents/deployment/pipeline/pipeline_executor.py +2 -2
  163. claude_mpm/services/agents/deployment/refactored_agent_deployment_service.py +1 -4
  164. claude_mpm/services/agents/deployment/remote_agent_discovery_service.py +557 -0
  165. claude_mpm/services/agents/deployment/single_agent_deployer.py +2 -2
  166. claude_mpm/services/agents/deployment/system_instructions_deployer.py +168 -46
  167. claude_mpm/services/agents/deployment/validation/deployment_validator.py +2 -2
  168. claude_mpm/services/agents/git_source_manager.py +629 -0
  169. claude_mpm/services/agents/loading/framework_agent_loader.py +9 -12
  170. claude_mpm/services/agents/local_template_manager.py +50 -10
  171. claude_mpm/services/agents/single_tier_deployment_service.py +696 -0
  172. claude_mpm/services/agents/sources/__init__.py +13 -0
  173. claude_mpm/services/agents/sources/agent_sync_state.py +516 -0
  174. claude_mpm/services/agents/sources/git_source_sync_service.py +1087 -0
  175. claude_mpm/services/agents/startup_sync.py +239 -0
  176. claude_mpm/services/agents/toolchain_detector.py +474 -0
  177. claude_mpm/services/analysis/__init__.py +25 -0
  178. claude_mpm/services/analysis/postmortem_reporter.py +474 -0
  179. claude_mpm/services/analysis/postmortem_service.py +765 -0
  180. claude_mpm/services/cli/session_pause_manager.py +1 -1
  181. claude_mpm/services/cli/unified_dashboard_manager.py +1 -1
  182. claude_mpm/services/command_deployment_service.py +200 -6
  183. claude_mpm/services/core/base.py +7 -2
  184. claude_mpm/services/core/interfaces/__init__.py +1 -3
  185. claude_mpm/services/core/interfaces/health.py +1 -4
  186. claude_mpm/services/core/models/__init__.py +2 -11
  187. claude_mpm/services/diagnostics/checks/__init__.py +4 -0
  188. claude_mpm/services/diagnostics/checks/agent_check.py +0 -2
  189. claude_mpm/services/diagnostics/checks/agent_sources_check.py +577 -0
  190. claude_mpm/services/diagnostics/checks/instructions_check.py +1 -2
  191. claude_mpm/services/diagnostics/checks/mcp_check.py +0 -1
  192. claude_mpm/services/diagnostics/checks/mcp_services_check.py +7 -15
  193. claude_mpm/services/diagnostics/checks/monitor_check.py +0 -1
  194. claude_mpm/services/diagnostics/checks/skill_sources_check.py +587 -0
  195. claude_mpm/services/diagnostics/diagnostic_runner.py +9 -0
  196. claude_mpm/services/diagnostics/doctor_reporter.py +40 -10
  197. claude_mpm/services/event_bus/direct_relay.py +3 -3
  198. claude_mpm/services/event_bus/event_bus.py +36 -3
  199. claude_mpm/services/events/consumers/logging.py +1 -2
  200. claude_mpm/services/git/__init__.py +21 -0
  201. claude_mpm/services/git/git_operations_service.py +494 -0
  202. claude_mpm/services/github/__init__.py +21 -0
  203. claude_mpm/services/github/github_cli_service.py +397 -0
  204. claude_mpm/services/infrastructure/monitoring/__init__.py +1 -5
  205. claude_mpm/services/infrastructure/monitoring/aggregator.py +1 -6
  206. claude_mpm/services/infrastructure/monitoring/resources.py +1 -1
  207. claude_mpm/services/instructions/__init__.py +9 -0
  208. claude_mpm/services/instructions/instruction_cache_service.py +374 -0
  209. claude_mpm/services/local_ops/__init__.py +3 -13
  210. claude_mpm/services/local_ops/health_checks/__init__.py +1 -3
  211. claude_mpm/services/local_ops/health_manager.py +1 -4
  212. claude_mpm/services/local_ops/process_manager.py +1 -1
  213. claude_mpm/services/local_ops/resource_monitor.py +2 -2
  214. claude_mpm/services/mcp_config_manager.py +75 -145
  215. claude_mpm/services/mcp_gateway/config/configuration.py +1 -1
  216. claude_mpm/services/mcp_gateway/core/process_pool.py +22 -16
  217. claude_mpm/services/mcp_gateway/server/mcp_gateway.py +1 -6
  218. claude_mpm/services/mcp_gateway/server/stdio_server.py +0 -2
  219. claude_mpm/services/mcp_gateway/tools/document_summarizer.py +1 -1
  220. claude_mpm/services/mcp_gateway/tools/kuzu_memory_service.py +6 -2
  221. claude_mpm/services/mcp_service_verifier.py +6 -3
  222. claude_mpm/services/memory/optimizer.py +1 -1
  223. claude_mpm/services/model/model_router.py +8 -9
  224. claude_mpm/services/monitor/daemon.py +29 -9
  225. claude_mpm/services/monitor/daemon_manager.py +96 -19
  226. claude_mpm/services/monitor/server.py +2 -2
  227. claude_mpm/services/native_agent_converter.py +356 -0
  228. claude_mpm/services/port_manager.py +1 -1
  229. claude_mpm/services/pr/__init__.py +14 -0
  230. claude_mpm/services/pr/pr_template_service.py +329 -0
  231. claude_mpm/services/project/documentation_manager.py +2 -1
  232. claude_mpm/services/project/project_organizer.py +4 -0
  233. claude_mpm/services/project/toolchain_analyzer.py +3 -1
  234. claude_mpm/services/runner_configuration_service.py +17 -3
  235. claude_mpm/services/self_upgrade_service.py +165 -7
  236. claude_mpm/services/session_management_service.py +16 -4
  237. claude_mpm/services/skills/__init__.py +18 -0
  238. claude_mpm/services/skills/git_skill_source_manager.py +1169 -0
  239. claude_mpm/services/skills/skill_discovery_service.py +568 -0
  240. claude_mpm/services/skills_config.py +547 -0
  241. claude_mpm/services/skills_deployer.py +955 -0
  242. claude_mpm/services/socketio/handlers/connection.py +1 -1
  243. claude_mpm/services/socketio/handlers/git.py +2 -2
  244. claude_mpm/services/socketio/server/core.py +1 -4
  245. claude_mpm/services/socketio/server/main.py +1 -3
  246. claude_mpm/services/system_instructions_service.py +1 -3
  247. claude_mpm/services/unified/analyzer_strategies/performance_analyzer.py +0 -3
  248. claude_mpm/services/unified/analyzer_strategies/security_analyzer.py +0 -1
  249. claude_mpm/services/unified/deployment_strategies/cloud_strategies.py +1 -1
  250. claude_mpm/services/unified/deployment_strategies/vercel.py +1 -5
  251. claude_mpm/services/unified/unified_deployment.py +1 -5
  252. claude_mpm/services/version_control/conflict_resolution.py +6 -4
  253. claude_mpm/services/visualization/__init__.py +1 -5
  254. claude_mpm/services/visualization/mermaid_generator.py +2 -3
  255. claude_mpm/skills/bundled/infrastructure/env-manager/scripts/validate_env.py +576 -0
  256. claude_mpm/skills/bundled/performance-profiling.md +6 -0
  257. claude_mpm/skills/bundled/testing/webapp-testing/scripts/with_server.py +2 -2
  258. claude_mpm/skills/skills_registry.py +0 -1
  259. claude_mpm/templates/questions/__init__.py +38 -0
  260. claude_mpm/templates/questions/base.py +193 -0
  261. claude_mpm/templates/questions/pr_strategy.py +311 -0
  262. claude_mpm/templates/questions/project_init.py +385 -0
  263. claude_mpm/templates/questions/ticket_mgmt.py +394 -0
  264. claude_mpm/tools/__main__.py +8 -8
  265. claude_mpm/tools/code_tree_analyzer/analysis.py +1 -1
  266. claude_mpm/utils/agent_dependency_loader.py +80 -13
  267. claude_mpm/utils/agent_filters.py +288 -0
  268. claude_mpm/utils/dependency_cache.py +3 -1
  269. claude_mpm/utils/gitignore.py +244 -0
  270. claude_mpm/utils/log_cleanup.py +3 -3
  271. claude_mpm/utils/migration.py +372 -0
  272. claude_mpm/utils/progress.py +387 -0
  273. claude_mpm/utils/robust_installer.py +3 -5
  274. claude_mpm/utils/structured_questions.py +619 -0
  275. {claude_mpm-4.21.3.dist-info → claude_mpm-5.1.9.dist-info}/METADATA +496 -65
  276. {claude_mpm-4.21.3.dist-info → claude_mpm-5.1.9.dist-info}/RECORD +284 -443
  277. claude_mpm/agents/templates/.claude-mpm/memories/README.md +0 -17
  278. claude_mpm/agents/templates/.claude-mpm/memories/engineer_memories.md +0 -3
  279. claude_mpm/agents/templates/agent-manager.json +0 -273
  280. claude_mpm/agents/templates/agentic-coder-optimizer.json +0 -248
  281. claude_mpm/agents/templates/api_qa.json +0 -180
  282. claude_mpm/agents/templates/circuit_breakers.md +0 -638
  283. claude_mpm/agents/templates/clerk-ops.json +0 -235
  284. claude_mpm/agents/templates/code_analyzer.json +0 -101
  285. claude_mpm/agents/templates/content-agent.json +0 -358
  286. claude_mpm/agents/templates/dart_engineer.json +0 -307
  287. claude_mpm/agents/templates/data_engineer.json +0 -225
  288. claude_mpm/agents/templates/documentation.json +0 -211
  289. claude_mpm/agents/templates/engineer.json +0 -210
  290. claude_mpm/agents/templates/gcp_ops_agent.json +0 -253
  291. claude_mpm/agents/templates/golang_engineer.json +0 -270
  292. claude_mpm/agents/templates/imagemagick.json +0 -264
  293. claude_mpm/agents/templates/java_engineer.json +0 -346
  294. claude_mpm/agents/templates/local_ops_agent.json +0 -1840
  295. claude_mpm/agents/templates/logs/prompts/agent_engineer_20250826_014258_728.md +0 -39
  296. claude_mpm/agents/templates/logs/prompts/agent_engineer_20250901_010124_142.md +0 -400
  297. claude_mpm/agents/templates/memory_manager.json +0 -158
  298. claude_mpm/agents/templates/nextjs_engineer.json +0 -285
  299. claude_mpm/agents/templates/ops.json +0 -185
  300. claude_mpm/agents/templates/php-engineer.json +0 -287
  301. claude_mpm/agents/templates/product_owner.json +0 -338
  302. claude_mpm/agents/templates/project_organizer.json +0 -140
  303. claude_mpm/agents/templates/prompt-engineer.json +0 -737
  304. claude_mpm/agents/templates/python_engineer.json +0 -387
  305. claude_mpm/agents/templates/qa.json +0 -242
  306. claude_mpm/agents/templates/react_engineer.json +0 -238
  307. claude_mpm/agents/templates/refactoring_engineer.json +0 -276
  308. claude_mpm/agents/templates/research.json +0 -188
  309. claude_mpm/agents/templates/ruby-engineer.json +0 -280
  310. claude_mpm/agents/templates/rust_engineer.json +0 -275
  311. claude_mpm/agents/templates/security.json +0 -202
  312. claude_mpm/agents/templates/svelte-engineer.json +0 -225
  313. claude_mpm/agents/templates/ticketing.json +0 -177
  314. claude_mpm/agents/templates/typescript_engineer.json +0 -285
  315. claude_mpm/agents/templates/vercel_ops_agent.json +0 -412
  316. claude_mpm/agents/templates/version_control.json +0 -157
  317. claude_mpm/agents/templates/web_qa.json +0 -399
  318. claude_mpm/agents/templates/web_ui.json +0 -189
  319. claude_mpm/commands/mpm-tickets.md +0 -102
  320. claude_mpm/dashboard/.claude-mpm/socketio-instances.json +0 -1
  321. claude_mpm/dashboard/react/components/DataInspector/DataInspector.module.css +0 -188
  322. claude_mpm/dashboard/react/components/EventViewer/EventViewer.module.css +0 -156
  323. claude_mpm/dashboard/react/components/shared/ConnectionStatus.module.css +0 -38
  324. claude_mpm/dashboard/react/components/shared/FilterBar.module.css +0 -92
  325. claude_mpm/dashboard/static/archive/activity_dashboard_fixed.html +0 -248
  326. claude_mpm/dashboard/static/archive/activity_dashboard_test.html +0 -61
  327. claude_mpm/dashboard/static/archive/test_activity_connection.html +0 -179
  328. claude_mpm/dashboard/static/archive/test_claude_tree_tab.html +0 -68
  329. claude_mpm/dashboard/static/archive/test_dashboard.html +0 -409
  330. claude_mpm/dashboard/static/archive/test_dashboard_fixed.html +0 -519
  331. claude_mpm/dashboard/static/archive/test_dashboard_verification.html +0 -181
  332. claude_mpm/dashboard/static/archive/test_file_data.html +0 -315
  333. claude_mpm/dashboard/static/archive/test_file_tree_empty_state.html +0 -243
  334. claude_mpm/dashboard/static/archive/test_file_tree_fix.html +0 -234
  335. claude_mpm/dashboard/static/archive/test_file_tree_rename.html +0 -117
  336. claude_mpm/dashboard/static/archive/test_file_tree_tab.html +0 -115
  337. claude_mpm/dashboard/static/archive/test_file_viewer.html +0 -224
  338. claude_mpm/dashboard/static/archive/test_final_activity.html +0 -220
  339. claude_mpm/dashboard/static/archive/test_tab_fix.html +0 -139
  340. claude_mpm/dashboard/static/built/assets/events.DjpNxWNo.css +0 -1
  341. claude_mpm/dashboard/static/built/components/activity-tree.js +0 -2
  342. claude_mpm/dashboard/static/built/components/agent-hierarchy.js +0 -777
  343. claude_mpm/dashboard/static/built/components/agent-inference.js +0 -2
  344. claude_mpm/dashboard/static/built/components/build-tracker.js +0 -333
  345. claude_mpm/dashboard/static/built/components/code-simple.js +0 -857
  346. claude_mpm/dashboard/static/built/components/code-tree/tree-breadcrumb.js +0 -353
  347. claude_mpm/dashboard/static/built/components/code-tree/tree-constants.js +0 -235
  348. claude_mpm/dashboard/static/built/components/code-tree/tree-search.js +0 -409
  349. claude_mpm/dashboard/static/built/components/code-tree/tree-utils.js +0 -435
  350. claude_mpm/dashboard/static/built/components/code-tree.js +0 -2
  351. claude_mpm/dashboard/static/built/components/code-viewer.js +0 -2
  352. claude_mpm/dashboard/static/built/components/connection-debug.js +0 -654
  353. claude_mpm/dashboard/static/built/components/diff-viewer.js +0 -891
  354. claude_mpm/dashboard/static/built/components/event-processor.js +0 -2
  355. claude_mpm/dashboard/static/built/components/event-viewer.js +0 -2
  356. claude_mpm/dashboard/static/built/components/export-manager.js +0 -2
  357. claude_mpm/dashboard/static/built/components/file-change-tracker.js +0 -443
  358. claude_mpm/dashboard/static/built/components/file-change-viewer.js +0 -690
  359. claude_mpm/dashboard/static/built/components/file-tool-tracker.js +0 -2
  360. claude_mpm/dashboard/static/built/components/file-viewer.js +0 -2
  361. claude_mpm/dashboard/static/built/components/hud-library-loader.js +0 -2
  362. claude_mpm/dashboard/static/built/components/hud-manager.js +0 -2
  363. claude_mpm/dashboard/static/built/components/hud-visualizer.js +0 -2
  364. claude_mpm/dashboard/static/built/components/module-viewer.js +0 -2
  365. claude_mpm/dashboard/static/built/components/nav-bar.js +0 -145
  366. claude_mpm/dashboard/static/built/components/page-structure.js +0 -429
  367. claude_mpm/dashboard/static/built/components/session-manager.js +0 -2
  368. claude_mpm/dashboard/static/built/components/socket-manager.js +0 -2
  369. claude_mpm/dashboard/static/built/components/ui-state-manager.js +0 -2
  370. claude_mpm/dashboard/static/built/components/unified-data-viewer.js +0 -2
  371. claude_mpm/dashboard/static/built/components/working-directory.js +0 -2
  372. claude_mpm/dashboard/static/built/connection-manager.js +0 -536
  373. claude_mpm/dashboard/static/built/dashboard.js +0 -2
  374. claude_mpm/dashboard/static/built/extension-error-handler.js +0 -164
  375. claude_mpm/dashboard/static/built/react/events.js +0 -30
  376. claude_mpm/dashboard/static/built/shared/dom-helpers.js +0 -396
  377. claude_mpm/dashboard/static/built/shared/event-bus.js +0 -330
  378. claude_mpm/dashboard/static/built/shared/event-filter-service.js +0 -540
  379. claude_mpm/dashboard/static/built/shared/logger.js +0 -385
  380. claude_mpm/dashboard/static/built/shared/page-structure.js +0 -249
  381. claude_mpm/dashboard/static/built/shared/tooltip-service.js +0 -253
  382. claude_mpm/dashboard/static/built/socket-client.js +0 -2
  383. claude_mpm/dashboard/static/built/tab-isolation-fix.js +0 -185
  384. claude_mpm/dashboard/static/dist/assets/events.DjpNxWNo.css +0 -1
  385. claude_mpm/dashboard/static/dist/components/activity-tree.js +0 -2
  386. claude_mpm/dashboard/static/dist/components/agent-inference.js +0 -2
  387. claude_mpm/dashboard/static/dist/components/code-tree.js +0 -2
  388. claude_mpm/dashboard/static/dist/components/code-viewer.js +0 -2
  389. claude_mpm/dashboard/static/dist/components/event-processor.js +0 -2
  390. claude_mpm/dashboard/static/dist/components/event-viewer.js +0 -2
  391. claude_mpm/dashboard/static/dist/components/export-manager.js +0 -2
  392. claude_mpm/dashboard/static/dist/components/file-tool-tracker.js +0 -2
  393. claude_mpm/dashboard/static/dist/components/file-viewer.js +0 -2
  394. claude_mpm/dashboard/static/dist/components/hud-library-loader.js +0 -2
  395. claude_mpm/dashboard/static/dist/components/hud-manager.js +0 -2
  396. claude_mpm/dashboard/static/dist/components/hud-visualizer.js +0 -2
  397. claude_mpm/dashboard/static/dist/components/module-viewer.js +0 -2
  398. claude_mpm/dashboard/static/dist/components/session-manager.js +0 -2
  399. claude_mpm/dashboard/static/dist/components/socket-manager.js +0 -2
  400. claude_mpm/dashboard/static/dist/components/ui-state-manager.js +0 -2
  401. claude_mpm/dashboard/static/dist/components/unified-data-viewer.js +0 -2
  402. claude_mpm/dashboard/static/dist/components/working-directory.js +0 -2
  403. claude_mpm/dashboard/static/dist/dashboard.js +0 -2
  404. claude_mpm/dashboard/static/dist/react/events.js +0 -30
  405. claude_mpm/dashboard/static/dist/socket-client.js +0 -2
  406. claude_mpm/dashboard/static/events.html +0 -607
  407. claude_mpm/dashboard/static/index.html +0 -635
  408. claude_mpm/dashboard/static/js/shared/dom-helpers.js +0 -396
  409. claude_mpm/dashboard/static/js/shared/event-bus.js +0 -330
  410. claude_mpm/dashboard/static/js/shared/logger.js +0 -385
  411. claude_mpm/dashboard/static/js/shared/tooltip-service.js +0 -253
  412. claude_mpm/dashboard/static/js/stores/dashboard-store.js +0 -562
  413. claude_mpm/dashboard/static/legacy/activity.html +0 -736
  414. claude_mpm/dashboard/static/legacy/agents.html +0 -786
  415. claude_mpm/dashboard/static/legacy/files.html +0 -747
  416. claude_mpm/dashboard/static/legacy/tools.html +0 -831
  417. claude_mpm/dashboard/static/monitors.html +0 -431
  418. claude_mpm/dashboard/static/production/events.html +0 -659
  419. claude_mpm/dashboard/static/production/main.html +0 -698
  420. claude_mpm/dashboard/static/production/monitors.html +0 -483
  421. claude_mpm/dashboard/static/test-archive/dashboard.html +0 -635
  422. claude_mpm/dashboard/static/test-archive/debug-events.html +0 -147
  423. claude_mpm/dashboard/static/test-archive/test-navigation.html +0 -256
  424. claude_mpm/dashboard/static/test-archive/test-react-exports.html +0 -180
  425. claude_mpm/dashboard/static/test-archive/test_debug.html +0 -25
  426. claude_mpm/skills/bundled/collaboration/brainstorming/SKILL.md +0 -79
  427. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/SKILL.md +0 -178
  428. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/agent-prompts.md +0 -577
  429. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/coordination-patterns.md +0 -467
  430. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/examples.md +0 -537
  431. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/troubleshooting.md +0 -730
  432. claude_mpm/skills/bundled/collaboration/requesting-code-review/SKILL.md +0 -112
  433. claude_mpm/skills/bundled/collaboration/requesting-code-review/references/code-reviewer-template.md +0 -146
  434. claude_mpm/skills/bundled/collaboration/requesting-code-review/references/review-examples.md +0 -412
  435. claude_mpm/skills/bundled/collaboration/writing-plans/SKILL.md +0 -81
  436. claude_mpm/skills/bundled/collaboration/writing-plans/references/best-practices.md +0 -362
  437. claude_mpm/skills/bundled/collaboration/writing-plans/references/plan-structure-templates.md +0 -312
  438. claude_mpm/skills/bundled/debugging/root-cause-tracing/SKILL.md +0 -152
  439. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/advanced-techniques.md +0 -668
  440. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/examples.md +0 -587
  441. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/integration.md +0 -438
  442. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/tracing-techniques.md +0 -391
  443. claude_mpm/skills/bundled/debugging/systematic-debugging/CREATION-LOG.md +0 -119
  444. claude_mpm/skills/bundled/debugging/systematic-debugging/SKILL.md +0 -148
  445. claude_mpm/skills/bundled/debugging/systematic-debugging/references/anti-patterns.md +0 -483
  446. claude_mpm/skills/bundled/debugging/systematic-debugging/references/examples.md +0 -452
  447. claude_mpm/skills/bundled/debugging/systematic-debugging/references/troubleshooting.md +0 -449
  448. claude_mpm/skills/bundled/debugging/systematic-debugging/references/workflow.md +0 -411
  449. claude_mpm/skills/bundled/debugging/systematic-debugging/test-academic.md +0 -14
  450. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-1.md +0 -58
  451. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-2.md +0 -68
  452. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-3.md +0 -69
  453. claude_mpm/skills/bundled/debugging/verification-before-completion/SKILL.md +0 -131
  454. claude_mpm/skills/bundled/debugging/verification-before-completion/references/gate-function.md +0 -325
  455. claude_mpm/skills/bundled/debugging/verification-before-completion/references/integration-and-workflows.md +0 -490
  456. claude_mpm/skills/bundled/debugging/verification-before-completion/references/red-flags-and-failures.md +0 -425
  457. claude_mpm/skills/bundled/debugging/verification-before-completion/references/verification-patterns.md +0 -499
  458. claude_mpm/skills/bundled/main/artifacts-builder/SKILL.md +0 -86
  459. claude_mpm/skills/bundled/main/internal-comms/SKILL.md +0 -43
  460. claude_mpm/skills/bundled/main/internal-comms/examples/3p-updates.md +0 -47
  461. claude_mpm/skills/bundled/main/internal-comms/examples/company-newsletter.md +0 -65
  462. claude_mpm/skills/bundled/main/internal-comms/examples/faq-answers.md +0 -30
  463. claude_mpm/skills/bundled/main/internal-comms/examples/general-comms.md +0 -16
  464. claude_mpm/skills/bundled/main/mcp-builder/SKILL.md +0 -160
  465. claude_mpm/skills/bundled/main/mcp-builder/reference/design_principles.md +0 -412
  466. claude_mpm/skills/bundled/main/mcp-builder/reference/evaluation.md +0 -602
  467. claude_mpm/skills/bundled/main/mcp-builder/reference/mcp_best_practices.md +0 -915
  468. claude_mpm/skills/bundled/main/mcp-builder/reference/node_mcp_server.md +0 -916
  469. claude_mpm/skills/bundled/main/mcp-builder/reference/python_mcp_server.md +0 -752
  470. claude_mpm/skills/bundled/main/mcp-builder/reference/workflow.md +0 -1237
  471. claude_mpm/skills/bundled/main/skill-creator/SKILL.md +0 -189
  472. claude_mpm/skills/bundled/main/skill-creator/references/best-practices.md +0 -500
  473. claude_mpm/skills/bundled/main/skill-creator/references/creation-workflow.md +0 -464
  474. claude_mpm/skills/bundled/main/skill-creator/references/examples.md +0 -619
  475. claude_mpm/skills/bundled/main/skill-creator/references/progressive-disclosure.md +0 -437
  476. claude_mpm/skills/bundled/main/skill-creator/references/skill-structure.md +0 -231
  477. claude_mpm/skills/bundled/php/espocrm-development/SKILL.md +0 -170
  478. claude_mpm/skills/bundled/php/espocrm-development/references/architecture.md +0 -602
  479. claude_mpm/skills/bundled/php/espocrm-development/references/common-tasks.md +0 -821
  480. claude_mpm/skills/bundled/php/espocrm-development/references/development-workflow.md +0 -742
  481. claude_mpm/skills/bundled/php/espocrm-development/references/frontend-customization.md +0 -726
  482. claude_mpm/skills/bundled/php/espocrm-development/references/hooks-and-services.md +0 -764
  483. claude_mpm/skills/bundled/php/espocrm-development/references/testing-debugging.md +0 -831
  484. claude_mpm/skills/bundled/rust/desktop-applications/SKILL.md +0 -226
  485. claude_mpm/skills/bundled/rust/desktop-applications/references/architecture-patterns.md +0 -901
  486. claude_mpm/skills/bundled/rust/desktop-applications/references/native-gui-frameworks.md +0 -901
  487. claude_mpm/skills/bundled/rust/desktop-applications/references/platform-integration.md +0 -775
  488. claude_mpm/skills/bundled/rust/desktop-applications/references/state-management.md +0 -937
  489. claude_mpm/skills/bundled/rust/desktop-applications/references/tauri-framework.md +0 -770
  490. claude_mpm/skills/bundled/rust/desktop-applications/references/testing-deployment.md +0 -961
  491. claude_mpm/skills/bundled/testing/condition-based-waiting/SKILL.md +0 -119
  492. claude_mpm/skills/bundled/testing/condition-based-waiting/references/patterns-and-implementation.md +0 -253
  493. claude_mpm/skills/bundled/testing/test-driven-development/SKILL.md +0 -145
  494. claude_mpm/skills/bundled/testing/test-driven-development/references/anti-patterns.md +0 -543
  495. claude_mpm/skills/bundled/testing/test-driven-development/references/examples.md +0 -741
  496. claude_mpm/skills/bundled/testing/test-driven-development/references/integration.md +0 -470
  497. claude_mpm/skills/bundled/testing/test-driven-development/references/philosophy.md +0 -458
  498. claude_mpm/skills/bundled/testing/test-driven-development/references/workflow.md +0 -639
  499. claude_mpm/skills/bundled/testing/testing-anti-patterns/SKILL.md +0 -140
  500. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/completeness-anti-patterns.md +0 -572
  501. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/core-anti-patterns.md +0 -411
  502. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/detection-guide.md +0 -569
  503. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/tdd-connection.md +0 -695
  504. claude_mpm/skills/bundled/testing/webapp-testing/SKILL.md +0 -184
  505. claude_mpm/skills/bundled/testing/webapp-testing/decision-tree.md +0 -459
  506. claude_mpm/skills/bundled/testing/webapp-testing/playwright-patterns.md +0 -479
  507. claude_mpm/skills/bundled/testing/webapp-testing/reconnaissance-pattern.md +0 -687
  508. claude_mpm/skills/bundled/testing/webapp-testing/server-management.md +0 -758
  509. claude_mpm/skills/bundled/testing/webapp-testing/troubleshooting.md +0 -868
  510. /claude_mpm/agents/templates/{git_file_tracking.md → git-file-tracking.md} +0 -0
  511. /claude_mpm/agents/templates/{pm_examples.md → pm-examples.md} +0 -0
  512. /claude_mpm/agents/templates/{response_format.md → response-format.md} +0 -0
  513. /claude_mpm/agents/templates/{validation_templates.md → validation-templates.md} +0 -0
  514. {claude_mpm-4.21.3.dist-info → claude_mpm-5.1.9.dist-info}/WHEEL +0 -0
  515. {claude_mpm-4.21.3.dist-info → claude_mpm-5.1.9.dist-info}/entry_points.txt +0 -0
  516. {claude_mpm-4.21.3.dist-info → claude_mpm-5.1.9.dist-info}/licenses/LICENSE +0 -0
  517. {claude_mpm-4.21.3.dist-info → claude_mpm-5.1.9.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
- ```