claude-mpm 4.21.0__py3-none-any.whl → 5.0.2__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (497) hide show
  1. claude_mpm/VERSION +1 -1
  2. claude_mpm/agents/BASE_PM.md +12 -0
  3. claude_mpm/agents/OUTPUT_STYLE.md +3 -48
  4. claude_mpm/agents/PM_INSTRUCTIONS.md +632 -334
  5. claude_mpm/agents/WORKFLOW.md +75 -2
  6. claude_mpm/agents/__init__.py +6 -0
  7. claude_mpm/agents/agent_loader.py +1 -4
  8. claude_mpm/agents/base_agent.json +6 -3
  9. claude_mpm/agents/frontmatter_validator.py +1 -1
  10. claude_mpm/agents/templates/{circuit_breakers.md → circuit-breakers.md} +370 -3
  11. claude_mpm/agents/templates/context-management-examples.md +544 -0
  12. claude_mpm/agents/templates/{pm_red_flags.md → pm-red-flags.md} +89 -19
  13. claude_mpm/agents/templates/pr-workflow-examples.md +427 -0
  14. claude_mpm/agents/templates/research-gate-examples.md +669 -0
  15. claude_mpm/agents/templates/structured-questions-examples.md +615 -0
  16. claude_mpm/agents/templates/ticket-completeness-examples.md +139 -0
  17. claude_mpm/agents/templates/ticketing-examples.md +277 -0
  18. claude_mpm/cli/__init__.py +38 -2
  19. claude_mpm/cli/commands/agent_source.py +774 -0
  20. claude_mpm/cli/commands/agent_state_manager.py +125 -20
  21. claude_mpm/cli/commands/agents.py +684 -13
  22. claude_mpm/cli/commands/agents_cleanup.py +210 -0
  23. claude_mpm/cli/commands/agents_discover.py +338 -0
  24. claude_mpm/cli/commands/aggregate.py +1 -1
  25. claude_mpm/cli/commands/analyze.py +3 -3
  26. claude_mpm/cli/commands/auto_configure.py +2 -6
  27. claude_mpm/cli/commands/cleanup.py +1 -1
  28. claude_mpm/cli/commands/config.py +7 -4
  29. claude_mpm/cli/commands/configure.py +478 -44
  30. claude_mpm/cli/commands/configure_agent_display.py +4 -4
  31. claude_mpm/cli/commands/configure_navigation.py +63 -46
  32. claude_mpm/cli/commands/debug.py +12 -12
  33. claude_mpm/cli/commands/doctor.py +10 -2
  34. claude_mpm/cli/commands/hook_errors.py +277 -0
  35. claude_mpm/cli/commands/local_deploy.py +1 -4
  36. claude_mpm/cli/commands/mcp_install_commands.py +1 -1
  37. claude_mpm/cli/commands/mpm_init/core.py +50 -2
  38. claude_mpm/cli/commands/mpm_init/git_activity.py +10 -10
  39. claude_mpm/cli/commands/mpm_init/prompts.py +6 -6
  40. claude_mpm/cli/commands/run.py +124 -128
  41. claude_mpm/cli/commands/skill_source.py +694 -0
  42. claude_mpm/cli/commands/skills.py +435 -1
  43. claude_mpm/cli/executor.py +78 -3
  44. claude_mpm/cli/interactive/agent_wizard.py +919 -41
  45. claude_mpm/cli/parsers/agent_source_parser.py +171 -0
  46. claude_mpm/cli/parsers/agents_parser.py +173 -4
  47. claude_mpm/cli/parsers/base_parser.py +49 -0
  48. claude_mpm/cli/parsers/config_parser.py +96 -43
  49. claude_mpm/cli/parsers/skill_source_parser.py +169 -0
  50. claude_mpm/cli/parsers/skills_parser.py +138 -0
  51. claude_mpm/cli/parsers/source_parser.py +138 -0
  52. claude_mpm/cli/startup.py +499 -84
  53. claude_mpm/cli/startup_display.py +480 -0
  54. claude_mpm/cli/utils.py +1 -1
  55. claude_mpm/cli_module/commands.py +1 -1
  56. claude_mpm/commands/{mpm-auto-configure.md → mpm-agents-auto-configure.md} +9 -0
  57. claude_mpm/commands/mpm-agents-detect.md +9 -0
  58. claude_mpm/commands/{mpm-agents.md → mpm-agents-list.md} +9 -0
  59. claude_mpm/commands/mpm-agents-recommend.md +9 -0
  60. claude_mpm/commands/{mpm-config.md → mpm-config-view.md} +9 -0
  61. claude_mpm/commands/mpm-doctor.md +9 -0
  62. claude_mpm/commands/mpm-help.md +14 -2
  63. claude_mpm/commands/mpm-init.md +27 -2
  64. claude_mpm/commands/mpm-monitor.md +9 -0
  65. claude_mpm/commands/mpm-session-resume.md +381 -0
  66. claude_mpm/commands/mpm-status.md +9 -0
  67. claude_mpm/commands/{mpm-organize.md → mpm-ticket-organize.md} +9 -0
  68. claude_mpm/commands/mpm-ticket-view.md +552 -0
  69. claude_mpm/commands/mpm-version.md +9 -0
  70. claude_mpm/commands/mpm.md +11 -0
  71. claude_mpm/config/agent_presets.py +258 -0
  72. claude_mpm/config/agent_sources.py +325 -0
  73. claude_mpm/config/skill_sources.py +590 -0
  74. claude_mpm/constants.py +12 -0
  75. claude_mpm/core/api_validator.py +1 -1
  76. claude_mpm/core/claude_runner.py +17 -10
  77. claude_mpm/core/config.py +24 -0
  78. claude_mpm/core/constants.py +1 -1
  79. claude_mpm/core/framework/__init__.py +3 -16
  80. claude_mpm/core/framework/loaders/instruction_loader.py +25 -5
  81. claude_mpm/core/framework/processors/metadata_processor.py +1 -1
  82. claude_mpm/core/hook_error_memory.py +381 -0
  83. claude_mpm/core/hook_manager.py +41 -2
  84. claude_mpm/core/interactive_session.py +112 -5
  85. claude_mpm/core/logger.py +3 -1
  86. claude_mpm/core/oneshot_session.py +94 -4
  87. claude_mpm/dashboard/static/css/activity.css +69 -69
  88. claude_mpm/dashboard/static/css/connection-status.css +10 -10
  89. claude_mpm/dashboard/static/css/dashboard.css +15 -15
  90. claude_mpm/dashboard/static/js/components/activity-tree.js +178 -178
  91. claude_mpm/dashboard/static/js/components/agent-hierarchy.js +101 -101
  92. claude_mpm/dashboard/static/js/components/agent-inference.js +31 -31
  93. claude_mpm/dashboard/static/js/components/build-tracker.js +59 -59
  94. claude_mpm/dashboard/static/js/components/code-simple.js +107 -107
  95. claude_mpm/dashboard/static/js/components/connection-debug.js +101 -101
  96. claude_mpm/dashboard/static/js/components/diff-viewer.js +113 -113
  97. claude_mpm/dashboard/static/js/components/event-viewer.js +12 -12
  98. claude_mpm/dashboard/static/js/components/file-change-tracker.js +57 -57
  99. claude_mpm/dashboard/static/js/components/file-change-viewer.js +74 -74
  100. claude_mpm/dashboard/static/js/components/file-tool-tracker.js +6 -6
  101. claude_mpm/dashboard/static/js/components/file-viewer.js +42 -42
  102. claude_mpm/dashboard/static/js/components/module-viewer.js +27 -27
  103. claude_mpm/dashboard/static/js/components/session-manager.js +14 -14
  104. claude_mpm/dashboard/static/js/components/socket-manager.js +1 -1
  105. claude_mpm/dashboard/static/js/components/ui-state-manager.js +14 -14
  106. claude_mpm/dashboard/static/js/components/unified-data-viewer.js +110 -110
  107. claude_mpm/dashboard/static/js/components/working-directory.js +8 -8
  108. claude_mpm/dashboard/static/js/connection-manager.js +76 -76
  109. claude_mpm/dashboard/static/js/dashboard.js +76 -58
  110. claude_mpm/dashboard/static/js/extension-error-handler.js +22 -22
  111. claude_mpm/dashboard/static/js/socket-client.js +138 -121
  112. claude_mpm/dashboard/templates/code_simple.html +23 -23
  113. claude_mpm/dashboard/templates/index.html +18 -18
  114. claude_mpm/experimental/cli_enhancements.py +1 -5
  115. claude_mpm/hooks/claude_hooks/event_handlers.py +3 -1
  116. claude_mpm/hooks/claude_hooks/hook_handler.py +24 -7
  117. claude_mpm/hooks/claude_hooks/installer.py +45 -0
  118. claude_mpm/hooks/failure_learning/__init__.py +2 -8
  119. claude_mpm/hooks/failure_learning/failure_detection_hook.py +1 -6
  120. claude_mpm/hooks/failure_learning/fix_detection_hook.py +1 -6
  121. claude_mpm/hooks/failure_learning/learning_extraction_hook.py +1 -6
  122. claude_mpm/hooks/kuzu_response_hook.py +1 -5
  123. claude_mpm/hooks/templates/pre_tool_use_simple.py +78 -0
  124. claude_mpm/hooks/templates/pre_tool_use_template.py +323 -0
  125. claude_mpm/models/git_repository.py +198 -0
  126. claude_mpm/scripts/claude-hook-handler.sh +3 -3
  127. claude_mpm/scripts/start_activity_logging.py +3 -1
  128. claude_mpm/services/agents/agent_builder.py +45 -9
  129. claude_mpm/services/agents/agent_preset_service.py +238 -0
  130. claude_mpm/services/agents/agent_selection_service.py +484 -0
  131. claude_mpm/services/agents/auto_deploy_index_parser.py +569 -0
  132. claude_mpm/services/agents/deployment/agent_deployment.py +126 -2
  133. claude_mpm/services/agents/deployment/agent_discovery_service.py +105 -73
  134. claude_mpm/services/agents/deployment/agent_format_converter.py +1 -1
  135. claude_mpm/services/agents/deployment/agent_lifecycle_manager.py +1 -5
  136. claude_mpm/services/agents/deployment/agent_metrics_collector.py +3 -3
  137. claude_mpm/services/agents/deployment/agent_restore_handler.py +1 -4
  138. claude_mpm/services/agents/deployment/agent_template_builder.py +236 -15
  139. claude_mpm/services/agents/deployment/agents_directory_resolver.py +101 -15
  140. claude_mpm/services/agents/deployment/async_agent_deployment.py +2 -1
  141. claude_mpm/services/agents/deployment/facade/deployment_facade.py +3 -3
  142. claude_mpm/services/agents/deployment/multi_source_deployment_service.py +115 -15
  143. claude_mpm/services/agents/deployment/pipeline/pipeline_executor.py +2 -2
  144. claude_mpm/services/agents/deployment/refactored_agent_deployment_service.py +1 -4
  145. claude_mpm/services/agents/deployment/remote_agent_discovery_service.py +363 -0
  146. claude_mpm/services/agents/deployment/single_agent_deployer.py +2 -2
  147. claude_mpm/services/agents/deployment/system_instructions_deployer.py +168 -46
  148. claude_mpm/services/agents/deployment/validation/deployment_validator.py +2 -2
  149. claude_mpm/services/agents/git_source_manager.py +629 -0
  150. claude_mpm/services/agents/loading/framework_agent_loader.py +9 -12
  151. claude_mpm/services/agents/local_template_manager.py +50 -10
  152. claude_mpm/services/agents/single_tier_deployment_service.py +696 -0
  153. claude_mpm/services/agents/sources/__init__.py +13 -0
  154. claude_mpm/services/agents/sources/agent_sync_state.py +516 -0
  155. claude_mpm/services/agents/sources/git_source_sync_service.py +1055 -0
  156. claude_mpm/services/agents/startup_sync.py +239 -0
  157. claude_mpm/services/agents/toolchain_detector.py +474 -0
  158. claude_mpm/services/cli/session_pause_manager.py +1 -1
  159. claude_mpm/services/cli/unified_dashboard_manager.py +1 -1
  160. claude_mpm/services/command_deployment_service.py +92 -1
  161. claude_mpm/services/core/base.py +26 -11
  162. claude_mpm/services/core/interfaces/__init__.py +1 -3
  163. claude_mpm/services/core/interfaces/health.py +1 -4
  164. claude_mpm/services/core/models/__init__.py +2 -11
  165. claude_mpm/services/diagnostics/checks/__init__.py +4 -0
  166. claude_mpm/services/diagnostics/checks/agent_check.py +0 -2
  167. claude_mpm/services/diagnostics/checks/agent_sources_check.py +577 -0
  168. claude_mpm/services/diagnostics/checks/instructions_check.py +1 -2
  169. claude_mpm/services/diagnostics/checks/mcp_check.py +0 -1
  170. claude_mpm/services/diagnostics/checks/monitor_check.py +0 -1
  171. claude_mpm/services/diagnostics/checks/skill_sources_check.py +587 -0
  172. claude_mpm/services/diagnostics/diagnostic_runner.py +9 -0
  173. claude_mpm/services/diagnostics/doctor_reporter.py +40 -10
  174. claude_mpm/services/event_bus/direct_relay.py +3 -3
  175. claude_mpm/services/event_bus/event_bus.py +36 -3
  176. claude_mpm/services/event_bus/relay.py +23 -7
  177. claude_mpm/services/events/consumers/logging.py +1 -2
  178. claude_mpm/services/git/__init__.py +21 -0
  179. claude_mpm/services/git/git_operations_service.py +494 -0
  180. claude_mpm/services/github/__init__.py +21 -0
  181. claude_mpm/services/github/github_cli_service.py +397 -0
  182. claude_mpm/services/infrastructure/monitoring/__init__.py +1 -5
  183. claude_mpm/services/infrastructure/monitoring/aggregator.py +1 -6
  184. claude_mpm/services/infrastructure/monitoring/resources.py +1 -1
  185. claude_mpm/services/instructions/__init__.py +9 -0
  186. claude_mpm/services/instructions/instruction_cache_service.py +374 -0
  187. claude_mpm/services/local_ops/__init__.py +3 -13
  188. claude_mpm/services/local_ops/health_checks/__init__.py +1 -3
  189. claude_mpm/services/local_ops/health_manager.py +1 -4
  190. claude_mpm/services/local_ops/process_manager.py +1 -1
  191. claude_mpm/services/local_ops/resource_monitor.py +2 -2
  192. claude_mpm/services/mcp_gateway/config/configuration.py +1 -1
  193. claude_mpm/services/mcp_gateway/server/mcp_gateway.py +1 -6
  194. claude_mpm/services/mcp_gateway/server/stdio_server.py +0 -2
  195. claude_mpm/services/mcp_gateway/tools/document_summarizer.py +1 -1
  196. claude_mpm/services/mcp_gateway/tools/kuzu_memory_service.py +6 -2
  197. claude_mpm/services/memory/failure_tracker.py +19 -4
  198. claude_mpm/services/memory/optimizer.py +1 -1
  199. claude_mpm/services/model/model_router.py +8 -9
  200. claude_mpm/services/monitor/daemon.py +1 -1
  201. claude_mpm/services/monitor/server.py +2 -2
  202. claude_mpm/services/native_agent_converter.py +356 -0
  203. claude_mpm/services/port_manager.py +1 -1
  204. claude_mpm/services/pr/__init__.py +14 -0
  205. claude_mpm/services/pr/pr_template_service.py +329 -0
  206. claude_mpm/services/project/documentation_manager.py +2 -1
  207. claude_mpm/services/project/toolchain_analyzer.py +3 -1
  208. claude_mpm/services/runner_configuration_service.py +1 -0
  209. claude_mpm/services/self_upgrade_service.py +165 -7
  210. claude_mpm/services/skills/__init__.py +18 -0
  211. claude_mpm/services/skills/git_skill_source_manager.py +1169 -0
  212. claude_mpm/services/skills/skill_discovery_service.py +568 -0
  213. claude_mpm/services/skills_config.py +547 -0
  214. claude_mpm/services/skills_deployer.py +955 -0
  215. claude_mpm/services/socketio/handlers/connection.py +1 -1
  216. claude_mpm/services/socketio/handlers/git.py +2 -2
  217. claude_mpm/services/socketio/server/core.py +1 -4
  218. claude_mpm/services/socketio/server/main.py +1 -3
  219. claude_mpm/services/system_instructions_service.py +1 -3
  220. claude_mpm/services/unified/analyzer_strategies/performance_analyzer.py +0 -3
  221. claude_mpm/services/unified/analyzer_strategies/security_analyzer.py +0 -1
  222. claude_mpm/services/unified/deployment_strategies/cloud_strategies.py +1 -1
  223. claude_mpm/services/unified/deployment_strategies/vercel.py +1 -5
  224. claude_mpm/services/unified/unified_deployment.py +1 -5
  225. claude_mpm/services/version_control/conflict_resolution.py +6 -4
  226. claude_mpm/services/visualization/__init__.py +1 -5
  227. claude_mpm/services/visualization/mermaid_generator.py +2 -3
  228. claude_mpm/skills/bundled/infrastructure/env-manager/scripts/validate_env.py +576 -0
  229. claude_mpm/skills/bundled/performance-profiling.md +6 -0
  230. claude_mpm/skills/bundled/testing/webapp-testing/scripts/with_server.py +2 -2
  231. claude_mpm/skills/skills_registry.py +0 -1
  232. claude_mpm/templates/questions/__init__.py +38 -0
  233. claude_mpm/templates/questions/base.py +193 -0
  234. claude_mpm/templates/questions/pr_strategy.py +311 -0
  235. claude_mpm/templates/questions/project_init.py +385 -0
  236. claude_mpm/templates/questions/ticket_mgmt.py +394 -0
  237. claude_mpm/tools/__main__.py +8 -8
  238. claude_mpm/tools/code_tree_analyzer/__init__.py +45 -0
  239. claude_mpm/tools/code_tree_analyzer/analysis.py +299 -0
  240. claude_mpm/tools/code_tree_analyzer/cache.py +131 -0
  241. claude_mpm/tools/code_tree_analyzer/core.py +380 -0
  242. claude_mpm/tools/code_tree_analyzer/discovery.py +403 -0
  243. claude_mpm/tools/code_tree_analyzer/events.py +168 -0
  244. claude_mpm/tools/code_tree_analyzer/gitignore.py +308 -0
  245. claude_mpm/tools/code_tree_analyzer/models.py +39 -0
  246. claude_mpm/tools/code_tree_analyzer/multilang_analyzer.py +224 -0
  247. claude_mpm/tools/code_tree_analyzer/python_analyzer.py +284 -0
  248. claude_mpm/utils/agent_dependency_loader.py +80 -13
  249. claude_mpm/utils/dependency_cache.py +3 -1
  250. claude_mpm/utils/gitignore.py +241 -0
  251. claude_mpm/utils/log_cleanup.py +3 -3
  252. claude_mpm/utils/progress.py +383 -0
  253. claude_mpm/utils/robust_installer.py +3 -5
  254. claude_mpm/utils/structured_questions.py +619 -0
  255. {claude_mpm-4.21.0.dist-info → claude_mpm-5.0.2.dist-info}/METADATA +429 -59
  256. {claude_mpm-4.21.0.dist-info → claude_mpm-5.0.2.dist-info}/RECORD +264 -427
  257. claude_mpm/agents/templates/.claude-mpm/memories/README.md +0 -17
  258. claude_mpm/agents/templates/.claude-mpm/memories/engineer_memories.md +0 -3
  259. claude_mpm/agents/templates/agent-manager.json +0 -273
  260. claude_mpm/agents/templates/agentic-coder-optimizer.json +0 -248
  261. claude_mpm/agents/templates/api_qa.json +0 -180
  262. claude_mpm/agents/templates/clerk-ops.json +0 -235
  263. claude_mpm/agents/templates/code_analyzer.json +0 -101
  264. claude_mpm/agents/templates/content-agent.json +0 -358
  265. claude_mpm/agents/templates/dart_engineer.json +0 -307
  266. claude_mpm/agents/templates/data_engineer.json +0 -225
  267. claude_mpm/agents/templates/documentation.json +0 -211
  268. claude_mpm/agents/templates/engineer.json +0 -210
  269. claude_mpm/agents/templates/gcp_ops_agent.json +0 -253
  270. claude_mpm/agents/templates/golang_engineer.json +0 -270
  271. claude_mpm/agents/templates/imagemagick.json +0 -264
  272. claude_mpm/agents/templates/java_engineer.json +0 -346
  273. claude_mpm/agents/templates/local_ops_agent.json +0 -1840
  274. claude_mpm/agents/templates/logs/prompts/agent_engineer_20250826_014258_728.md +0 -39
  275. claude_mpm/agents/templates/logs/prompts/agent_engineer_20250901_010124_142.md +0 -400
  276. claude_mpm/agents/templates/memory_manager.json +0 -158
  277. claude_mpm/agents/templates/nextjs_engineer.json +0 -285
  278. claude_mpm/agents/templates/ops.json +0 -185
  279. claude_mpm/agents/templates/php-engineer.json +0 -287
  280. claude_mpm/agents/templates/product_owner.json +0 -338
  281. claude_mpm/agents/templates/project_organizer.json +0 -140
  282. claude_mpm/agents/templates/prompt-engineer.json +0 -737
  283. claude_mpm/agents/templates/python_engineer.json +0 -387
  284. claude_mpm/agents/templates/qa.json +0 -242
  285. claude_mpm/agents/templates/react_engineer.json +0 -238
  286. claude_mpm/agents/templates/refactoring_engineer.json +0 -276
  287. claude_mpm/agents/templates/research.json +0 -188
  288. claude_mpm/agents/templates/ruby-engineer.json +0 -280
  289. claude_mpm/agents/templates/rust_engineer.json +0 -275
  290. claude_mpm/agents/templates/security.json +0 -202
  291. claude_mpm/agents/templates/svelte-engineer.json +0 -225
  292. claude_mpm/agents/templates/ticketing.json +0 -177
  293. claude_mpm/agents/templates/typescript_engineer.json +0 -285
  294. claude_mpm/agents/templates/vercel_ops_agent.json +0 -412
  295. claude_mpm/agents/templates/version_control.json +0 -157
  296. claude_mpm/agents/templates/web_qa.json +0 -399
  297. claude_mpm/agents/templates/web_ui.json +0 -189
  298. claude_mpm/commands/mpm-tickets.md +0 -102
  299. claude_mpm/dashboard/.claude-mpm/socketio-instances.json +0 -1
  300. claude_mpm/dashboard/react/components/DataInspector/DataInspector.module.css +0 -188
  301. claude_mpm/dashboard/react/components/EventViewer/EventViewer.module.css +0 -156
  302. claude_mpm/dashboard/react/components/shared/ConnectionStatus.module.css +0 -38
  303. claude_mpm/dashboard/react/components/shared/FilterBar.module.css +0 -92
  304. claude_mpm/dashboard/static/archive/activity_dashboard_fixed.html +0 -248
  305. claude_mpm/dashboard/static/archive/activity_dashboard_test.html +0 -61
  306. claude_mpm/dashboard/static/archive/test_activity_connection.html +0 -179
  307. claude_mpm/dashboard/static/archive/test_claude_tree_tab.html +0 -68
  308. claude_mpm/dashboard/static/archive/test_dashboard.html +0 -409
  309. claude_mpm/dashboard/static/archive/test_dashboard_fixed.html +0 -519
  310. claude_mpm/dashboard/static/archive/test_dashboard_verification.html +0 -181
  311. claude_mpm/dashboard/static/archive/test_file_data.html +0 -315
  312. claude_mpm/dashboard/static/archive/test_file_tree_empty_state.html +0 -243
  313. claude_mpm/dashboard/static/archive/test_file_tree_fix.html +0 -234
  314. claude_mpm/dashboard/static/archive/test_file_tree_rename.html +0 -117
  315. claude_mpm/dashboard/static/archive/test_file_tree_tab.html +0 -115
  316. claude_mpm/dashboard/static/archive/test_file_viewer.html +0 -224
  317. claude_mpm/dashboard/static/archive/test_final_activity.html +0 -220
  318. claude_mpm/dashboard/static/archive/test_tab_fix.html +0 -139
  319. claude_mpm/dashboard/static/built/assets/events.DjpNxWNo.css +0 -1
  320. claude_mpm/dashboard/static/built/components/activity-tree.js +0 -2
  321. claude_mpm/dashboard/static/built/components/agent-hierarchy.js +0 -777
  322. claude_mpm/dashboard/static/built/components/agent-inference.js +0 -2
  323. claude_mpm/dashboard/static/built/components/build-tracker.js +0 -333
  324. claude_mpm/dashboard/static/built/components/code-simple.js +0 -857
  325. claude_mpm/dashboard/static/built/components/code-tree/tree-breadcrumb.js +0 -353
  326. claude_mpm/dashboard/static/built/components/code-tree/tree-constants.js +0 -235
  327. claude_mpm/dashboard/static/built/components/code-tree/tree-search.js +0 -409
  328. claude_mpm/dashboard/static/built/components/code-tree/tree-utils.js +0 -435
  329. claude_mpm/dashboard/static/built/components/code-tree.js +0 -2
  330. claude_mpm/dashboard/static/built/components/code-viewer.js +0 -2
  331. claude_mpm/dashboard/static/built/components/connection-debug.js +0 -654
  332. claude_mpm/dashboard/static/built/components/diff-viewer.js +0 -891
  333. claude_mpm/dashboard/static/built/components/event-processor.js +0 -2
  334. claude_mpm/dashboard/static/built/components/event-viewer.js +0 -2
  335. claude_mpm/dashboard/static/built/components/export-manager.js +0 -2
  336. claude_mpm/dashboard/static/built/components/file-change-tracker.js +0 -443
  337. claude_mpm/dashboard/static/built/components/file-change-viewer.js +0 -690
  338. claude_mpm/dashboard/static/built/components/file-tool-tracker.js +0 -2
  339. claude_mpm/dashboard/static/built/components/file-viewer.js +0 -2
  340. claude_mpm/dashboard/static/built/components/hud-library-loader.js +0 -2
  341. claude_mpm/dashboard/static/built/components/hud-manager.js +0 -2
  342. claude_mpm/dashboard/static/built/components/hud-visualizer.js +0 -2
  343. claude_mpm/dashboard/static/built/components/module-viewer.js +0 -2
  344. claude_mpm/dashboard/static/built/components/nav-bar.js +0 -145
  345. claude_mpm/dashboard/static/built/components/page-structure.js +0 -429
  346. claude_mpm/dashboard/static/built/components/session-manager.js +0 -2
  347. claude_mpm/dashboard/static/built/components/socket-manager.js +0 -2
  348. claude_mpm/dashboard/static/built/components/ui-state-manager.js +0 -2
  349. claude_mpm/dashboard/static/built/components/unified-data-viewer.js +0 -2
  350. claude_mpm/dashboard/static/built/components/working-directory.js +0 -2
  351. claude_mpm/dashboard/static/built/connection-manager.js +0 -536
  352. claude_mpm/dashboard/static/built/dashboard.js +0 -2
  353. claude_mpm/dashboard/static/built/extension-error-handler.js +0 -164
  354. claude_mpm/dashboard/static/built/react/events.js +0 -30
  355. claude_mpm/dashboard/static/built/shared/dom-helpers.js +0 -396
  356. claude_mpm/dashboard/static/built/shared/event-bus.js +0 -330
  357. claude_mpm/dashboard/static/built/shared/event-filter-service.js +0 -540
  358. claude_mpm/dashboard/static/built/shared/logger.js +0 -385
  359. claude_mpm/dashboard/static/built/shared/page-structure.js +0 -249
  360. claude_mpm/dashboard/static/built/shared/tooltip-service.js +0 -253
  361. claude_mpm/dashboard/static/built/socket-client.js +0 -2
  362. claude_mpm/dashboard/static/built/tab-isolation-fix.js +0 -185
  363. claude_mpm/dashboard/static/dist/assets/events.DjpNxWNo.css +0 -1
  364. claude_mpm/dashboard/static/dist/components/activity-tree.js +0 -2
  365. claude_mpm/dashboard/static/dist/components/agent-inference.js +0 -2
  366. claude_mpm/dashboard/static/dist/components/code-tree.js +0 -2
  367. claude_mpm/dashboard/static/dist/components/code-viewer.js +0 -2
  368. claude_mpm/dashboard/static/dist/components/event-processor.js +0 -2
  369. claude_mpm/dashboard/static/dist/components/event-viewer.js +0 -2
  370. claude_mpm/dashboard/static/dist/components/export-manager.js +0 -2
  371. claude_mpm/dashboard/static/dist/components/file-tool-tracker.js +0 -2
  372. claude_mpm/dashboard/static/dist/components/file-viewer.js +0 -2
  373. claude_mpm/dashboard/static/dist/components/hud-library-loader.js +0 -2
  374. claude_mpm/dashboard/static/dist/components/hud-manager.js +0 -2
  375. claude_mpm/dashboard/static/dist/components/hud-visualizer.js +0 -2
  376. claude_mpm/dashboard/static/dist/components/module-viewer.js +0 -2
  377. claude_mpm/dashboard/static/dist/components/session-manager.js +0 -2
  378. claude_mpm/dashboard/static/dist/components/socket-manager.js +0 -2
  379. claude_mpm/dashboard/static/dist/components/ui-state-manager.js +0 -2
  380. claude_mpm/dashboard/static/dist/components/unified-data-viewer.js +0 -2
  381. claude_mpm/dashboard/static/dist/components/working-directory.js +0 -2
  382. claude_mpm/dashboard/static/dist/dashboard.js +0 -2
  383. claude_mpm/dashboard/static/dist/react/events.js +0 -30
  384. claude_mpm/dashboard/static/dist/socket-client.js +0 -2
  385. claude_mpm/dashboard/static/events.html +0 -607
  386. claude_mpm/dashboard/static/index.html +0 -635
  387. claude_mpm/dashboard/static/js/shared/dom-helpers.js +0 -396
  388. claude_mpm/dashboard/static/js/shared/event-bus.js +0 -330
  389. claude_mpm/dashboard/static/js/shared/logger.js +0 -385
  390. claude_mpm/dashboard/static/js/shared/tooltip-service.js +0 -253
  391. claude_mpm/dashboard/static/js/stores/dashboard-store.js +0 -562
  392. claude_mpm/dashboard/static/legacy/activity.html +0 -736
  393. claude_mpm/dashboard/static/legacy/agents.html +0 -786
  394. claude_mpm/dashboard/static/legacy/files.html +0 -747
  395. claude_mpm/dashboard/static/legacy/tools.html +0 -831
  396. claude_mpm/dashboard/static/monitors.html +0 -431
  397. claude_mpm/dashboard/static/production/events.html +0 -659
  398. claude_mpm/dashboard/static/production/main.html +0 -698
  399. claude_mpm/dashboard/static/production/monitors.html +0 -483
  400. claude_mpm/dashboard/static/test-archive/dashboard.html +0 -635
  401. claude_mpm/dashboard/static/test-archive/debug-events.html +0 -147
  402. claude_mpm/dashboard/static/test-archive/test-navigation.html +0 -256
  403. claude_mpm/dashboard/static/test-archive/test-react-exports.html +0 -180
  404. claude_mpm/dashboard/static/test-archive/test_debug.html +0 -25
  405. claude_mpm/skills/bundled/collaboration/brainstorming/SKILL.md +0 -79
  406. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/SKILL.md +0 -178
  407. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/agent-prompts.md +0 -577
  408. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/coordination-patterns.md +0 -467
  409. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/examples.md +0 -537
  410. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/troubleshooting.md +0 -730
  411. claude_mpm/skills/bundled/collaboration/requesting-code-review/SKILL.md +0 -112
  412. claude_mpm/skills/bundled/collaboration/requesting-code-review/references/code-reviewer-template.md +0 -146
  413. claude_mpm/skills/bundled/collaboration/requesting-code-review/references/review-examples.md +0 -412
  414. claude_mpm/skills/bundled/collaboration/writing-plans/SKILL.md +0 -81
  415. claude_mpm/skills/bundled/collaboration/writing-plans/references/best-practices.md +0 -362
  416. claude_mpm/skills/bundled/collaboration/writing-plans/references/plan-structure-templates.md +0 -312
  417. claude_mpm/skills/bundled/debugging/root-cause-tracing/SKILL.md +0 -152
  418. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/advanced-techniques.md +0 -668
  419. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/examples.md +0 -587
  420. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/integration.md +0 -438
  421. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/tracing-techniques.md +0 -391
  422. claude_mpm/skills/bundled/debugging/systematic-debugging/CREATION-LOG.md +0 -119
  423. claude_mpm/skills/bundled/debugging/systematic-debugging/SKILL.md +0 -148
  424. claude_mpm/skills/bundled/debugging/systematic-debugging/references/anti-patterns.md +0 -483
  425. claude_mpm/skills/bundled/debugging/systematic-debugging/references/examples.md +0 -452
  426. claude_mpm/skills/bundled/debugging/systematic-debugging/references/troubleshooting.md +0 -449
  427. claude_mpm/skills/bundled/debugging/systematic-debugging/references/workflow.md +0 -411
  428. claude_mpm/skills/bundled/debugging/systematic-debugging/test-academic.md +0 -14
  429. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-1.md +0 -58
  430. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-2.md +0 -68
  431. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-3.md +0 -69
  432. claude_mpm/skills/bundled/debugging/verification-before-completion/SKILL.md +0 -131
  433. claude_mpm/skills/bundled/debugging/verification-before-completion/references/gate-function.md +0 -325
  434. claude_mpm/skills/bundled/debugging/verification-before-completion/references/integration-and-workflows.md +0 -490
  435. claude_mpm/skills/bundled/debugging/verification-before-completion/references/red-flags-and-failures.md +0 -425
  436. claude_mpm/skills/bundled/debugging/verification-before-completion/references/verification-patterns.md +0 -499
  437. claude_mpm/skills/bundled/main/artifacts-builder/SKILL.md +0 -86
  438. claude_mpm/skills/bundled/main/internal-comms/SKILL.md +0 -43
  439. claude_mpm/skills/bundled/main/internal-comms/examples/3p-updates.md +0 -47
  440. claude_mpm/skills/bundled/main/internal-comms/examples/company-newsletter.md +0 -65
  441. claude_mpm/skills/bundled/main/internal-comms/examples/faq-answers.md +0 -30
  442. claude_mpm/skills/bundled/main/internal-comms/examples/general-comms.md +0 -16
  443. claude_mpm/skills/bundled/main/mcp-builder/SKILL.md +0 -160
  444. claude_mpm/skills/bundled/main/mcp-builder/reference/design_principles.md +0 -412
  445. claude_mpm/skills/bundled/main/mcp-builder/reference/evaluation.md +0 -602
  446. claude_mpm/skills/bundled/main/mcp-builder/reference/mcp_best_practices.md +0 -915
  447. claude_mpm/skills/bundled/main/mcp-builder/reference/node_mcp_server.md +0 -916
  448. claude_mpm/skills/bundled/main/mcp-builder/reference/python_mcp_server.md +0 -752
  449. claude_mpm/skills/bundled/main/mcp-builder/reference/workflow.md +0 -1237
  450. claude_mpm/skills/bundled/main/skill-creator/SKILL.md +0 -189
  451. claude_mpm/skills/bundled/main/skill-creator/references/best-practices.md +0 -500
  452. claude_mpm/skills/bundled/main/skill-creator/references/creation-workflow.md +0 -464
  453. claude_mpm/skills/bundled/main/skill-creator/references/examples.md +0 -619
  454. claude_mpm/skills/bundled/main/skill-creator/references/progressive-disclosure.md +0 -437
  455. claude_mpm/skills/bundled/main/skill-creator/references/skill-structure.md +0 -231
  456. claude_mpm/skills/bundled/php/espocrm-development/SKILL.md +0 -170
  457. claude_mpm/skills/bundled/php/espocrm-development/references/architecture.md +0 -602
  458. claude_mpm/skills/bundled/php/espocrm-development/references/common-tasks.md +0 -821
  459. claude_mpm/skills/bundled/php/espocrm-development/references/development-workflow.md +0 -742
  460. claude_mpm/skills/bundled/php/espocrm-development/references/frontend-customization.md +0 -726
  461. claude_mpm/skills/bundled/php/espocrm-development/references/hooks-and-services.md +0 -764
  462. claude_mpm/skills/bundled/php/espocrm-development/references/testing-debugging.md +0 -831
  463. claude_mpm/skills/bundled/rust/desktop-applications/SKILL.md +0 -226
  464. claude_mpm/skills/bundled/rust/desktop-applications/references/architecture-patterns.md +0 -901
  465. claude_mpm/skills/bundled/rust/desktop-applications/references/native-gui-frameworks.md +0 -901
  466. claude_mpm/skills/bundled/rust/desktop-applications/references/platform-integration.md +0 -775
  467. claude_mpm/skills/bundled/rust/desktop-applications/references/state-management.md +0 -937
  468. claude_mpm/skills/bundled/rust/desktop-applications/references/tauri-framework.md +0 -770
  469. claude_mpm/skills/bundled/rust/desktop-applications/references/testing-deployment.md +0 -961
  470. claude_mpm/skills/bundled/testing/condition-based-waiting/SKILL.md +0 -119
  471. claude_mpm/skills/bundled/testing/condition-based-waiting/references/patterns-and-implementation.md +0 -253
  472. claude_mpm/skills/bundled/testing/test-driven-development/SKILL.md +0 -145
  473. claude_mpm/skills/bundled/testing/test-driven-development/references/anti-patterns.md +0 -543
  474. claude_mpm/skills/bundled/testing/test-driven-development/references/examples.md +0 -741
  475. claude_mpm/skills/bundled/testing/test-driven-development/references/integration.md +0 -470
  476. claude_mpm/skills/bundled/testing/test-driven-development/references/philosophy.md +0 -458
  477. claude_mpm/skills/bundled/testing/test-driven-development/references/workflow.md +0 -639
  478. claude_mpm/skills/bundled/testing/testing-anti-patterns/SKILL.md +0 -140
  479. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/completeness-anti-patterns.md +0 -572
  480. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/core-anti-patterns.md +0 -411
  481. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/detection-guide.md +0 -569
  482. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/tdd-connection.md +0 -695
  483. claude_mpm/skills/bundled/testing/webapp-testing/SKILL.md +0 -184
  484. claude_mpm/skills/bundled/testing/webapp-testing/decision-tree.md +0 -459
  485. claude_mpm/skills/bundled/testing/webapp-testing/playwright-patterns.md +0 -479
  486. claude_mpm/skills/bundled/testing/webapp-testing/reconnaissance-pattern.md +0 -687
  487. claude_mpm/skills/bundled/testing/webapp-testing/server-management.md +0 -758
  488. claude_mpm/skills/bundled/testing/webapp-testing/troubleshooting.md +0 -868
  489. claude_mpm/tools/code_tree_analyzer.py +0 -1825
  490. /claude_mpm/agents/templates/{git_file_tracking.md → git-file-tracking.md} +0 -0
  491. /claude_mpm/agents/templates/{pm_examples.md → pm-examples.md} +0 -0
  492. /claude_mpm/agents/templates/{response_format.md → response-format.md} +0 -0
  493. /claude_mpm/agents/templates/{validation_templates.md → validation-templates.md} +0 -0
  494. {claude_mpm-4.21.0.dist-info → claude_mpm-5.0.2.dist-info}/WHEEL +0 -0
  495. {claude_mpm-4.21.0.dist-info → claude_mpm-5.0.2.dist-info}/entry_points.txt +0 -0
  496. {claude_mpm-4.21.0.dist-info → claude_mpm-5.0.2.dist-info}/licenses/LICENSE +0 -0
  497. {claude_mpm-4.21.0.dist-info → claude_mpm-5.0.2.dist-info}/top_level.txt +0 -0
