claude-mpm 4.21.3__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 (484) 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 +11 -2
  63. claude_mpm/commands/mpm-init.md +27 -2
  64. claude_mpm/commands/mpm-monitor.md +9 -0
  65. claude_mpm/commands/{mpm-resume.md → mpm-session-resume.md} +9 -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 +10 -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/interfaces/__init__.py +1 -3
  162. claude_mpm/services/core/interfaces/health.py +1 -4
  163. claude_mpm/services/core/models/__init__.py +2 -11
  164. claude_mpm/services/diagnostics/checks/__init__.py +4 -0
  165. claude_mpm/services/diagnostics/checks/agent_check.py +0 -2
  166. claude_mpm/services/diagnostics/checks/agent_sources_check.py +577 -0
  167. claude_mpm/services/diagnostics/checks/instructions_check.py +1 -2
  168. claude_mpm/services/diagnostics/checks/mcp_check.py +0 -1
  169. claude_mpm/services/diagnostics/checks/monitor_check.py +0 -1
  170. claude_mpm/services/diagnostics/checks/skill_sources_check.py +587 -0
  171. claude_mpm/services/diagnostics/diagnostic_runner.py +9 -0
  172. claude_mpm/services/diagnostics/doctor_reporter.py +40 -10
  173. claude_mpm/services/event_bus/direct_relay.py +3 -3
  174. claude_mpm/services/event_bus/event_bus.py +36 -3
  175. claude_mpm/services/events/consumers/logging.py +1 -2
  176. claude_mpm/services/git/__init__.py +21 -0
  177. claude_mpm/services/git/git_operations_service.py +494 -0
  178. claude_mpm/services/github/__init__.py +21 -0
  179. claude_mpm/services/github/github_cli_service.py +397 -0
  180. claude_mpm/services/infrastructure/monitoring/__init__.py +1 -5
  181. claude_mpm/services/infrastructure/monitoring/aggregator.py +1 -6
  182. claude_mpm/services/infrastructure/monitoring/resources.py +1 -1
  183. claude_mpm/services/instructions/__init__.py +9 -0
  184. claude_mpm/services/instructions/instruction_cache_service.py +374 -0
  185. claude_mpm/services/local_ops/__init__.py +3 -13
  186. claude_mpm/services/local_ops/health_checks/__init__.py +1 -3
  187. claude_mpm/services/local_ops/health_manager.py +1 -4
  188. claude_mpm/services/local_ops/process_manager.py +1 -1
  189. claude_mpm/services/local_ops/resource_monitor.py +2 -2
  190. claude_mpm/services/mcp_gateway/config/configuration.py +1 -1
  191. claude_mpm/services/mcp_gateway/server/mcp_gateway.py +1 -6
  192. claude_mpm/services/mcp_gateway/server/stdio_server.py +0 -2
  193. claude_mpm/services/mcp_gateway/tools/document_summarizer.py +1 -1
  194. claude_mpm/services/mcp_gateway/tools/kuzu_memory_service.py +6 -2
  195. claude_mpm/services/memory/optimizer.py +1 -1
  196. claude_mpm/services/model/model_router.py +8 -9
  197. claude_mpm/services/monitor/daemon.py +1 -1
  198. claude_mpm/services/monitor/server.py +2 -2
  199. claude_mpm/services/native_agent_converter.py +356 -0
  200. claude_mpm/services/port_manager.py +1 -1
  201. claude_mpm/services/pr/__init__.py +14 -0
  202. claude_mpm/services/pr/pr_template_service.py +329 -0
  203. claude_mpm/services/project/documentation_manager.py +2 -1
  204. claude_mpm/services/project/toolchain_analyzer.py +3 -1
  205. claude_mpm/services/runner_configuration_service.py +1 -0
  206. claude_mpm/services/self_upgrade_service.py +165 -7
  207. claude_mpm/services/skills/__init__.py +18 -0
  208. claude_mpm/services/skills/git_skill_source_manager.py +1169 -0
  209. claude_mpm/services/skills/skill_discovery_service.py +568 -0
  210. claude_mpm/services/skills_config.py +547 -0
  211. claude_mpm/services/skills_deployer.py +955 -0
  212. claude_mpm/services/socketio/handlers/connection.py +1 -1
  213. claude_mpm/services/socketio/handlers/git.py +2 -2
  214. claude_mpm/services/socketio/server/core.py +1 -4
  215. claude_mpm/services/socketio/server/main.py +1 -3
  216. claude_mpm/services/system_instructions_service.py +1 -3
  217. claude_mpm/services/unified/analyzer_strategies/performance_analyzer.py +0 -3
  218. claude_mpm/services/unified/analyzer_strategies/security_analyzer.py +0 -1
  219. claude_mpm/services/unified/deployment_strategies/cloud_strategies.py +1 -1
  220. claude_mpm/services/unified/deployment_strategies/vercel.py +1 -5
  221. claude_mpm/services/unified/unified_deployment.py +1 -5
  222. claude_mpm/services/version_control/conflict_resolution.py +6 -4
  223. claude_mpm/services/visualization/__init__.py +1 -5
  224. claude_mpm/services/visualization/mermaid_generator.py +2 -3
  225. claude_mpm/skills/bundled/infrastructure/env-manager/scripts/validate_env.py +576 -0
  226. claude_mpm/skills/bundled/performance-profiling.md +6 -0
  227. claude_mpm/skills/bundled/testing/webapp-testing/scripts/with_server.py +2 -2
  228. claude_mpm/skills/skills_registry.py +0 -1
  229. claude_mpm/templates/questions/__init__.py +38 -0
  230. claude_mpm/templates/questions/base.py +193 -0
  231. claude_mpm/templates/questions/pr_strategy.py +311 -0
  232. claude_mpm/templates/questions/project_init.py +385 -0
  233. claude_mpm/templates/questions/ticket_mgmt.py +394 -0
  234. claude_mpm/tools/__main__.py +8 -8
  235. claude_mpm/tools/code_tree_analyzer/analysis.py +1 -1
  236. claude_mpm/utils/agent_dependency_loader.py +80 -13
  237. claude_mpm/utils/dependency_cache.py +3 -1
  238. claude_mpm/utils/gitignore.py +241 -0
  239. claude_mpm/utils/log_cleanup.py +3 -3
  240. claude_mpm/utils/progress.py +383 -0
  241. claude_mpm/utils/robust_installer.py +3 -5
  242. claude_mpm/utils/structured_questions.py +619 -0
  243. {claude_mpm-4.21.3.dist-info → claude_mpm-5.0.2.dist-info}/METADATA +429 -59
  244. {claude_mpm-4.21.3.dist-info → claude_mpm-5.0.2.dist-info}/RECORD +252 -425
  245. claude_mpm/agents/templates/.claude-mpm/memories/README.md +0 -17
  246. claude_mpm/agents/templates/.claude-mpm/memories/engineer_memories.md +0 -3
  247. claude_mpm/agents/templates/agent-manager.json +0 -273
  248. claude_mpm/agents/templates/agentic-coder-optimizer.json +0 -248
  249. claude_mpm/agents/templates/api_qa.json +0 -180
  250. claude_mpm/agents/templates/clerk-ops.json +0 -235
  251. claude_mpm/agents/templates/code_analyzer.json +0 -101
  252. claude_mpm/agents/templates/content-agent.json +0 -358
  253. claude_mpm/agents/templates/dart_engineer.json +0 -307
  254. claude_mpm/agents/templates/data_engineer.json +0 -225
  255. claude_mpm/agents/templates/documentation.json +0 -211
  256. claude_mpm/agents/templates/engineer.json +0 -210
  257. claude_mpm/agents/templates/gcp_ops_agent.json +0 -253
  258. claude_mpm/agents/templates/golang_engineer.json +0 -270
  259. claude_mpm/agents/templates/imagemagick.json +0 -264
  260. claude_mpm/agents/templates/java_engineer.json +0 -346
  261. claude_mpm/agents/templates/local_ops_agent.json +0 -1840
  262. claude_mpm/agents/templates/logs/prompts/agent_engineer_20250826_014258_728.md +0 -39
  263. claude_mpm/agents/templates/logs/prompts/agent_engineer_20250901_010124_142.md +0 -400
  264. claude_mpm/agents/templates/memory_manager.json +0 -158
  265. claude_mpm/agents/templates/nextjs_engineer.json +0 -285
  266. claude_mpm/agents/templates/ops.json +0 -185
  267. claude_mpm/agents/templates/php-engineer.json +0 -287
  268. claude_mpm/agents/templates/product_owner.json +0 -338
  269. claude_mpm/agents/templates/project_organizer.json +0 -140
  270. claude_mpm/agents/templates/prompt-engineer.json +0 -737
  271. claude_mpm/agents/templates/python_engineer.json +0 -387
  272. claude_mpm/agents/templates/qa.json +0 -242
  273. claude_mpm/agents/templates/react_engineer.json +0 -238
  274. claude_mpm/agents/templates/refactoring_engineer.json +0 -276
  275. claude_mpm/agents/templates/research.json +0 -188
  276. claude_mpm/agents/templates/ruby-engineer.json +0 -280
  277. claude_mpm/agents/templates/rust_engineer.json +0 -275
  278. claude_mpm/agents/templates/security.json +0 -202
  279. claude_mpm/agents/templates/svelte-engineer.json +0 -225
  280. claude_mpm/agents/templates/ticketing.json +0 -177
  281. claude_mpm/agents/templates/typescript_engineer.json +0 -285
  282. claude_mpm/agents/templates/vercel_ops_agent.json +0 -412
  283. claude_mpm/agents/templates/version_control.json +0 -157
  284. claude_mpm/agents/templates/web_qa.json +0 -399
  285. claude_mpm/agents/templates/web_ui.json +0 -189
  286. claude_mpm/commands/mpm-tickets.md +0 -102
  287. claude_mpm/dashboard/.claude-mpm/socketio-instances.json +0 -1
  288. claude_mpm/dashboard/react/components/DataInspector/DataInspector.module.css +0 -188
  289. claude_mpm/dashboard/react/components/EventViewer/EventViewer.module.css +0 -156
  290. claude_mpm/dashboard/react/components/shared/ConnectionStatus.module.css +0 -38
  291. claude_mpm/dashboard/react/components/shared/FilterBar.module.css +0 -92
  292. claude_mpm/dashboard/static/archive/activity_dashboard_fixed.html +0 -248
  293. claude_mpm/dashboard/static/archive/activity_dashboard_test.html +0 -61
  294. claude_mpm/dashboard/static/archive/test_activity_connection.html +0 -179
  295. claude_mpm/dashboard/static/archive/test_claude_tree_tab.html +0 -68
  296. claude_mpm/dashboard/static/archive/test_dashboard.html +0 -409
  297. claude_mpm/dashboard/static/archive/test_dashboard_fixed.html +0 -519
  298. claude_mpm/dashboard/static/archive/test_dashboard_verification.html +0 -181
  299. claude_mpm/dashboard/static/archive/test_file_data.html +0 -315
  300. claude_mpm/dashboard/static/archive/test_file_tree_empty_state.html +0 -243
  301. claude_mpm/dashboard/static/archive/test_file_tree_fix.html +0 -234
  302. claude_mpm/dashboard/static/archive/test_file_tree_rename.html +0 -117
  303. claude_mpm/dashboard/static/archive/test_file_tree_tab.html +0 -115
  304. claude_mpm/dashboard/static/archive/test_file_viewer.html +0 -224
  305. claude_mpm/dashboard/static/archive/test_final_activity.html +0 -220
  306. claude_mpm/dashboard/static/archive/test_tab_fix.html +0 -139
  307. claude_mpm/dashboard/static/built/assets/events.DjpNxWNo.css +0 -1
  308. claude_mpm/dashboard/static/built/components/activity-tree.js +0 -2
  309. claude_mpm/dashboard/static/built/components/agent-hierarchy.js +0 -777
  310. claude_mpm/dashboard/static/built/components/agent-inference.js +0 -2
  311. claude_mpm/dashboard/static/built/components/build-tracker.js +0 -333
  312. claude_mpm/dashboard/static/built/components/code-simple.js +0 -857
  313. claude_mpm/dashboard/static/built/components/code-tree/tree-breadcrumb.js +0 -353
  314. claude_mpm/dashboard/static/built/components/code-tree/tree-constants.js +0 -235
  315. claude_mpm/dashboard/static/built/components/code-tree/tree-search.js +0 -409
  316. claude_mpm/dashboard/static/built/components/code-tree/tree-utils.js +0 -435
  317. claude_mpm/dashboard/static/built/components/code-tree.js +0 -2
  318. claude_mpm/dashboard/static/built/components/code-viewer.js +0 -2
  319. claude_mpm/dashboard/static/built/components/connection-debug.js +0 -654
  320. claude_mpm/dashboard/static/built/components/diff-viewer.js +0 -891
  321. claude_mpm/dashboard/static/built/components/event-processor.js +0 -2
  322. claude_mpm/dashboard/static/built/components/event-viewer.js +0 -2
  323. claude_mpm/dashboard/static/built/components/export-manager.js +0 -2
  324. claude_mpm/dashboard/static/built/components/file-change-tracker.js +0 -443
  325. claude_mpm/dashboard/static/built/components/file-change-viewer.js +0 -690
  326. claude_mpm/dashboard/static/built/components/file-tool-tracker.js +0 -2
  327. claude_mpm/dashboard/static/built/components/file-viewer.js +0 -2
  328. claude_mpm/dashboard/static/built/components/hud-library-loader.js +0 -2
  329. claude_mpm/dashboard/static/built/components/hud-manager.js +0 -2
  330. claude_mpm/dashboard/static/built/components/hud-visualizer.js +0 -2
  331. claude_mpm/dashboard/static/built/components/module-viewer.js +0 -2
  332. claude_mpm/dashboard/static/built/components/nav-bar.js +0 -145
  333. claude_mpm/dashboard/static/built/components/page-structure.js +0 -429
  334. claude_mpm/dashboard/static/built/components/session-manager.js +0 -2
  335. claude_mpm/dashboard/static/built/components/socket-manager.js +0 -2
  336. claude_mpm/dashboard/static/built/components/ui-state-manager.js +0 -2
  337. claude_mpm/dashboard/static/built/components/unified-data-viewer.js +0 -2
  338. claude_mpm/dashboard/static/built/components/working-directory.js +0 -2
  339. claude_mpm/dashboard/static/built/connection-manager.js +0 -536
  340. claude_mpm/dashboard/static/built/dashboard.js +0 -2
  341. claude_mpm/dashboard/static/built/extension-error-handler.js +0 -164
  342. claude_mpm/dashboard/static/built/react/events.js +0 -30
  343. claude_mpm/dashboard/static/built/shared/dom-helpers.js +0 -396
  344. claude_mpm/dashboard/static/built/shared/event-bus.js +0 -330
  345. claude_mpm/dashboard/static/built/shared/event-filter-service.js +0 -540
  346. claude_mpm/dashboard/static/built/shared/logger.js +0 -385
  347. claude_mpm/dashboard/static/built/shared/page-structure.js +0 -249
  348. claude_mpm/dashboard/static/built/shared/tooltip-service.js +0 -253
  349. claude_mpm/dashboard/static/built/socket-client.js +0 -2
  350. claude_mpm/dashboard/static/built/tab-isolation-fix.js +0 -185
  351. claude_mpm/dashboard/static/dist/assets/events.DjpNxWNo.css +0 -1
  352. claude_mpm/dashboard/static/dist/components/activity-tree.js +0 -2
  353. claude_mpm/dashboard/static/dist/components/agent-inference.js +0 -2
  354. claude_mpm/dashboard/static/dist/components/code-tree.js +0 -2
  355. claude_mpm/dashboard/static/dist/components/code-viewer.js +0 -2
  356. claude_mpm/dashboard/static/dist/components/event-processor.js +0 -2
  357. claude_mpm/dashboard/static/dist/components/event-viewer.js +0 -2
  358. claude_mpm/dashboard/static/dist/components/export-manager.js +0 -2
  359. claude_mpm/dashboard/static/dist/components/file-tool-tracker.js +0 -2
  360. claude_mpm/dashboard/static/dist/components/file-viewer.js +0 -2
  361. claude_mpm/dashboard/static/dist/components/hud-library-loader.js +0 -2
  362. claude_mpm/dashboard/static/dist/components/hud-manager.js +0 -2
  363. claude_mpm/dashboard/static/dist/components/hud-visualizer.js +0 -2
  364. claude_mpm/dashboard/static/dist/components/module-viewer.js +0 -2
  365. claude_mpm/dashboard/static/dist/components/session-manager.js +0 -2
  366. claude_mpm/dashboard/static/dist/components/socket-manager.js +0 -2
  367. claude_mpm/dashboard/static/dist/components/ui-state-manager.js +0 -2
  368. claude_mpm/dashboard/static/dist/components/unified-data-viewer.js +0 -2
  369. claude_mpm/dashboard/static/dist/components/working-directory.js +0 -2
  370. claude_mpm/dashboard/static/dist/dashboard.js +0 -2
  371. claude_mpm/dashboard/static/dist/react/events.js +0 -30
  372. claude_mpm/dashboard/static/dist/socket-client.js +0 -2
  373. claude_mpm/dashboard/static/events.html +0 -607
  374. claude_mpm/dashboard/static/index.html +0 -635
  375. claude_mpm/dashboard/static/js/shared/dom-helpers.js +0 -396
  376. claude_mpm/dashboard/static/js/shared/event-bus.js +0 -330
  377. claude_mpm/dashboard/static/js/shared/logger.js +0 -385
  378. claude_mpm/dashboard/static/js/shared/tooltip-service.js +0 -253
  379. claude_mpm/dashboard/static/js/stores/dashboard-store.js +0 -562
  380. claude_mpm/dashboard/static/legacy/activity.html +0 -736
  381. claude_mpm/dashboard/static/legacy/agents.html +0 -786
  382. claude_mpm/dashboard/static/legacy/files.html +0 -747
  383. claude_mpm/dashboard/static/legacy/tools.html +0 -831
  384. claude_mpm/dashboard/static/monitors.html +0 -431
  385. claude_mpm/dashboard/static/production/events.html +0 -659
  386. claude_mpm/dashboard/static/production/main.html +0 -698
  387. claude_mpm/dashboard/static/production/monitors.html +0 -483
  388. claude_mpm/dashboard/static/test-archive/dashboard.html +0 -635
  389. claude_mpm/dashboard/static/test-archive/debug-events.html +0 -147
  390. claude_mpm/dashboard/static/test-archive/test-navigation.html +0 -256
  391. claude_mpm/dashboard/static/test-archive/test-react-exports.html +0 -180
  392. claude_mpm/dashboard/static/test-archive/test_debug.html +0 -25
  393. claude_mpm/skills/bundled/collaboration/brainstorming/SKILL.md +0 -79
  394. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/SKILL.md +0 -178
  395. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/agent-prompts.md +0 -577
  396. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/coordination-patterns.md +0 -467
  397. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/examples.md +0 -537
  398. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/troubleshooting.md +0 -730
  399. claude_mpm/skills/bundled/collaboration/requesting-code-review/SKILL.md +0 -112
  400. claude_mpm/skills/bundled/collaboration/requesting-code-review/references/code-reviewer-template.md +0 -146
  401. claude_mpm/skills/bundled/collaboration/requesting-code-review/references/review-examples.md +0 -412
  402. claude_mpm/skills/bundled/collaboration/writing-plans/SKILL.md +0 -81
  403. claude_mpm/skills/bundled/collaboration/writing-plans/references/best-practices.md +0 -362
  404. claude_mpm/skills/bundled/collaboration/writing-plans/references/plan-structure-templates.md +0 -312
  405. claude_mpm/skills/bundled/debugging/root-cause-tracing/SKILL.md +0 -152
  406. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/advanced-techniques.md +0 -668
  407. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/examples.md +0 -587
  408. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/integration.md +0 -438
  409. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/tracing-techniques.md +0 -391
  410. claude_mpm/skills/bundled/debugging/systematic-debugging/CREATION-LOG.md +0 -119
  411. claude_mpm/skills/bundled/debugging/systematic-debugging/SKILL.md +0 -148
  412. claude_mpm/skills/bundled/debugging/systematic-debugging/references/anti-patterns.md +0 -483
  413. claude_mpm/skills/bundled/debugging/systematic-debugging/references/examples.md +0 -452
  414. claude_mpm/skills/bundled/debugging/systematic-debugging/references/troubleshooting.md +0 -449
  415. claude_mpm/skills/bundled/debugging/systematic-debugging/references/workflow.md +0 -411
  416. claude_mpm/skills/bundled/debugging/systematic-debugging/test-academic.md +0 -14
  417. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-1.md +0 -58
  418. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-2.md +0 -68
  419. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-3.md +0 -69
  420. claude_mpm/skills/bundled/debugging/verification-before-completion/SKILL.md +0 -131
  421. claude_mpm/skills/bundled/debugging/verification-before-completion/references/gate-function.md +0 -325
  422. claude_mpm/skills/bundled/debugging/verification-before-completion/references/integration-and-workflows.md +0 -490
  423. claude_mpm/skills/bundled/debugging/verification-before-completion/references/red-flags-and-failures.md +0 -425
  424. claude_mpm/skills/bundled/debugging/verification-before-completion/references/verification-patterns.md +0 -499
  425. claude_mpm/skills/bundled/main/artifacts-builder/SKILL.md +0 -86
  426. claude_mpm/skills/bundled/main/internal-comms/SKILL.md +0 -43
  427. claude_mpm/skills/bundled/main/internal-comms/examples/3p-updates.md +0 -47
  428. claude_mpm/skills/bundled/main/internal-comms/examples/company-newsletter.md +0 -65
  429. claude_mpm/skills/bundled/main/internal-comms/examples/faq-answers.md +0 -30
  430. claude_mpm/skills/bundled/main/internal-comms/examples/general-comms.md +0 -16
  431. claude_mpm/skills/bundled/main/mcp-builder/SKILL.md +0 -160
  432. claude_mpm/skills/bundled/main/mcp-builder/reference/design_principles.md +0 -412
  433. claude_mpm/skills/bundled/main/mcp-builder/reference/evaluation.md +0 -602
  434. claude_mpm/skills/bundled/main/mcp-builder/reference/mcp_best_practices.md +0 -915
  435. claude_mpm/skills/bundled/main/mcp-builder/reference/node_mcp_server.md +0 -916
  436. claude_mpm/skills/bundled/main/mcp-builder/reference/python_mcp_server.md +0 -752
  437. claude_mpm/skills/bundled/main/mcp-builder/reference/workflow.md +0 -1237
  438. claude_mpm/skills/bundled/main/skill-creator/SKILL.md +0 -189
  439. claude_mpm/skills/bundled/main/skill-creator/references/best-practices.md +0 -500
  440. claude_mpm/skills/bundled/main/skill-creator/references/creation-workflow.md +0 -464
  441. claude_mpm/skills/bundled/main/skill-creator/references/examples.md +0 -619
  442. claude_mpm/skills/bundled/main/skill-creator/references/progressive-disclosure.md +0 -437
  443. claude_mpm/skills/bundled/main/skill-creator/references/skill-structure.md +0 -231
  444. claude_mpm/skills/bundled/php/espocrm-development/SKILL.md +0 -170
  445. claude_mpm/skills/bundled/php/espocrm-development/references/architecture.md +0 -602
  446. claude_mpm/skills/bundled/php/espocrm-development/references/common-tasks.md +0 -821
  447. claude_mpm/skills/bundled/php/espocrm-development/references/development-workflow.md +0 -742
  448. claude_mpm/skills/bundled/php/espocrm-development/references/frontend-customization.md +0 -726
  449. claude_mpm/skills/bundled/php/espocrm-development/references/hooks-and-services.md +0 -764
  450. claude_mpm/skills/bundled/php/espocrm-development/references/testing-debugging.md +0 -831
  451. claude_mpm/skills/bundled/rust/desktop-applications/SKILL.md +0 -226
  452. claude_mpm/skills/bundled/rust/desktop-applications/references/architecture-patterns.md +0 -901
  453. claude_mpm/skills/bundled/rust/desktop-applications/references/native-gui-frameworks.md +0 -901
  454. claude_mpm/skills/bundled/rust/desktop-applications/references/platform-integration.md +0 -775
  455. claude_mpm/skills/bundled/rust/desktop-applications/references/state-management.md +0 -937
  456. claude_mpm/skills/bundled/rust/desktop-applications/references/tauri-framework.md +0 -770
  457. claude_mpm/skills/bundled/rust/desktop-applications/references/testing-deployment.md +0 -961
  458. claude_mpm/skills/bundled/testing/condition-based-waiting/SKILL.md +0 -119
  459. claude_mpm/skills/bundled/testing/condition-based-waiting/references/patterns-and-implementation.md +0 -253
  460. claude_mpm/skills/bundled/testing/test-driven-development/SKILL.md +0 -145
  461. claude_mpm/skills/bundled/testing/test-driven-development/references/anti-patterns.md +0 -543
  462. claude_mpm/skills/bundled/testing/test-driven-development/references/examples.md +0 -741
  463. claude_mpm/skills/bundled/testing/test-driven-development/references/integration.md +0 -470
  464. claude_mpm/skills/bundled/testing/test-driven-development/references/philosophy.md +0 -458
  465. claude_mpm/skills/bundled/testing/test-driven-development/references/workflow.md +0 -639
  466. claude_mpm/skills/bundled/testing/testing-anti-patterns/SKILL.md +0 -140
  467. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/completeness-anti-patterns.md +0 -572
  468. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/core-anti-patterns.md +0 -411
  469. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/detection-guide.md +0 -569
  470. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/tdd-connection.md +0 -695
  471. claude_mpm/skills/bundled/testing/webapp-testing/SKILL.md +0 -184
  472. claude_mpm/skills/bundled/testing/webapp-testing/decision-tree.md +0 -459
  473. claude_mpm/skills/bundled/testing/webapp-testing/playwright-patterns.md +0 -479
  474. claude_mpm/skills/bundled/testing/webapp-testing/reconnaissance-pattern.md +0 -687
  475. claude_mpm/skills/bundled/testing/webapp-testing/server-management.md +0 -758
  476. claude_mpm/skills/bundled/testing/webapp-testing/troubleshooting.md +0 -868
  477. /claude_mpm/agents/templates/{git_file_tracking.md → git-file-tracking.md} +0 -0
  478. /claude_mpm/agents/templates/{pm_examples.md → pm-examples.md} +0 -0
  479. /claude_mpm/agents/templates/{response_format.md → response-format.md} +0 -0
  480. /claude_mpm/agents/templates/{validation_templates.md → validation-templates.md} +0 -0
  481. {claude_mpm-4.21.3.dist-info → claude_mpm-5.0.2.dist-info}/WHEEL +0 -0
  482. {claude_mpm-4.21.3.dist-info → claude_mpm-5.0.2.dist-info}/entry_points.txt +0 -0
  483. {claude_mpm-4.21.3.dist-info → claude_mpm-5.0.2.dist-info}/licenses/LICENSE +0 -0
  484. {claude_mpm-4.21.3.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.