@@ -1,937 +0,0 @@
1
- # State Management
2
-
3
- Comprehensive guide to managing application state in Rust desktop applications, from simple local state to complex async operations and multi-window synchronization.
4
-
5
- ## State Management Strategies
6
-
7
- ### Local State (Single Component)
8
-
9
- Simplest form - state lives within a single component or module.
10
-
11
- ```rust
12
- // egui example
13
- struct MyApp {
14
- counter: i32,
15
- text: String,
16
- selected: Option<usize>,
17
- }
18
-
19
- impl eframe::App for MyApp {
20
- fn update(&mut self, ctx: &egui::Context, _frame: &mut eframe::Frame) {
21
- egui::CentralPanel::default().show(ctx, |ui| {
22
- ui.label(format!("Counter: {}", self.counter));
23
-
24
- if ui.button("Increment").clicked() {
25
- self.counter += 1;
26
- }
27
-
28
- ui.text_edit_singleline(&mut self.text);
29
- });
30
- }
31
- }
32
- ```
33
-
34
- **Tauri example:**
35
- ```rust
36
- use std::sync::Mutex;
37
-
38
- struct AppState {
39
- counter: Mutex<i32>,
40
- }
41
-
42
- #[tauri::command]
43
- fn increment(state: tauri::State<AppState>) -> i32 {
44
- let mut counter = state.counter.lock().unwrap();
45
- *counter += 1;
46
- *counter
47
- }
48
-
49
- #[tauri::command]
50
- fn get_counter(state: tauri::State<AppState>) -> i32 {
51
- *state.counter.lock().unwrap()
52
- }
53
-
54
- fn main() {
55
- tauri::Builder::default()
56
- .manage(AppState {
57
- counter: Mutex::new(0),
58
- })
59
- .invoke_handler(tauri::generate_handler![increment, get_counter])
60
- .run(tauri::generate_context!())
61
- .expect("error while running tauri application");
62
- }
63
- ```
64
-
65
- ### Shared State with Arc<Mutex<T>>
66
-
67
- Thread-safe shared state for multi-threaded applications.
68
-
69
- ```rust
70
- use std::sync::{Arc, Mutex};
71
-
72
- #[derive(Clone)]
73
- struct SharedState {
74
- data: Arc<Mutex<AppData>>,
75
- }
76
-
77
- struct AppData {
78
- users: Vec<User>,
79
- settings: Settings,
80
- }
81
-
82
- impl SharedState {
83
- fn new() -> Self {
84
- Self {
85
- data: Arc::new(Mutex::new(AppData {
86
- users: Vec::new(),
87
- settings: Settings::default(),
88
- })),
89
- }
90
- }
91
-
92
- fn add_user(&self, user: User) {
93
- let mut data = self.data.lock().unwrap();
94
- data.users.push(user);
95
- }
96
-
97
- fn get_users(&self) -> Vec<User> {
98
- let data = self.data.lock().unwrap();
99
- data.users.clone()
100
- }
101
- }
102
-
103
- // Tauri commands
104
- #[tauri::command]
105
- fn add_user(state: tauri::State<SharedState>, name: String, email: String) {
106
- let user = User {
107
- id: generate_id(),
108
- name,
109
- email,
110
- };
111
- state.add_user(user);
112
- }
113
-
114
- #[tauri::command]
115
- fn get_users(state: tauri::State<SharedState>) -> Vec<User> {
116
- state.get_users()
117
- }
118
- ```
119
-
120
- ### RwLock for Read-Heavy Workloads
121
-
122
- Better performance when reads outnumber writes.
123
-
124
- ```rust
125
- use std::sync::{Arc, RwLock};
126
-
127
- struct AppState {
128
- cache: Arc<RwLock<HashMap<String, String>>>,
129
- }
130
-
131
- impl AppState {
132
- fn new() -> Self {
133
- Self {
134
- cache: Arc::new(RwLock::new(HashMap::new())),
135
- }
136
- }
137
-
138
- // Multiple readers can access simultaneously
139
- fn get(&self, key: &str) -> Option<String> {
140
- let cache = self.cache.read().unwrap();
141
- cache.get(key).cloned()
142
- }
143
-
144
- // Exclusive write access
145
- fn set(&self, key: String, value: String) {
146
- let mut cache = self.cache.write().unwrap();
147
- cache.insert(key, value);
148
- }
149
-
150
- // Bulk read operation
151
- fn get_all(&self) -> HashMap<String, String> {
152
- let cache = self.cache.read().unwrap();
153
- cache.clone()
154
- }
155
- }
156
-
157
- #[tauri::command]
158
- fn cache_get(state: tauri::State<AppState>, key: String) -> Option<String> {
159
- state.get(&key)
160
- }
161
-
162
- #[tauri::command]
163
- fn cache_set(state: tauri::State<AppState>, key: String, value: String) {
164
- state.set(key, value);
165
- }
166
- ```
167
-
168
- ## Async Runtime Integration
169
-
170
- ### Tokio Integration with Tauri
171
-
172
- ```rust
173
- use tokio::sync::RwLock as TokioRwLock;
174
- use std::sync::Arc;
175
-
176
- struct AsyncState {
177
- data: Arc<TokioRwLock<AppData>>,
178
- }
179
-
180
- #[derive(Clone)]
181
- struct AppData {
182
- items: Vec<Item>,
183
- loading: bool,
184
- }
185
-
186
- impl AsyncState {
187
- fn new() -> Self {
188
- Self {
189
- data: Arc::new(TokioRwLock::new(AppData {
190
- items: Vec::new(),
191
- loading: false,
192
- })),
193
- }
194
- }
195
-
196
- async fn fetch_items(&self) -> Result<Vec<Item>, String> {
197
- // Set loading state
198
- {
199
- let mut data = self.data.write().await;
200
- data.loading = true;
201
- }
202
-
203
- // Perform async operation
204
- let items = fetch_from_api().await.map_err(|e| e.to_string())?;
205
-
206
- // Update state
207
- {
208
- let mut data = self.data.write().await;
209
- data.items = items.clone();
210
- data.loading = false;
211
- }
212
-
213
- Ok(items)
214
- }
215
-
216
- async fn get_items(&self) -> Vec<Item> {
217
- let data = self.data.read().await;
218
- data.items.clone()
219
- }
220
-
221
- async fn is_loading(&self) -> bool {
222
- let data = self.data.read().await;
223
- data.loading
224
- }
225
- }
226
-
227
- #[tauri::command]
228
- async fn fetch_items(state: tauri::State<'_, AsyncState>) -> Result<Vec<Item>, String> {
229
- state.fetch_items().await
230
- }
231
-
232
- #[tauri::command]
233
- async fn get_items(state: tauri::State<'_, AsyncState>) -> Vec<Item> {
234
- state.get_items().await
235
- }
236
-
237
- async fn fetch_from_api() -> Result<Vec<Item>, Box<dyn std::error::Error>> {
238
- use reqwest;
239
-
240
- let response = reqwest::get("https://api.example.com/items")
241
- .await?
242
- .json::<Vec<Item>>()
243
- .await?;
244
-
245
- Ok(response)
246
- }
247
- ```
248
-
249
- ### Background Tasks and Channels
250
-
251
- ```rust
252
- use tokio::sync::mpsc;
253
- use tokio::time::{interval, Duration};
254
-
255
- struct BackgroundWorker {
256
- tx: mpsc::UnboundedSender<WorkerMessage>,
257
- }
258
-
259
- enum WorkerMessage {
260
- ProcessData(String),
261
- Stop,
262
- }
263
-
264
- impl BackgroundWorker {
265
- fn new(app_handle: tauri::AppHandle) -> Self {
266
- let (tx, mut rx) = mpsc::unbounded_channel();
267
-
268
- tokio::spawn(async move {
269
- let mut ticker = interval(Duration::from_secs(1));
270
-
271
- loop {
272
- tokio::select! {
273
- _ = ticker.tick() => {
274
- // Periodic task
275
- let _ = app_handle.emit("tick", "Periodic update");
276
- }
277
- Some(msg) = rx.recv() => {
278
- match msg {
279
- WorkerMessage::ProcessData(data) => {
280
- // Process data
281
- println!("Processing: {}", data);
282
- let _ = app_handle.emit("data-processed", data);
283
- }
284
- WorkerMessage::Stop => {
285
- println!("Stopping worker");
286
- break;
287
- }
288
- }
289
- }
290
- }
291
- }
292
- });
293
-
294
- Self { tx }
295
- }
296
-
297
- fn send(&self, msg: WorkerMessage) {
298
- let _ = self.tx.send(msg);
299
- }
300
- }
301
-
302
- #[tauri::command]
303
- fn process_data(worker: tauri::State<BackgroundWorker>, data: String) {
304
- worker.send(WorkerMessage::ProcessData(data));
305
- }
306
-
307
- fn main() {
308
- tauri::Builder::default()
309
- .setup(|app| {
310
- let worker = BackgroundWorker::new(app.handle());
311
- app.manage(worker);
312
- Ok(())
313
- })
314
- .invoke_handler(tauri::generate_handler![process_data])
315
- .run(tauri::generate_context!())
316
- .expect("error while running tauri application");
317
- }
318
- ```
319
-
320
- ## Message Passing Patterns
321
-
322
- ### Command-Query Separation
323
-
324
- ```rust
325
- use tokio::sync::mpsc;
326
-
327
- // Commands (modify state)
328
- enum Command {
329
- AddUser { name: String, email: String },
330
- RemoveUser { id: u64 },
331
- UpdateSettings { key: String, value: String },
332
- }
333
-
334
- // Queries (read state)
335
- enum Query {
336
- GetUser { id: u64, response: oneshot::Sender<Option<User>> },
337
- GetAllUsers { response: oneshot::Sender<Vec<User>> },
338
- GetSettings { response: oneshot::Sender<Settings> },
339
- }
340
-
341
- struct StateManager {
342
- command_tx: mpsc::UnboundedSender<Command>,
343
- query_tx: mpsc::UnboundedSender<Query>,
344
- }
345
-
346
- impl StateManager {
347
- fn new() -> Self {
348
- let (command_tx, mut command_rx) = mpsc::unbounded_channel();
349
- let (query_tx, mut query_rx) = mpsc::unbounded_channel();
350
-
351
- // State lives in this task
352
- tokio::spawn(async move {
353
- let mut state = AppState::new();
354
-
355
- loop {
356
- tokio::select! {
357
- Some(cmd) = command_rx.recv() => {
358
- match cmd {
359
- Command::AddUser { name, email } => {
360
- state.add_user(User { id: generate_id(), name, email });
361
- }
362
- Command::RemoveUser { id } => {
363
- state.remove_user(id);
364
- }
365
- Command::UpdateSettings { key, value } => {
366
- state.update_setting(key, value);
367
- }
368
- }
369
- }
370
- Some(query) = query_rx.recv() => {
371
- match query {
372
- Query::GetUser { id, response } => {
373
- let _ = response.send(state.get_user(id));
374
- }
375
- Query::GetAllUsers { response } => {
376
- let _ = response.send(state.get_all_users());
377
- }
378
- Query::GetSettings { response } => {
379
- let _ = response.send(state.get_settings());
380
- }
381
- }
382
- }
383
- }
384
- }
385
- });
386
-
387
- Self { command_tx, query_tx }
388
- }
389
-
390
- fn send_command(&self, cmd: Command) {
391
- let _ = self.command_tx.send(cmd);
392
- }
393
-
394
- async fn query_user(&self, id: u64) -> Option<User> {
395
- let (tx, rx) = oneshot::channel();
396
- let _ = self.query_tx.send(Query::GetUser { id, response: tx });
397
- rx.await.unwrap()
398
- }
399
-
400
- async fn query_all_users(&self) -> Vec<User> {
401
- let (tx, rx) = oneshot::channel();
402
- let _ = self.query_tx.send(Query::GetAllUsers { response: tx });
403
- rx.await.unwrap()
404
- }
405
- }
406
-
407
- // Tauri commands
408
- #[tauri::command]
409
- fn add_user(manager: tauri::State<StateManager>, name: String, email: String) {
410
- manager.send_command(Command::AddUser { name, email });
411
- }
412
-
413
- #[tauri::command]
414
- async fn get_user(manager: tauri::State<'_, StateManager>, id: u64) -> Option<User> {
415
- manager.query_user(id).await
416
- }
417
- ```
418
-
419
- ### Actor Pattern
420
-
421
- ```rust
422
- use tokio::sync::mpsc;
423
-
424
- trait Actor {
425
- type Message;
426
-
427
- fn handle(&mut self, msg: Self::Message);
428
- }
429
-
430
- struct ActorHandle<M> {
431
- tx: mpsc::UnboundedSender<M>,
432
- }
433
-
434
- impl<M: Send + 'static> ActorHandle<M> {
435
- fn new<A>(mut actor: A) -> Self
436
- where
437
- A: Actor<Message = M> + Send + 'static,
438
- {
439
- let (tx, mut rx) = mpsc::unbounded_channel();
440
-
441
- tokio::spawn(async move {
442
- while let Some(msg) = rx.recv().await {
443
- actor.handle(msg);
444
- }
445
- });
446
-
447
- Self { tx }
448
- }
449
-
450
- fn send(&self, msg: M) {
451
- let _ = self.tx.send(msg);
452
- }
453
- }
454
-
455
- // Example actor
456
- struct UserActor {
457
- users: HashMap<u64, User>,
458
- }
459
-
460
- enum UserMessage {
461
- Add(User),
462
- Remove(u64),
463
- Get { id: u64, response: oneshot::Sender<Option<User>> },
464
- }
465
-
466
- impl Actor for UserActor {
467
- type Message = UserMessage;
468
-
469
- fn handle(&mut self, msg: Self::Message) {
470
- match msg {
471
- UserMessage::Add(user) => {
472
- self.users.insert(user.id, user);
473
- }
474
- UserMessage::Remove(id) => {
475
- self.users.remove(&id);
476
- }
477
- UserMessage::Get { id, response } => {
478
- let user = self.users.get(&id).cloned();
479
- let _ = response.send(user);
480
- }
481
- }
482
- }
483
- }
484
-
485
- // Usage
486
- fn setup_actors() -> ActorHandle<UserMessage> {
487
- let actor = UserActor {
488
- users: HashMap::new(),
489
- };
490
- ActorHandle::new(actor)
491
- }
492
- ```
493
-
494
- ## Reactive State Patterns
495
-
496
- ### Observable State with Signals
497
-
498
- ```rust
499
- use std::sync::{Arc, Mutex};
500
- use std::collections::HashMap;
501
-
502
- type Listener<T> = Box<dyn Fn(&T) + Send + Sync>;
503
-
504
- struct Signal<T: Clone> {
505
- value: Arc<Mutex<T>>,
506
- listeners: Arc<Mutex<Vec<Listener<T>>>>,
507
- }
508
-
509
- impl<T: Clone + Send + Sync + 'static> Signal<T> {
510
- fn new(initial: T) -> Self {
511
- Self {
512
- value: Arc::new(Mutex::new(initial)),
513
- listeners: Arc::new(Mutex::new(Vec::new())),
514
- }
515
- }
516
-
517
- fn get(&self) -> T {
518
- self.value.lock().unwrap().clone()
519
- }
520
-
521
- fn set(&self, new_value: T) {
522
- {
523
- let mut value = self.value.lock().unwrap();
524
- *value = new_value.clone();
525
- }
526
-
527
- // Notify listeners
528
- let listeners = self.listeners.lock().unwrap();
529
- for listener in listeners.iter() {
530
- listener(&new_value);
531
- }
532
- }
533
-
534
- fn update<F>(&self, f: F)
535
- where
536
- F: FnOnce(&mut T),
537
- {
538
- let new_value = {
539
- let mut value = self.value.lock().unwrap();
540
- f(&mut value);
541
- value.clone()
542
- };
543
-
544
- // Notify listeners
545
- let listeners = self.listeners.lock().unwrap();
546
- for listener in listeners.iter() {
547
- listener(&new_value);
548
- }
549
- }
550
-
551
- fn subscribe<F>(&self, listener: F)
552
- where
553
- F: Fn(&T) + Send + Sync + 'static,
554
- {
555
- let mut listeners = self.listeners.lock().unwrap();
556
- listeners.push(Box::new(listener));
557
- }
558
- }
559
-
560
- // Example usage
561
- struct AppState {
562
- counter: Signal<i32>,
563
- username: Signal<String>,
564
- }
565
-
566
- impl AppState {
567
- fn new() -> Self {
568
- Self {
569
- counter: Signal::new(0),
570
- username: Signal::new(String::from("Guest")),
571
- }
572
- }
573
- }
574
-
575
- fn setup_state(app_handle: tauri::AppHandle) -> AppState {
576
- let state = AppState::new();
577
-
578
- // Subscribe to changes
579
- let handle = app_handle.clone();
580
- state.counter.subscribe(move |value| {
581
- let _ = handle.emit("counter-changed", value);
582
- });
583
-
584
- let handle = app_handle.clone();
585
- state.username.subscribe(move |value| {
586
- let _ = handle.emit("username-changed", value);
587
- });
588
-
589
- state
590
- }
591
-
592
- #[tauri::command]
593
- fn increment_counter(state: tauri::State<AppState>) {
594
- state.counter.update(|c| *c += 1);
595
- }
596
-
597
- #[tauri::command]
598
- fn set_username(state: tauri::State<AppState>, name: String) {
599
- state.username.set(name);
600
- }
601
-
602
- #[tauri::command]
603
- fn get_counter(state: tauri::State<AppState>) -> i32 {
604
- state.counter.get()
605
- }
606
- ```
607
-
608
- ### Computed Values
609
-
610
- ```rust
611
- struct Computed<T, F>
612
- where
613
- T: Clone,
614
- F: Fn() -> T,
615
- {
616
- compute: F,
617
- cached: Arc<Mutex<Option<T>>>,
618
- }
619
-
620
- impl<T: Clone, F: Fn() -> T> Computed<T, F> {
621
- fn new(compute: F) -> Self {
622
- Self {
623
- compute,
624
- cached: Arc::new(Mutex::new(None)),
625
- }
626
- }
627
-
628
- fn get(&self) -> T {
629
- let mut cached = self.cached.lock().unwrap();
630
-
631
- if let Some(value) = cached.as_ref() {
632
- value.clone()
633
- } else {
634
- let value = (self.compute)();
635
- *cached = Some(value.clone());
636
- value
637
- }
638
- }
639
-
640
- fn invalidate(&self) {
641
- let mut cached = self.cached.lock().unwrap();
642
- *cached = None;
643
- }
644
- }
645
-
646
- // Example
647
- struct TodoState {
648
- todos: Signal<Vec<Todo>>,
649
- completed_count: Computed<usize, Box<dyn Fn() -> usize + Send + Sync>>,
650
- }
651
-
652
- impl TodoState {
653
- fn new() -> Self {
654
- let todos = Signal::new(Vec::new());
655
- let todos_clone = todos.clone();
656
-
657
- let completed_count = Computed::new(Box::new(move || {
658
- todos_clone
659
- .get()
660
- .iter()
661
- .filter(|t| t.completed)
662
- .count()
663
- }));
664
-
665
- Self {
666
- todos,
667
- completed_count,
668
- }
669
- }
670
-
671
- fn add_todo(&self, todo: Todo) {
672
- self.todos.update(|todos| todos.push(todo));
673
- self.completed_count.invalidate();
674
- }
675
-
676
- fn toggle_todo(&self, id: u64) {
677
- self.todos.update(|todos| {
678
- if let Some(todo) = todos.iter_mut().find(|t| t.id == id) {
679
- todo.completed = !todo.completed;
680
- }
681
- });
682
- self.completed_count.invalidate();
683
- }
684
-
685
- fn get_completed_count(&self) -> usize {
686
- self.completed_count.get()
687
- }
688
- }
689
- ```
690
-
691
- ## Persistence
692
-
693
- ### File-Based Persistence
694
-
695
- ```rust
696
- use serde::{Deserialize, Serialize};
697
- use std::path::PathBuf;
698
-
699
- #[derive(Serialize, Deserialize, Clone)]
700
- struct AppSettings {
701
- theme: String,
702
- language: String,
703
- window_size: (u32, u32),
704
- }
705
-
706
- struct PersistedState {
707
- settings: Signal<AppSettings>,
708
- config_path: PathBuf,
709
- }
710
-
711
- impl PersistedState {
712
- fn new(config_path: PathBuf) -> Self {
713
- let settings = Self::load_settings(&config_path)
714
- .unwrap_or_else(|_| AppSettings::default());
715
-
716
- let state = Self {
717
- settings: Signal::new(settings),
718
- config_path,
719
- };
720
-
721
- // Auto-save on changes
722
- let config_path = state.config_path.clone();
723
- state.settings.subscribe(move |settings| {
724
- let _ = Self::save_settings(&config_path, settings);
725
- });
726
-
727
- state
728
- }
729
-
730
- fn load_settings(path: &PathBuf) -> Result<AppSettings, Box<dyn std::error::Error>> {
731
- let content = std::fs::read_to_string(path)?;
732
- let settings = serde_json::from_str(&content)?;
733
- Ok(settings)
734
- }
735
-
736
- fn save_settings(path: &PathBuf, settings: &AppSettings) -> Result<(), Box<dyn std::error::Error>> {
737
- let content = serde_json::to_string_pretty(settings)?;
738
- std::fs::write(path, content)?;
739
- Ok(())
740
- }
741
-
742
- fn update_settings<F>(&self, f: F)
743
- where
744
- F: FnOnce(&mut AppSettings),
745
- {
746
- self.settings.update(f);
747
- }
748
- }
749
-
750
- #[tauri::command]
751
- fn update_theme(state: tauri::State<PersistedState>, theme: String) {
752
- state.update_settings(|s| s.theme = theme);
753
- }
754
-
755
- #[tauri::command]
756
- fn get_settings(state: tauri::State<PersistedState>) -> AppSettings {
757
- state.settings.get()
758
- }
759
- ```
760
-
761
- ### Database Integration with sqlx
762
-
763
- ```rust
764
- use sqlx::{SqlitePool, FromRow};
765
-
766
- #[derive(FromRow, Serialize, Clone)]
767
- struct Note {
768
- id: i64,
769
- title: String,
770
- content: String,
771
- created_at: String,
772
- }
773
-
774
- struct DatabaseState {
775
- pool: SqlitePool,
776
- }
777
-
778
- impl DatabaseState {
779
- async fn new(database_url: &str) -> Result<Self, sqlx::Error> {
780
- let pool = SqlitePool::connect(database_url).await?;
781
-
782
- // Run migrations
783
- sqlx::query(
784
- r#"
785
- CREATE TABLE IF NOT EXISTS notes (
786
- id INTEGER PRIMARY KEY AUTOINCREMENT,
787
- title TEXT NOT NULL,
788
- content TEXT NOT NULL,
789
- created_at DATETIME DEFAULT CURRENT_TIMESTAMP
790
- )
791
- "#,
792
- )
793
- .execute(&pool)
794
- .await?;
795
-
796
- Ok(Self { pool })
797
- }
798
-
799
- async fn create_note(&self, title: String, content: String) -> Result<Note, sqlx::Error> {
800
- let note = sqlx::query_as::<_, Note>(
801
- "INSERT INTO notes (title, content) VALUES (?, ?) RETURNING *",
802
- )
803
- .bind(title)
804
- .bind(content)
805
- .fetch_one(&self.pool)
806
- .await?;
807
-
808
- Ok(note)
809
- }
810
-
811
- async fn get_all_notes(&self) -> Result<Vec<Note>, sqlx::Error> {
812
- sqlx::query_as::<_, Note>("SELECT * FROM notes ORDER BY created_at DESC")
813
- .fetch_all(&self.pool)
814
- .await
815
- }
816
-
817
- async fn update_note(&self, id: i64, title: String, content: String) -> Result<(), sqlx::Error> {
818
- sqlx::query("UPDATE notes SET title = ?, content = ? WHERE id = ?")
819
- .bind(title)
820
- .bind(content)
821
- .bind(id)
822
- .execute(&self.pool)
823
- .await?;
824
-
825
- Ok(())
826
- }
827
-
828
- async fn delete_note(&self, id: i64) -> Result<(), sqlx::Error> {
829
- sqlx::query("DELETE FROM notes WHERE id = ?")
830
- .bind(id)
831
- .execute(&self.pool)
832
- .await?;
833
-
834
- Ok(())
835
- }
836
- }
837
-
838
- #[tauri::command]
839
- async fn create_note(
840
- state: tauri::State<'_, DatabaseState>,
841
- title: String,
842
- content: String,
843
- ) -> Result<Note, String> {
844
- state
845
- .create_note(title, content)
846
- .await
847
- .map_err(|e| e.to_string())
848
- }
849
-
850
- #[tauri::command]
851
- async fn get_all_notes(state: tauri::State<'_, DatabaseState>) -> Result<Vec<Note>, String> {
852
- state.get_all_notes().await.map_err(|e| e.to_string())
853
- }
854
- ```
855
-
856
- ## Multi-Window State Sharing
857
-
858
- ```rust
859
- use std::sync::Arc;
860
- use tokio::sync::RwLock;
861
-
862
- #[derive(Clone)]
863
- struct SharedAppState {
864
- data: Arc<RwLock<GlobalData>>,
865
- }
866
-
867
- struct GlobalData {
868
- current_user: Option<User>,
869
- notifications: Vec<Notification>,
870
- }
871
-
872
- impl SharedAppState {
873
- fn new() -> Self {
874
- Self {
875
- data: Arc::new(RwLock::new(GlobalData {
876
- current_user: None,
877
- notifications: Vec::new(),
878
- })),
879
- }
880
- }
881
-
882
- async fn set_user(&self, user: User) {
883
- let mut data = self.data.write().await;
884
- data.current_user = Some(user);
885
- }
886
-
887
- async fn add_notification(&self, notification: Notification) {
888
- let mut data = self.data.write().await;
889
- data.notifications.push(notification);
890
- }
891
-
892
- async fn get_user(&self) -> Option<User> {
893
- let data = self.data.read().await;
894
- data.current_user.clone()
895
- }
896
- }
897
-
898
- // Broadcast state changes to all windows
899
- use tauri::{Emitter, Manager};
900
-
901
- #[tauri::command]
902
- async fn login_user(
903
- app: tauri::AppHandle,
904
- state: tauri::State<'_, SharedAppState>,
905
- username: String,
906
- ) -> Result<(), String> {
907
- let user = User {
908
- id: 1,
909
- name: username,
910
- email: "user@example.com".to_string(),
911
- };
912
-
913
- state.set_user(user.clone()).await;
914
-
915
- // Notify all windows
916
- app.emit("user-logged-in", &user).map_err(|e| e.to_string())?;
917
-
918
- Ok(())
919
- }
920
-
921
- // Open new window with shared state
922
- #[tauri::command]
923
- fn open_settings_window(app: tauri::AppHandle) -> Result<(), String> {
924
- tauri::WebviewWindowBuilder::new(
925
- &app,
926
- "settings",
927
- tauri::WebviewUrl::App("settings.html".into()),
928
- )
929
- .title("Settings")
930
- .build()
931
- .map_err(|e| e.to_string())?;
932
-
933
- Ok(())
934
- }
935
- ```
936
-
937
- These state management patterns provide flexibility for applications of all sizes - from simple local state to complex distributed state with persistence and multi-window synchronization.