claude-mpm 5.4.22__py3-none-any.whl → 5.6.34__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.

Potentially problematic release.


This version of claude-mpm might be problematic. Click here for more details.

Files changed (487) hide show
  1. claude_mpm/VERSION +1 -1
  2. claude_mpm/agents/BASE_AGENT.md +164 -0
  3. claude_mpm/agents/BASE_ENGINEER.md +658 -0
  4. claude_mpm/agents/CLAUDE_MPM_OUTPUT_STYLE.md +66 -241
  5. claude_mpm/agents/CLAUDE_MPM_RESEARCH_OUTPUT_STYLE.md +413 -0
  6. claude_mpm/agents/CLAUDE_MPM_TEACHER_OUTPUT_STYLE.md +109 -1925
  7. claude_mpm/agents/MEMORY.md +1 -1
  8. claude_mpm/agents/PM_INSTRUCTIONS.md +374 -1257
  9. claude_mpm/agents/WORKFLOW.md +6 -253
  10. claude_mpm/agents/agent_loader.py +1 -1
  11. claude_mpm/agents/base_agent.json +31 -0
  12. claude_mpm/agents/frontmatter_validator.py +2 -2
  13. claude_mpm/agents/templates/circuit-breakers.md +26 -17
  14. claude_mpm/cli/__init__.py +5 -1
  15. claude_mpm/cli/commands/agent_state_manager.py +10 -10
  16. claude_mpm/cli/commands/agents.py +11 -13
  17. claude_mpm/cli/commands/agents_reconcile.py +197 -0
  18. claude_mpm/cli/commands/auto_configure.py +4 -4
  19. claude_mpm/cli/commands/autotodos.py +566 -0
  20. claude_mpm/cli/commands/commander.py +216 -0
  21. claude_mpm/cli/commands/configure.py +621 -22
  22. claude_mpm/cli/commands/configure_agent_display.py +12 -0
  23. claude_mpm/cli/commands/hook_errors.py +60 -60
  24. claude_mpm/cli/commands/monitor.py +2 -2
  25. claude_mpm/cli/commands/mpm_init/core.py +72 -0
  26. claude_mpm/cli/commands/postmortem.py +1 -1
  27. claude_mpm/cli/commands/profile.py +276 -0
  28. claude_mpm/cli/commands/run.py +35 -3
  29. claude_mpm/cli/commands/skill_source.py +51 -2
  30. claude_mpm/cli/commands/skills.py +182 -32
  31. claude_mpm/cli/executor.py +130 -16
  32. claude_mpm/cli/interactive/__init__.py +10 -0
  33. claude_mpm/cli/interactive/agent_wizard.py +32 -52
  34. claude_mpm/cli/interactive/questionary_styles.py +65 -0
  35. claude_mpm/cli/interactive/skill_selector.py +481 -0
  36. claude_mpm/cli/parsers/base_parser.py +83 -1
  37. claude_mpm/cli/parsers/commander_parser.py +116 -0
  38. claude_mpm/cli/parsers/profile_parser.py +147 -0
  39. claude_mpm/cli/parsers/run_parser.py +10 -0
  40. claude_mpm/cli/parsers/skill_source_parser.py +4 -0
  41. claude_mpm/cli/parsers/skills_parser.py +2 -3
  42. claude_mpm/cli/startup.py +690 -386
  43. claude_mpm/cli/startup_display.py +74 -6
  44. claude_mpm/cli/startup_logging.py +2 -2
  45. claude_mpm/cli/utils.py +7 -3
  46. claude_mpm/commander/__init__.py +78 -0
  47. claude_mpm/commander/adapters/__init__.py +60 -0
  48. claude_mpm/commander/adapters/auggie.py +260 -0
  49. claude_mpm/commander/adapters/base.py +288 -0
  50. claude_mpm/commander/adapters/claude_code.py +392 -0
  51. claude_mpm/commander/adapters/codex.py +237 -0
  52. claude_mpm/commander/adapters/communication.py +366 -0
  53. claude_mpm/commander/adapters/example_usage.py +310 -0
  54. claude_mpm/commander/adapters/mpm.py +389 -0
  55. claude_mpm/commander/adapters/registry.py +204 -0
  56. claude_mpm/commander/api/__init__.py +16 -0
  57. claude_mpm/commander/api/app.py +121 -0
  58. claude_mpm/commander/api/errors.py +133 -0
  59. claude_mpm/commander/api/routes/__init__.py +8 -0
  60. claude_mpm/commander/api/routes/events.py +184 -0
  61. claude_mpm/commander/api/routes/inbox.py +171 -0
  62. claude_mpm/commander/api/routes/messages.py +148 -0
  63. claude_mpm/commander/api/routes/projects.py +271 -0
  64. claude_mpm/commander/api/routes/sessions.py +226 -0
  65. claude_mpm/commander/api/routes/work.py +296 -0
  66. claude_mpm/commander/api/schemas.py +186 -0
  67. claude_mpm/commander/chat/__init__.py +7 -0
  68. claude_mpm/commander/chat/cli.py +146 -0
  69. claude_mpm/commander/chat/commands.py +96 -0
  70. claude_mpm/commander/chat/repl.py +310 -0
  71. claude_mpm/commander/config.py +51 -0
  72. claude_mpm/commander/config_loader.py +115 -0
  73. claude_mpm/commander/core/__init__.py +10 -0
  74. claude_mpm/commander/core/block_manager.py +325 -0
  75. claude_mpm/commander/core/response_manager.py +323 -0
  76. claude_mpm/commander/daemon.py +603 -0
  77. claude_mpm/commander/env_loader.py +59 -0
  78. claude_mpm/commander/events/__init__.py +26 -0
  79. claude_mpm/commander/events/manager.py +332 -0
  80. claude_mpm/commander/frameworks/__init__.py +12 -0
  81. claude_mpm/commander/frameworks/base.py +146 -0
  82. claude_mpm/commander/frameworks/claude_code.py +58 -0
  83. claude_mpm/commander/frameworks/mpm.py +62 -0
  84. claude_mpm/commander/inbox/__init__.py +16 -0
  85. claude_mpm/commander/inbox/dedup.py +128 -0
  86. claude_mpm/commander/inbox/inbox.py +224 -0
  87. claude_mpm/commander/inbox/models.py +70 -0
  88. claude_mpm/commander/instance_manager.py +450 -0
  89. claude_mpm/commander/llm/__init__.py +6 -0
  90. claude_mpm/commander/llm/openrouter_client.py +167 -0
  91. claude_mpm/commander/llm/summarizer.py +70 -0
  92. claude_mpm/commander/memory/__init__.py +45 -0
  93. claude_mpm/commander/memory/compression.py +347 -0
  94. claude_mpm/commander/memory/embeddings.py +230 -0
  95. claude_mpm/commander/memory/entities.py +310 -0
  96. claude_mpm/commander/memory/example_usage.py +290 -0
  97. claude_mpm/commander/memory/integration.py +325 -0
  98. claude_mpm/commander/memory/search.py +381 -0
  99. claude_mpm/commander/memory/store.py +657 -0
  100. claude_mpm/commander/models/__init__.py +18 -0
  101. claude_mpm/commander/models/events.py +121 -0
  102. claude_mpm/commander/models/project.py +162 -0
  103. claude_mpm/commander/models/work.py +214 -0
  104. claude_mpm/commander/parsing/__init__.py +20 -0
  105. claude_mpm/commander/parsing/extractor.py +132 -0
  106. claude_mpm/commander/parsing/output_parser.py +270 -0
  107. claude_mpm/commander/parsing/patterns.py +100 -0
  108. claude_mpm/commander/persistence/__init__.py +11 -0
  109. claude_mpm/commander/persistence/event_store.py +274 -0
  110. claude_mpm/commander/persistence/state_store.py +309 -0
  111. claude_mpm/commander/persistence/work_store.py +164 -0
  112. claude_mpm/commander/polling/__init__.py +13 -0
  113. claude_mpm/commander/polling/event_detector.py +104 -0
  114. claude_mpm/commander/polling/output_buffer.py +49 -0
  115. claude_mpm/commander/polling/output_poller.py +153 -0
  116. claude_mpm/commander/project_session.py +268 -0
  117. claude_mpm/commander/proxy/__init__.py +12 -0
  118. claude_mpm/commander/proxy/formatter.py +89 -0
  119. claude_mpm/commander/proxy/output_handler.py +191 -0
  120. claude_mpm/commander/proxy/relay.py +155 -0
  121. claude_mpm/commander/registry.py +410 -0
  122. claude_mpm/commander/runtime/__init__.py +10 -0
  123. claude_mpm/commander/runtime/executor.py +191 -0
  124. claude_mpm/commander/runtime/monitor.py +346 -0
  125. claude_mpm/commander/session/__init__.py +6 -0
  126. claude_mpm/commander/session/context.py +81 -0
  127. claude_mpm/commander/session/manager.py +59 -0
  128. claude_mpm/commander/tmux_orchestrator.py +361 -0
  129. claude_mpm/commander/web/__init__.py +1 -0
  130. claude_mpm/commander/work/__init__.py +30 -0
  131. claude_mpm/commander/work/executor.py +207 -0
  132. claude_mpm/commander/work/queue.py +405 -0
  133. claude_mpm/commander/workflow/__init__.py +27 -0
  134. claude_mpm/commander/workflow/event_handler.py +241 -0
  135. claude_mpm/commander/workflow/notifier.py +146 -0
  136. claude_mpm/commands/mpm-config.md +20 -249
  137. claude_mpm/commands/mpm-doctor.md +16 -21
  138. claude_mpm/commands/mpm-help.md +12 -205
  139. claude_mpm/commands/mpm-init.md +88 -506
  140. claude_mpm/commands/mpm-monitor.md +22 -401
  141. claude_mpm/commands/mpm-organize.md +70 -442
  142. claude_mpm/commands/mpm-postmortem.md +13 -107
  143. claude_mpm/commands/mpm-session-resume.md +20 -363
  144. claude_mpm/commands/mpm-status.md +13 -69
  145. claude_mpm/commands/mpm-ticket-view.md +60 -495
  146. claude_mpm/commands/mpm-version.md +13 -107
  147. claude_mpm/commands/mpm.md +8 -0
  148. claude_mpm/config/agent_presets.py +8 -7
  149. claude_mpm/config/skill_sources.py +16 -0
  150. claude_mpm/constants.py +1 -0
  151. claude_mpm/core/claude_runner.py +154 -2
  152. claude_mpm/core/config.py +37 -26
  153. claude_mpm/core/config_constants.py +74 -9
  154. claude_mpm/core/constants.py +56 -12
  155. claude_mpm/core/framework/loaders/agent_loader.py +1 -1
  156. claude_mpm/core/framework/loaders/instruction_loader.py +52 -11
  157. claude_mpm/core/hook_manager.py +51 -3
  158. claude_mpm/core/interactive_session.py +12 -11
  159. claude_mpm/core/logger.py +26 -9
  160. claude_mpm/core/logging_utils.py +39 -13
  161. claude_mpm/core/network_config.py +148 -0
  162. claude_mpm/core/oneshot_session.py +7 -6
  163. claude_mpm/core/optimized_startup.py +61 -0
  164. claude_mpm/core/output_style_manager.py +66 -18
  165. claude_mpm/core/shared/config_loader.py +3 -1
  166. claude_mpm/core/socketio_pool.py +47 -15
  167. claude_mpm/core/unified_agent_registry.py +1 -1
  168. claude_mpm/core/unified_config.py +54 -8
  169. claude_mpm/core/unified_paths.py +95 -90
  170. claude_mpm/dashboard/static/svelte-build/_app/env.js +1 -0
  171. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/0.C33zOoyM.css +1 -0
  172. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/2.CW1J-YuA.css +1 -0
  173. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/1WZnGYqX.js +24 -0
  174. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/67pF3qNn.js +1 -0
  175. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/6RxdMKe4.js +1 -0
  176. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/8cZrfX0h.js +60 -0
  177. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/9a6T2nm-.js +7 -0
  178. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/B443AUzu.js +1 -0
  179. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/B8AwtY2H.js +1 -0
  180. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BF15LAsF.js +1 -0
  181. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BQaXIfA_.js +331 -0
  182. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BRcwIQNr.js +4 -0
  183. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BSNlmTZj.js +1 -0
  184. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BV6nKitt.js +43 -0
  185. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BViJ8lZt.js +128 -0
  186. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BcQ-Q0FE.js +1 -0
  187. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Bpyvgze_.js +30 -0
  188. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BzTRqg-z.js +1 -0
  189. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C0Fr8dve.js +1 -0
  190. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C3rbW_a-.js +1 -0
  191. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C8WYN38h.js +1 -0
  192. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C9I8FlXH.js +61 -0
  193. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CIQcWgO2.js +36 -0
  194. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CIctN7YN.js +7 -0
  195. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CKrS_JZW.js +145 -0
  196. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CR6P9C4A.js +89 -0
  197. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CRRR9MD_.js +2 -0
  198. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CRcR2DqT.js +334 -0
  199. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CSXtMOf0.js +1 -0
  200. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CT-sbxSk.js +1 -0
  201. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CWm6DJsp.js +1 -0
  202. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CmKTTxBW.js +1 -0
  203. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CpqQ1Kzn.js +1 -0
  204. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Cu_Erd72.js +261 -0
  205. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/D2nGpDRe.js +1 -0
  206. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/D9iCMida.js +267 -0
  207. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/D9ykgMoY.js +10 -0
  208. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DL2Ldur1.js +1 -0
  209. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DPfltzjH.js +165 -0
  210. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DR8nis88.js +2 -0
  211. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DUliQN2b.js +1 -0
  212. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DVp1hx9R.js +1 -0
  213. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DXlhR01x.js +122 -0
  214. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/D_lyTybS.js +1 -0
  215. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DngoTTgh.js +1 -0
  216. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DqkmHtDC.js +220 -0
  217. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DsDh8EYs.js +1 -0
  218. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DypDmXgd.js +139 -0
  219. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Gi6I4Gst.js +1 -0
  220. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/IPYC-LnN.js +162 -0
  221. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/JTLiF7dt.js +24 -0
  222. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/JpevfAFt.js +68 -0
  223. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/NqQ1dWOy.js +1 -0
  224. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/R8CEIRAd.js +2 -0
  225. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Zxy7qc-l.js +64 -0
  226. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/q9Hm6zAU.js +1 -0
  227. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/qtd3IeO4.js +15 -0
  228. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/ulBFON_C.js +65 -0
  229. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/wQVh1CoA.js +10 -0
  230. claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/app.Dr7t0z2J.js +2 -0
  231. claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/start.BGhZHUS3.js +1 -0
  232. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/0.RgBboRvH.js +1 -0
  233. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/1.DG-KkbDf.js +1 -0
  234. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/2.D_jnf-x6.js +1 -0
  235. claude_mpm/dashboard/static/svelte-build/_app/version.json +1 -0
  236. claude_mpm/dashboard/static/svelte-build/favicon.svg +7 -0
  237. claude_mpm/dashboard/static/svelte-build/index.html +36 -0
  238. claude_mpm/dashboard-svelte/node_modules/katex/src/fonts/generate_fonts.py +58 -0
  239. claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/extract_tfms.py +114 -0
  240. claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/extract_ttfs.py +122 -0
  241. claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/format_json.py +28 -0
  242. claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/parse_tfm.py +211 -0
  243. claude_mpm/experimental/cli_enhancements.py +2 -1
  244. claude_mpm/hooks/claude_hooks/INTEGRATION_EXAMPLE.md +243 -0
  245. claude_mpm/hooks/claude_hooks/README_AUTO_PAUSE.md +403 -0
  246. claude_mpm/hooks/claude_hooks/__pycache__/__init__.cpython-311.pyc +0 -0
  247. claude_mpm/hooks/claude_hooks/__pycache__/auto_pause_handler.cpython-311.pyc +0 -0
  248. claude_mpm/hooks/claude_hooks/__pycache__/correlation_manager.cpython-311.pyc +0 -0
  249. claude_mpm/hooks/claude_hooks/__pycache__/event_handlers.cpython-311.pyc +0 -0
  250. claude_mpm/hooks/claude_hooks/__pycache__/hook_handler.cpython-311.pyc +0 -0
  251. claude_mpm/hooks/claude_hooks/__pycache__/memory_integration.cpython-311.pyc +0 -0
  252. claude_mpm/hooks/claude_hooks/__pycache__/response_tracking.cpython-311.pyc +0 -0
  253. claude_mpm/hooks/claude_hooks/__pycache__/tool_analysis.cpython-311.pyc +0 -0
  254. claude_mpm/hooks/claude_hooks/auto_pause_handler.py +485 -0
  255. claude_mpm/hooks/claude_hooks/event_handlers.py +527 -136
  256. claude_mpm/hooks/claude_hooks/hook_handler.py +313 -99
  257. claude_mpm/hooks/claude_hooks/hook_wrapper.sh +6 -11
  258. claude_mpm/hooks/claude_hooks/installer.py +206 -36
  259. claude_mpm/hooks/claude_hooks/memory_integration.py +52 -32
  260. claude_mpm/hooks/claude_hooks/response_tracking.py +43 -60
  261. claude_mpm/hooks/claude_hooks/services/__init__.py +21 -0
  262. claude_mpm/hooks/claude_hooks/services/__pycache__/__init__.cpython-311.pyc +0 -0
  263. claude_mpm/hooks/claude_hooks/services/__pycache__/connection_manager_http.cpython-311.pyc +0 -0
  264. claude_mpm/hooks/claude_hooks/services/__pycache__/container.cpython-311.pyc +0 -0
  265. claude_mpm/hooks/claude_hooks/services/__pycache__/duplicate_detector.cpython-311.pyc +0 -0
  266. claude_mpm/hooks/claude_hooks/services/__pycache__/protocols.cpython-311.pyc +0 -0
  267. claude_mpm/hooks/claude_hooks/services/__pycache__/state_manager.cpython-311.pyc +0 -0
  268. claude_mpm/hooks/claude_hooks/services/__pycache__/subagent_processor.cpython-311.pyc +0 -0
  269. claude_mpm/hooks/claude_hooks/services/connection_manager.py +67 -32
  270. claude_mpm/hooks/claude_hooks/services/connection_manager_http.py +38 -105
  271. claude_mpm/hooks/claude_hooks/services/container.py +310 -0
  272. claude_mpm/hooks/claude_hooks/services/protocols.py +328 -0
  273. claude_mpm/hooks/claude_hooks/services/state_manager.py +25 -38
  274. claude_mpm/hooks/claude_hooks/services/subagent_processor.py +75 -77
  275. claude_mpm/hooks/kuzu_memory_hook.py +5 -5
  276. claude_mpm/hooks/session_resume_hook.py +89 -1
  277. claude_mpm/hooks/templates/pre_tool_use_simple.py +6 -6
  278. claude_mpm/hooks/templates/pre_tool_use_template.py +16 -8
  279. claude_mpm/init.py +276 -0
  280. claude_mpm/models/git_repository.py +3 -3
  281. claude_mpm/scripts/claude-hook-handler.sh +46 -19
  282. claude_mpm/services/agents/agent_builder.py +3 -3
  283. claude_mpm/services/agents/agent_recommendation_service.py +8 -8
  284. claude_mpm/services/agents/agent_selection_service.py +2 -2
  285. claude_mpm/services/agents/cache_git_manager.py +7 -7
  286. claude_mpm/services/agents/deployment/agent_deployment.py +29 -7
  287. claude_mpm/services/agents/deployment/agent_discovery_service.py +4 -2
  288. claude_mpm/services/agents/deployment/agent_format_converter.py +25 -13
  289. claude_mpm/services/agents/deployment/agent_template_builder.py +39 -19
  290. claude_mpm/services/agents/deployment/agents_directory_resolver.py +2 -2
  291. claude_mpm/services/agents/deployment/async_agent_deployment.py +31 -27
  292. claude_mpm/services/agents/deployment/deployment_reconciler.py +577 -0
  293. claude_mpm/services/agents/deployment/local_template_deployment.py +3 -1
  294. claude_mpm/services/agents/deployment/multi_source_deployment_service.py +169 -26
  295. claude_mpm/services/agents/deployment/remote_agent_discovery_service.py +101 -75
  296. claude_mpm/services/agents/deployment/startup_reconciliation.py +138 -0
  297. claude_mpm/services/agents/git_source_manager.py +23 -4
  298. claude_mpm/services/agents/loading/framework_agent_loader.py +75 -2
  299. claude_mpm/services/agents/recommender.py +5 -3
  300. claude_mpm/services/agents/single_tier_deployment_service.py +6 -6
  301. claude_mpm/services/agents/sources/git_source_sync_service.py +121 -10
  302. claude_mpm/services/agents/startup_sync.py +27 -4
  303. claude_mpm/services/cli/__init__.py +3 -0
  304. claude_mpm/services/cli/incremental_pause_manager.py +561 -0
  305. claude_mpm/services/cli/session_resume_helper.py +10 -2
  306. claude_mpm/services/command_deployment_service.py +44 -26
  307. claude_mpm/services/delegation_detector.py +175 -0
  308. claude_mpm/services/diagnostics/checks/agent_check.py +2 -2
  309. claude_mpm/services/diagnostics/checks/agent_sources_check.py +31 -1
  310. claude_mpm/services/diagnostics/checks/configuration_check.py +24 -0
  311. claude_mpm/services/diagnostics/checks/installation_check.py +22 -0
  312. claude_mpm/services/diagnostics/checks/mcp_services_check.py +23 -0
  313. claude_mpm/services/diagnostics/doctor_reporter.py +31 -1
  314. claude_mpm/services/diagnostics/models.py +14 -1
  315. claude_mpm/services/event_log.py +325 -0
  316. claude_mpm/services/git/git_operations_service.py +8 -8
  317. claude_mpm/services/hook_installer_service.py +77 -8
  318. claude_mpm/services/infrastructure/__init__.py +4 -0
  319. claude_mpm/services/infrastructure/context_usage_tracker.py +291 -0
  320. claude_mpm/services/infrastructure/resume_log_generator.py +24 -5
  321. claude_mpm/services/monitor/daemon_manager.py +15 -4
  322. claude_mpm/services/monitor/management/lifecycle.py +15 -3
  323. claude_mpm/services/monitor/server.py +571 -11
  324. claude_mpm/services/pm_skills_deployer.py +884 -0
  325. claude_mpm/services/profile_manager.py +337 -0
  326. claude_mpm/services/skills/git_skill_source_manager.py +281 -20
  327. claude_mpm/services/skills/selective_skill_deployer.py +211 -46
  328. claude_mpm/services/skills/skill_discovery_service.py +74 -4
  329. claude_mpm/services/skills_deployer.py +192 -70
  330. claude_mpm/services/socketio/dashboard_server.py +1 -0
  331. claude_mpm/services/socketio/event_normalizer.py +37 -6
  332. claude_mpm/services/socketio/handlers/hook.py +14 -7
  333. claude_mpm/services/socketio/server/core.py +262 -123
  334. claude_mpm/services/socketio/server/main.py +12 -4
  335. claude_mpm/skills/__init__.py +2 -1
  336. claude_mpm/skills/bundled/collaboration/brainstorming/SKILL.md +79 -0
  337. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/SKILL.md +178 -0
  338. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/agent-prompts.md +577 -0
  339. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/coordination-patterns.md +467 -0
  340. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/examples.md +537 -0
  341. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/troubleshooting.md +730 -0
  342. claude_mpm/skills/bundled/collaboration/git-worktrees.md +317 -0
  343. claude_mpm/skills/bundled/collaboration/requesting-code-review/SKILL.md +112 -0
  344. claude_mpm/skills/bundled/collaboration/requesting-code-review/references/code-reviewer-template.md +146 -0
  345. claude_mpm/skills/bundled/collaboration/requesting-code-review/references/review-examples.md +412 -0
  346. claude_mpm/skills/bundled/collaboration/stacked-prs.md +251 -0
  347. claude_mpm/skills/bundled/collaboration/writing-plans/SKILL.md +81 -0
  348. claude_mpm/skills/bundled/collaboration/writing-plans/references/best-practices.md +362 -0
  349. claude_mpm/skills/bundled/collaboration/writing-plans/references/plan-structure-templates.md +312 -0
  350. claude_mpm/skills/bundled/debugging/root-cause-tracing/SKILL.md +152 -0
  351. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/advanced-techniques.md +668 -0
  352. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/examples.md +587 -0
  353. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/integration.md +438 -0
  354. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/tracing-techniques.md +391 -0
  355. claude_mpm/skills/bundled/debugging/systematic-debugging/CREATION-LOG.md +119 -0
  356. claude_mpm/skills/bundled/debugging/systematic-debugging/SKILL.md +148 -0
  357. claude_mpm/skills/bundled/debugging/systematic-debugging/references/anti-patterns.md +483 -0
  358. claude_mpm/skills/bundled/debugging/systematic-debugging/references/examples.md +452 -0
  359. claude_mpm/skills/bundled/debugging/systematic-debugging/references/troubleshooting.md +449 -0
  360. claude_mpm/skills/bundled/debugging/systematic-debugging/references/workflow.md +411 -0
  361. claude_mpm/skills/bundled/debugging/systematic-debugging/test-academic.md +14 -0
  362. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-1.md +58 -0
  363. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-2.md +68 -0
  364. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-3.md +69 -0
  365. claude_mpm/skills/bundled/debugging/verification-before-completion/SKILL.md +131 -0
  366. claude_mpm/skills/bundled/debugging/verification-before-completion/references/gate-function.md +325 -0
  367. claude_mpm/skills/bundled/debugging/verification-before-completion/references/integration-and-workflows.md +490 -0
  368. claude_mpm/skills/bundled/debugging/verification-before-completion/references/red-flags-and-failures.md +425 -0
  369. claude_mpm/skills/bundled/debugging/verification-before-completion/references/verification-patterns.md +499 -0
  370. claude_mpm/skills/bundled/infrastructure/env-manager/INTEGRATION.md +611 -0
  371. claude_mpm/skills/bundled/infrastructure/env-manager/README.md +596 -0
  372. claude_mpm/skills/bundled/infrastructure/env-manager/SKILL.md +260 -0
  373. claude_mpm/skills/bundled/infrastructure/env-manager/examples/nextjs-env-structure.md +315 -0
  374. claude_mpm/skills/bundled/infrastructure/env-manager/references/frameworks.md +436 -0
  375. claude_mpm/skills/bundled/infrastructure/env-manager/references/security.md +433 -0
  376. claude_mpm/skills/bundled/infrastructure/env-manager/references/synchronization.md +452 -0
  377. claude_mpm/skills/bundled/infrastructure/env-manager/references/troubleshooting.md +404 -0
  378. claude_mpm/skills/bundled/infrastructure/env-manager/references/validation.md +420 -0
  379. claude_mpm/skills/bundled/main/artifacts-builder/SKILL.md +86 -0
  380. claude_mpm/skills/bundled/main/internal-comms/SKILL.md +43 -0
  381. claude_mpm/skills/bundled/main/internal-comms/examples/3p-updates.md +47 -0
  382. claude_mpm/skills/bundled/main/internal-comms/examples/company-newsletter.md +65 -0
  383. claude_mpm/skills/bundled/main/internal-comms/examples/faq-answers.md +30 -0
  384. claude_mpm/skills/bundled/main/internal-comms/examples/general-comms.md +16 -0
  385. claude_mpm/skills/bundled/main/mcp-builder/SKILL.md +160 -0
  386. claude_mpm/skills/bundled/main/mcp-builder/reference/design_principles.md +412 -0
  387. claude_mpm/skills/bundled/main/mcp-builder/reference/evaluation.md +602 -0
  388. claude_mpm/skills/bundled/main/mcp-builder/reference/mcp_best_practices.md +915 -0
  389. claude_mpm/skills/bundled/main/mcp-builder/reference/node_mcp_server.md +916 -0
  390. claude_mpm/skills/bundled/main/mcp-builder/reference/python_mcp_server.md +752 -0
  391. claude_mpm/skills/bundled/main/mcp-builder/reference/workflow.md +1237 -0
  392. claude_mpm/skills/bundled/main/skill-creator/SKILL.md +189 -0
  393. claude_mpm/skills/bundled/main/skill-creator/references/best-practices.md +500 -0
  394. claude_mpm/skills/bundled/main/skill-creator/references/creation-workflow.md +464 -0
  395. claude_mpm/skills/bundled/main/skill-creator/references/examples.md +619 -0
  396. claude_mpm/skills/bundled/main/skill-creator/references/progressive-disclosure.md +437 -0
  397. claude_mpm/skills/bundled/main/skill-creator/references/skill-structure.md +231 -0
  398. claude_mpm/skills/bundled/php/espocrm-development/SKILL.md +170 -0
  399. claude_mpm/skills/bundled/php/espocrm-development/references/architecture.md +602 -0
  400. claude_mpm/skills/bundled/php/espocrm-development/references/common-tasks.md +821 -0
  401. claude_mpm/skills/bundled/php/espocrm-development/references/development-workflow.md +742 -0
  402. claude_mpm/skills/bundled/php/espocrm-development/references/frontend-customization.md +726 -0
  403. claude_mpm/skills/bundled/php/espocrm-development/references/hooks-and-services.md +764 -0
  404. claude_mpm/skills/bundled/php/espocrm-development/references/testing-debugging.md +831 -0
  405. claude_mpm/skills/bundled/pm/mpm/SKILL.md +38 -0
  406. claude_mpm/skills/bundled/pm/mpm-agent-update-workflow/SKILL.md +75 -0
  407. claude_mpm/skills/bundled/pm/mpm-bug-reporting/SKILL.md +248 -0
  408. claude_mpm/skills/bundled/pm/mpm-circuit-breaker-enforcement/SKILL.md +476 -0
  409. claude_mpm/skills/bundled/pm/mpm-config/SKILL.md +29 -0
  410. claude_mpm/skills/bundled/pm/mpm-delegation-patterns/SKILL.md +167 -0
  411. claude_mpm/skills/bundled/pm/mpm-doctor/SKILL.md +53 -0
  412. claude_mpm/skills/bundled/pm/mpm-git-file-tracking/SKILL.md +113 -0
  413. claude_mpm/skills/bundled/pm/mpm-help/SKILL.md +35 -0
  414. claude_mpm/skills/bundled/pm/mpm-init/SKILL.md +125 -0
  415. claude_mpm/skills/bundled/pm/mpm-monitor/SKILL.md +32 -0
  416. claude_mpm/skills/bundled/pm/mpm-organize/SKILL.md +121 -0
  417. claude_mpm/skills/bundled/pm/mpm-postmortem/SKILL.md +22 -0
  418. claude_mpm/skills/bundled/pm/mpm-pr-workflow/SKILL.md +124 -0
  419. claude_mpm/skills/bundled/pm/mpm-session-management/SKILL.md +312 -0
  420. claude_mpm/skills/bundled/pm/mpm-session-pause/SKILL.md +170 -0
  421. claude_mpm/skills/bundled/pm/mpm-session-resume/SKILL.md +31 -0
  422. claude_mpm/skills/bundled/pm/mpm-status/SKILL.md +37 -0
  423. claude_mpm/skills/bundled/pm/mpm-teaching-mode/SKILL.md +657 -0
  424. claude_mpm/skills/bundled/pm/mpm-ticket-view/SKILL.md +110 -0
  425. claude_mpm/skills/bundled/pm/mpm-ticketing-integration/SKILL.md +154 -0
  426. claude_mpm/skills/bundled/pm/mpm-tool-usage-guide/SKILL.md +386 -0
  427. claude_mpm/skills/bundled/pm/mpm-verification-protocols/SKILL.md +198 -0
  428. claude_mpm/skills/bundled/pm/mpm-version/SKILL.md +21 -0
  429. claude_mpm/skills/bundled/react/flexlayout-react.md +742 -0
  430. claude_mpm/skills/bundled/rust/desktop-applications/SKILL.md +226 -0
  431. claude_mpm/skills/bundled/rust/desktop-applications/references/architecture-patterns.md +901 -0
  432. claude_mpm/skills/bundled/rust/desktop-applications/references/native-gui-frameworks.md +901 -0
  433. claude_mpm/skills/bundled/rust/desktop-applications/references/platform-integration.md +775 -0
  434. claude_mpm/skills/bundled/rust/desktop-applications/references/state-management.md +937 -0
  435. claude_mpm/skills/bundled/rust/desktop-applications/references/tauri-framework.md +770 -0
  436. claude_mpm/skills/bundled/rust/desktop-applications/references/testing-deployment.md +961 -0
  437. claude_mpm/skills/bundled/security-scanning.md +112 -0
  438. claude_mpm/skills/bundled/tauri/tauri-async-patterns.md +495 -0
  439. claude_mpm/skills/bundled/tauri/tauri-build-deploy.md +599 -0
  440. claude_mpm/skills/bundled/tauri/tauri-command-patterns.md +535 -0
  441. claude_mpm/skills/bundled/tauri/tauri-error-handling.md +613 -0
  442. claude_mpm/skills/bundled/tauri/tauri-event-system.md +648 -0
  443. claude_mpm/skills/bundled/tauri/tauri-file-system.md +673 -0
  444. claude_mpm/skills/bundled/tauri/tauri-frontend-integration.md +767 -0
  445. claude_mpm/skills/bundled/tauri/tauri-performance.md +669 -0
  446. claude_mpm/skills/bundled/tauri/tauri-state-management.md +573 -0
  447. claude_mpm/skills/bundled/tauri/tauri-testing.md +384 -0
  448. claude_mpm/skills/bundled/tauri/tauri-window-management.md +628 -0
  449. claude_mpm/skills/bundled/testing/condition-based-waiting/SKILL.md +119 -0
  450. claude_mpm/skills/bundled/testing/condition-based-waiting/references/patterns-and-implementation.md +253 -0
  451. claude_mpm/skills/bundled/testing/test-driven-development/SKILL.md +145 -0
  452. claude_mpm/skills/bundled/testing/test-driven-development/references/anti-patterns.md +543 -0
  453. claude_mpm/skills/bundled/testing/test-driven-development/references/examples.md +741 -0
  454. claude_mpm/skills/bundled/testing/test-driven-development/references/integration.md +470 -0
  455. claude_mpm/skills/bundled/testing/test-driven-development/references/philosophy.md +458 -0
  456. claude_mpm/skills/bundled/testing/test-driven-development/references/workflow.md +639 -0
  457. claude_mpm/skills/bundled/testing/test-quality-inspector/SKILL.md +458 -0
  458. claude_mpm/skills/bundled/testing/test-quality-inspector/examples/example-inspection-report.md +411 -0
  459. claude_mpm/skills/bundled/testing/test-quality-inspector/references/assertion-quality.md +317 -0
  460. claude_mpm/skills/bundled/testing/test-quality-inspector/references/inspection-checklist.md +270 -0
  461. claude_mpm/skills/bundled/testing/test-quality-inspector/references/red-flags.md +436 -0
  462. claude_mpm/skills/bundled/testing/testing-anti-patterns/SKILL.md +140 -0
  463. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/completeness-anti-patterns.md +572 -0
  464. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/core-anti-patterns.md +411 -0
  465. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/detection-guide.md +569 -0
  466. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/tdd-connection.md +695 -0
  467. claude_mpm/skills/bundled/testing/webapp-testing/SKILL.md +184 -0
  468. claude_mpm/skills/bundled/testing/webapp-testing/decision-tree.md +459 -0
  469. claude_mpm/skills/bundled/testing/webapp-testing/playwright-patterns.md +479 -0
  470. claude_mpm/skills/bundled/testing/webapp-testing/reconnaissance-pattern.md +687 -0
  471. claude_mpm/skills/bundled/testing/webapp-testing/server-management.md +758 -0
  472. claude_mpm/skills/bundled/testing/webapp-testing/troubleshooting.md +868 -0
  473. claude_mpm/skills/registry.py +295 -90
  474. claude_mpm/skills/skill_manager.py +98 -3
  475. claude_mpm/templates/.pre-commit-config.yaml +112 -0
  476. claude_mpm/utils/agent_dependency_loader.py +115 -4
  477. claude_mpm/utils/agent_filters.py +1 -1
  478. claude_mpm/utils/migration.py +4 -4
  479. claude_mpm/utils/robust_installer.py +86 -21
  480. claude_mpm-5.6.34.dist-info/METADATA +393 -0
  481. {claude_mpm-5.4.22.dist-info → claude_mpm-5.6.34.dist-info}/RECORD +486 -145
  482. claude_mpm-5.4.22.dist-info/METADATA +0 -996
  483. {claude_mpm-5.4.22.dist-info → claude_mpm-5.6.34.dist-info}/WHEEL +0 -0
  484. {claude_mpm-5.4.22.dist-info → claude_mpm-5.6.34.dist-info}/entry_points.txt +0 -0
  485. {claude_mpm-5.4.22.dist-info → claude_mpm-5.6.34.dist-info}/licenses/LICENSE +0 -0
  486. {claude_mpm-5.4.22.dist-info → claude_mpm-5.6.34.dist-info}/licenses/LICENSE-FAQ.md +0 -0
  487. {claude_mpm-5.4.22.dist-info → claude_mpm-5.6.34.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,961 @@
1
+ # Testing and Deployment
2
+
3
+ Complete guide to testing strategies, cross-platform builds, distribution, and CI/CD pipelines for Rust desktop applications.
4
+
5
+ ## Testing Strategies
6
+
7
+ ### Unit Testing Tauri Commands
8
+
9
+ ```rust
10
+ // src/commands.rs
11
+ use serde::{Deserialize, Serialize};
12
+
13
+ #[derive(Debug, Clone, Serialize, Deserialize)]
14
+ pub struct User {
15
+ pub id: u64,
16
+ pub name: String,
17
+ pub email: String,
18
+ }
19
+
20
+ pub struct UserService {
21
+ users: Vec<User>,
22
+ }
23
+
24
+ impl UserService {
25
+ pub fn new() -> Self {
26
+ Self { users: Vec::new() }
27
+ }
28
+
29
+ pub fn add_user(&mut self, name: String, email: String) -> Result<User, String> {
30
+ if name.is_empty() {
31
+ return Err("Name cannot be empty".to_string());
32
+ }
33
+
34
+ if !email.contains('@') {
35
+ return Err("Invalid email".to_string());
36
+ }
37
+
38
+ let user = User {
39
+ id: self.users.len() as u64 + 1,
40
+ name,
41
+ email,
42
+ };
43
+
44
+ self.users.push(user.clone());
45
+ Ok(user)
46
+ }
47
+
48
+ pub fn get_user(&self, id: u64) -> Option<&User> {
49
+ self.users.iter().find(|u| u.id == id)
50
+ }
51
+
52
+ pub fn get_all_users(&self) -> &[User] {
53
+ &self.users
54
+ }
55
+ }
56
+
57
+ #[cfg(test)]
58
+ mod tests {
59
+ use super::*;
60
+
61
+ #[test]
62
+ fn test_add_user_success() {
63
+ let mut service = UserService::new();
64
+ let result = service.add_user("John Doe".to_string(), "john@example.com".to_string());
65
+
66
+ assert!(result.is_ok());
67
+ let user = result.unwrap();
68
+ assert_eq!(user.name, "John Doe");
69
+ assert_eq!(user.email, "john@example.com");
70
+ assert_eq!(service.get_all_users().len(), 1);
71
+ }
72
+
73
+ #[test]
74
+ fn test_add_user_empty_name() {
75
+ let mut service = UserService::new();
76
+ let result = service.add_user("".to_string(), "john@example.com".to_string());
77
+
78
+ assert!(result.is_err());
79
+ assert_eq!(result.unwrap_err(), "Name cannot be empty");
80
+ }
81
+
82
+ #[test]
83
+ fn test_add_user_invalid_email() {
84
+ let mut service = UserService::new();
85
+ let result = service.add_user("John Doe".to_string(), "invalid-email".to_string());
86
+
87
+ assert!(result.is_err());
88
+ assert_eq!(result.unwrap_err(), "Invalid email");
89
+ }
90
+
91
+ #[test]
92
+ fn test_get_user() {
93
+ let mut service = UserService::new();
94
+ service.add_user("John Doe".to_string(), "john@example.com".to_string()).unwrap();
95
+
96
+ let user = service.get_user(1);
97
+ assert!(user.is_some());
98
+ assert_eq!(user.unwrap().name, "John Doe");
99
+ }
100
+
101
+ #[test]
102
+ fn test_get_user_not_found() {
103
+ let service = UserService::new();
104
+ let user = service.get_user(999);
105
+ assert!(user.is_none());
106
+ }
107
+ }
108
+ ```
109
+
110
+ ### Integration Testing with Mock State
111
+
112
+ ```rust
113
+ #[cfg(test)]
114
+ mod integration_tests {
115
+ use super::*;
116
+ use std::sync::Mutex;
117
+
118
+ struct TestState {
119
+ service: Mutex<UserService>,
120
+ }
121
+
122
+ fn setup_test_state() -> TestState {
123
+ TestState {
124
+ service: Mutex::new(UserService::new()),
125
+ }
126
+ }
127
+
128
+ #[test]
129
+ fn test_user_workflow() {
130
+ let state = setup_test_state();
131
+
132
+ // Add multiple users
133
+ {
134
+ let mut service = state.service.lock().unwrap();
135
+ service.add_user("Alice".to_string(), "alice@example.com".to_string()).unwrap();
136
+ service.add_user("Bob".to_string(), "bob@example.com".to_string()).unwrap();
137
+ }
138
+
139
+ // Verify users were added
140
+ {
141
+ let service = state.service.lock().unwrap();
142
+ assert_eq!(service.get_all_users().len(), 2);
143
+ }
144
+
145
+ // Get specific user
146
+ {
147
+ let service = state.service.lock().unwrap();
148
+ let alice = service.get_user(1);
149
+ assert!(alice.is_some());
150
+ assert_eq!(alice.unwrap().name, "Alice");
151
+ }
152
+ }
153
+ }
154
+ ```
155
+
156
+ ### Async Testing with Tokio
157
+
158
+ ```rust
159
+ #[cfg(test)]
160
+ mod async_tests {
161
+ use super::*;
162
+ use tokio::test;
163
+
164
+ #[tokio::test]
165
+ async fn test_async_operation() {
166
+ let result = fetch_data_async().await;
167
+ assert!(result.is_ok());
168
+ }
169
+
170
+ #[tokio::test]
171
+ async fn test_concurrent_operations() {
172
+ let futures = vec![
173
+ fetch_data_async(),
174
+ fetch_data_async(),
175
+ fetch_data_async(),
176
+ ];
177
+
178
+ let results = futures::future::join_all(futures).await;
179
+
180
+ assert_eq!(results.len(), 3);
181
+ for result in results {
182
+ assert!(result.is_ok());
183
+ }
184
+ }
185
+
186
+ async fn fetch_data_async() -> Result<String, String> {
187
+ tokio::time::sleep(tokio::time::Duration::from_millis(100)).await;
188
+ Ok("data".to_string())
189
+ }
190
+ }
191
+ ```
192
+
193
+ ### Testing with Database
194
+
195
+ ```rust
196
+ #[cfg(test)]
197
+ mod database_tests {
198
+ use super::*;
199
+ use sqlx::SqlitePool;
200
+
201
+ async fn setup_test_db() -> SqlitePool {
202
+ let pool = SqlitePool::connect(":memory:")
203
+ .await
204
+ .expect("Failed to create in-memory database");
205
+
206
+ // Run migrations
207
+ sqlx::query(
208
+ "CREATE TABLE users (
209
+ id INTEGER PRIMARY KEY,
210
+ name TEXT NOT NULL,
211
+ email TEXT NOT NULL
212
+ )"
213
+ )
214
+ .execute(&pool)
215
+ .await
216
+ .expect("Failed to create table");
217
+
218
+ pool
219
+ }
220
+
221
+ #[tokio::test]
222
+ async fn test_create_user() {
223
+ let pool = setup_test_db().await;
224
+
225
+ let result = sqlx::query(
226
+ "INSERT INTO users (name, email) VALUES (?, ?)"
227
+ )
228
+ .bind("John Doe")
229
+ .bind("john@example.com")
230
+ .execute(&pool)
231
+ .await;
232
+
233
+ assert!(result.is_ok());
234
+
235
+ let count: (i64,) = sqlx::query_as("SELECT COUNT(*) FROM users")
236
+ .fetch_one(&pool)
237
+ .await
238
+ .unwrap();
239
+
240
+ assert_eq!(count.0, 1);
241
+ }
242
+ }
243
+ ```
244
+
245
+ ## UI Testing Approaches
246
+
247
+ ### egui Testing
248
+
249
+ ```rust
250
+ #[cfg(test)]
251
+ mod ui_tests {
252
+ use super::*;
253
+ use eframe::egui;
254
+
255
+ #[test]
256
+ fn test_ui_state_update() {
257
+ let mut app = MyApp::default();
258
+
259
+ // Simulate UI interaction
260
+ app.counter = 0;
261
+
262
+ // Trigger button click logic
263
+ app.counter += 1;
264
+
265
+ assert_eq!(app.counter, 1);
266
+ }
267
+
268
+ #[test]
269
+ fn test_ui_rendering() {
270
+ let mut app = MyApp::default();
271
+ let ctx = egui::Context::default();
272
+
273
+ // Test that UI can be rendered without panic
274
+ app.update(&ctx, &mut eframe::Frame::new(eframe::FrameData {
275
+ info: eframe::IntegrationInfo::default(),
276
+ output: Default::default(),
277
+ repaint_signal: std::sync::Arc::new(eframe::RepaintSignal::default()),
278
+ }));
279
+ }
280
+ }
281
+ ```
282
+
283
+ ### Tauri Frontend-Backend Integration Tests
284
+
285
+ ```typescript
286
+ // src/tests/integration.test.ts
287
+ import { invoke } from '@tauri-apps/api/core';
288
+
289
+ describe('User Management', () => {
290
+ beforeEach(async () => {
291
+ // Reset state
292
+ await invoke('reset_users');
293
+ });
294
+
295
+ test('should add user', async () => {
296
+ const user = await invoke<User>('add_user', {
297
+ name: 'John Doe',
298
+ email: 'john@example.com',
299
+ });
300
+
301
+ expect(user.name).toBe('John Doe');
302
+ expect(user.email).toBe('john@example.com');
303
+ });
304
+
305
+ test('should reject invalid email', async () => {
306
+ await expect(
307
+ invoke('add_user', {
308
+ name: 'John Doe',
309
+ email: 'invalid',
310
+ })
311
+ ).rejects.toThrow('Invalid email');
312
+ });
313
+
314
+ test('should get all users', async () => {
315
+ await invoke('add_user', {
316
+ name: 'Alice',
317
+ email: 'alice@example.com',
318
+ });
319
+
320
+ await invoke('add_user', {
321
+ name: 'Bob',
322
+ email: 'bob@example.com',
323
+ });
324
+
325
+ const users = await invoke<User[]>('get_all_users');
326
+ expect(users).toHaveLength(2);
327
+ });
328
+ });
329
+ ```
330
+
331
+ ### E2E Testing with WebDriver
332
+
333
+ ```typescript
334
+ // tests/e2e/app.spec.ts
335
+ import { test, expect } from '@playwright/test';
336
+
337
+ test.describe('Desktop App E2E', () => {
338
+ test('should load application', async ({ page }) => {
339
+ await page.goto('http://localhost:1420'); // Tauri dev server
340
+
341
+ await expect(page.locator('h1')).toContainText('My App');
342
+ });
343
+
344
+ test('should add user through UI', async ({ page }) => {
345
+ await page.goto('http://localhost:1420');
346
+
347
+ await page.fill('input[name="name"]', 'John Doe');
348
+ await page.fill('input[name="email"]', 'john@example.com');
349
+ await page.click('button:has-text("Add User")');
350
+
351
+ await expect(page.locator('.user-item')).toContainText('John Doe');
352
+ });
353
+
354
+ test('should show error for invalid input', async ({ page }) => {
355
+ await page.goto('http://localhost:1420');
356
+
357
+ await page.fill('input[name="name"]', '');
358
+ await page.fill('input[name="email"]', 'invalid');
359
+ await page.click('button:has-text("Add User")');
360
+
361
+ await expect(page.locator('.error')).toBeVisible();
362
+ });
363
+ });
364
+ ```
365
+
366
+ ## Cross-Compilation and Builds
367
+
368
+ ### Build Scripts
369
+
370
+ ```bash
371
+ #!/bin/bash
372
+ # scripts/build.sh
373
+
374
+ set -e
375
+
376
+ TARGET_TRIPLE="${1:-}"
377
+ BUILD_TYPE="${2:-release}"
378
+
379
+ if [ -z "$TARGET_TRIPLE" ]; then
380
+ echo "Building for current platform..."
381
+ cargo tauri build
382
+ else
383
+ echo "Building for $TARGET_TRIPLE..."
384
+
385
+ # Install target if needed
386
+ rustup target add "$TARGET_TRIPLE"
387
+
388
+ # Build
389
+ if [ "$BUILD_TYPE" = "debug" ]; then
390
+ cargo tauri build --debug --target "$TARGET_TRIPLE"
391
+ else
392
+ cargo tauri build --target "$TARGET_TRIPLE"
393
+ fi
394
+ fi
395
+
396
+ echo "Build completed successfully!"
397
+ ```
398
+
399
+ ### Cross-Compilation Setup
400
+
401
+ **For Windows from Linux/macOS:**
402
+ ```bash
403
+ # Install cross-compilation tools
404
+ cargo install cargo-xwin
405
+
406
+ # Add Windows target
407
+ rustup target add x86_64-pc-windows-msvc
408
+
409
+ # Build for Windows
410
+ cargo xwin build --release --target x86_64-pc-windows-msvc
411
+ ```
412
+
413
+ **For macOS from Linux:**
414
+ ```bash
415
+ # Install osxcross (complex setup, see osxcross documentation)
416
+ # Add macOS targets
417
+ rustup target add x86_64-apple-darwin
418
+ rustup target add aarch64-apple-darwin
419
+
420
+ # Build for macOS
421
+ CROSS_COMPILE=x86_64-apple-darwin- \
422
+ cargo build --release --target x86_64-apple-darwin
423
+ ```
424
+
425
+ **For Linux from macOS/Windows:**
426
+ ```bash
427
+ # Install cross
428
+ cargo install cross
429
+
430
+ # Build for Linux
431
+ cross build --release --target x86_64-unknown-linux-gnu
432
+ ```
433
+
434
+ ### Universal Binaries (macOS)
435
+
436
+ ```bash
437
+ #!/bin/bash
438
+ # scripts/build-universal-macos.sh
439
+
440
+ set -e
441
+
442
+ echo "Building universal macOS binary..."
443
+
444
+ # Build for both architectures
445
+ cargo tauri build --target x86_64-apple-darwin
446
+ cargo tauri build --target aarch64-apple-darwin
447
+
448
+ # Create universal binary
449
+ lipo -create \
450
+ target/x86_64-apple-darwin/release/bundle/macos/MyApp.app/Contents/MacOS/MyApp \
451
+ target/aarch64-apple-darwin/release/bundle/macos/MyApp.app/Contents/MacOS/MyApp \
452
+ -output target/universal-apple-darwin/release/MyApp
453
+
454
+ echo "Universal binary created!"
455
+ ```
456
+
457
+ ## Platform-Specific Installers
458
+
459
+ ### Windows Installer (WiX)
460
+
461
+ **wix/main.wxs:**
462
+ ```xml
463
+ <?xml version="1.0" encoding="UTF-8"?>
464
+ <Wix xmlns="http://schemas.microsoft.com/wix/2006/wi">
465
+ <Product
466
+ Id="*"
467
+ Name="My Application"
468
+ Language="1033"
469
+ Version="1.0.0"
470
+ Manufacturer="My Company"
471
+ UpgradeCode="PUT-GUID-HERE">
472
+
473
+ <Package
474
+ InstallerVersion="200"
475
+ Compressed="yes"
476
+ InstallScope="perMachine" />
477
+
478
+ <MajorUpgrade
479
+ DowngradeErrorMessage="A newer version is already installed." />
480
+
481
+ <MediaTemplate EmbedCab="yes" />
482
+
483
+ <Feature
484
+ Id="MainApplication"
485
+ Title="My Application"
486
+ Level="1">
487
+ <ComponentGroupRef Id="AppFiles" />
488
+ </Feature>
489
+
490
+ <Directory Id="TARGETDIR" Name="SourceDir">
491
+ <Directory Id="ProgramFilesFolder">
492
+ <Directory Id="INSTALLFOLDER" Name="MyApp" />
493
+ </Directory>
494
+ <Directory Id="ProgramMenuFolder">
495
+ <Directory Id="ApplicationProgramsFolder" Name="MyApp"/>
496
+ </Directory>
497
+ </Directory>
498
+
499
+ <ComponentGroup Id="AppFiles" Directory="INSTALLFOLDER">
500
+ <Component Id="MainExecutable">
501
+ <File
502
+ Id="MainExe"
503
+ Source="$(var.SourceDir)\MyApp.exe"
504
+ KeyPath="yes">
505
+ <Shortcut
506
+ Id="ApplicationStartMenuShortcut"
507
+ Directory="ApplicationProgramsFolder"
508
+ Name="MyApp"
509
+ Description="My Application"
510
+ WorkingDirectory="INSTALLFOLDER"
511
+ Icon="AppIcon.exe"
512
+ IconIndex="0"
513
+ Advertise="yes" />
514
+ </File>
515
+ </Component>
516
+ </ComponentGroup>
517
+ </Product>
518
+ </Wix>
519
+ ```
520
+
521
+ ### macOS DMG Creation
522
+
523
+ ```bash
524
+ #!/bin/bash
525
+ # scripts/create-dmg.sh
526
+
527
+ set -e
528
+
529
+ APP_NAME="MyApp"
530
+ VERSION="1.0.0"
531
+ DMG_NAME="${APP_NAME}-${VERSION}.dmg"
532
+ SOURCE_FOLDER="target/release/bundle/macos/${APP_NAME}.app"
533
+ DMG_FOLDER="dmg_temp"
534
+
535
+ echo "Creating DMG for ${APP_NAME}..."
536
+
537
+ # Create temporary folder
538
+ mkdir -p "$DMG_FOLDER"
539
+
540
+ # Copy app
541
+ cp -R "$SOURCE_FOLDER" "$DMG_FOLDER/"
542
+
543
+ # Create symbolic link to Applications
544
+ ln -s /Applications "$DMG_FOLDER/Applications"
545
+
546
+ # Create DMG
547
+ hdiutil create -volname "$APP_NAME" \
548
+ -srcfolder "$DMG_FOLDER" \
549
+ -ov \
550
+ -format UDZO \
551
+ "$DMG_NAME"
552
+
553
+ # Cleanup
554
+ rm -rf "$DMG_FOLDER"
555
+
556
+ echo "DMG created: $DMG_NAME"
557
+ ```
558
+
559
+ ### Linux Packages
560
+
561
+ **Debian Package (.deb):**
562
+ ```bash
563
+ #!/bin/bash
564
+ # scripts/build-deb.sh
565
+
566
+ set -e
567
+
568
+ APP_NAME="myapp"
569
+ VERSION="1.0.0"
570
+ ARCH="amd64"
571
+
572
+ DEB_DIR="target/debian"
573
+ mkdir -p "$DEB_DIR/DEBIAN"
574
+ mkdir -p "$DEB_DIR/usr/bin"
575
+ mkdir -p "$DEB_DIR/usr/share/applications"
576
+ mkdir -p "$DEB_DIR/usr/share/icons/hicolor/256x256/apps"
577
+
578
+ # Create control file
579
+ cat > "$DEB_DIR/DEBIAN/control" << EOF
580
+ Package: $APP_NAME
581
+ Version: $VERSION
582
+ Section: utils
583
+ Priority: optional
584
+ Architecture: $ARCH
585
+ Maintainer: Your Name <your@email.com>
586
+ Description: My Application
587
+ A desktop application built with Tauri
588
+ EOF
589
+
590
+ # Copy binary
591
+ cp "target/release/$APP_NAME" "$DEB_DIR/usr/bin/"
592
+
593
+ # Create desktop entry
594
+ cat > "$DEB_DIR/usr/share/applications/$APP_NAME.desktop" << EOF
595
+ [Desktop Entry]
596
+ Name=My App
597
+ Exec=/usr/bin/$APP_NAME
598
+ Icon=$APP_NAME
599
+ Type=Application
600
+ Categories=Utility;
601
+ EOF
602
+
603
+ # Copy icon
604
+ cp "icons/icon.png" "$DEB_DIR/usr/share/icons/hicolor/256x256/apps/$APP_NAME.png"
605
+
606
+ # Build package
607
+ dpkg-deb --build "$DEB_DIR" "${APP_NAME}_${VERSION}_${ARCH}.deb"
608
+
609
+ echo "Debian package created!"
610
+ ```
611
+
612
+ **AppImage:**
613
+ ```bash
614
+ #!/bin/bash
615
+ # scripts/build-appimage.sh
616
+
617
+ set -e
618
+
619
+ APP_NAME="MyApp"
620
+ APPDIR="AppDir"
621
+
622
+ # Create AppDir structure
623
+ mkdir -p "$APPDIR/usr/bin"
624
+ mkdir -p "$APPDIR/usr/share/applications"
625
+ mkdir -p "$APPDIR/usr/share/icons/hicolor/256x256/apps"
626
+
627
+ # Copy files
628
+ cp "target/release/myapp" "$APPDIR/usr/bin/"
629
+ cp "myapp.desktop" "$APPDIR/usr/share/applications/"
630
+ cp "icons/icon.png" "$APPDIR/usr/share/icons/hicolor/256x256/apps/myapp.png"
631
+
632
+ # Create AppRun
633
+ cat > "$APPDIR/AppRun" << 'EOF'
634
+ #!/bin/bash
635
+ SELF=$(readlink -f "$0")
636
+ HERE=${SELF%/*}
637
+ export PATH="${HERE}/usr/bin:${PATH}"
638
+ exec "${HERE}/usr/bin/myapp" "$@"
639
+ EOF
640
+ chmod +x "$APPDIR/AppRun"
641
+
642
+ # Download appimagetool
643
+ wget -O appimagetool "https://github.com/AppImage/AppImageKit/releases/download/continuous/appimagetool-x86_64.AppImage"
644
+ chmod +x appimagetool
645
+
646
+ # Build AppImage
647
+ ./appimagetool "$APPDIR" "${APP_NAME}.AppImage"
648
+
649
+ echo "AppImage created!"
650
+ ```
651
+
652
+ ## Code Signing and Notarization
653
+
654
+ ### macOS Code Signing
655
+
656
+ ```bash
657
+ #!/bin/bash
658
+ # scripts/sign-macos.sh
659
+
660
+ set -e
661
+
662
+ APP_PATH="$1"
663
+ IDENTITY="Developer ID Application: Your Name (TEAMID)"
664
+
665
+ echo "Signing $APP_PATH..."
666
+
667
+ # Sign all frameworks and dylibs first
668
+ find "$APP_PATH/Contents" -name "*.framework" -or -name "*.dylib" | while read file; do
669
+ codesign --force --sign "$IDENTITY" \
670
+ --options runtime \
671
+ --timestamp \
672
+ "$file"
673
+ done
674
+
675
+ # Sign the app bundle
676
+ codesign --force --deep --sign "$IDENTITY" \
677
+ --options runtime \
678
+ --entitlements entitlements.plist \
679
+ --timestamp \
680
+ "$APP_PATH"
681
+
682
+ # Verify signature
683
+ codesign --verify --verbose=4 "$APP_PATH"
684
+
685
+ echo "Signing completed!"
686
+ ```
687
+
688
+ **entitlements.plist:**
689
+ ```xml
690
+ <?xml version="1.0" encoding="UTF-8"?>
691
+ <!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
692
+ <plist version="1.0">
693
+ <dict>
694
+ <key>com.apple.security.cs.allow-unsigned-executable-memory</key>
695
+ <true/>
696
+ <key>com.apple.security.cs.allow-jit</key>
697
+ <true/>
698
+ <key>com.apple.security.cs.disable-library-validation</key>
699
+ <true/>
700
+ </dict>
701
+ </plist>
702
+ ```
703
+
704
+ ### macOS Notarization
705
+
706
+ ```bash
707
+ #!/bin/bash
708
+ # scripts/notarize-macos.sh
709
+
710
+ set -e
711
+
712
+ APP_PATH="$1"
713
+ BUNDLE_ID="com.yourcompany.myapp"
714
+ APPLE_ID="your@email.com"
715
+ TEAM_ID="TEAMID"
716
+ APP_SPECIFIC_PASSWORD="xxxx-xxxx-xxxx-xxxx"
717
+
718
+ # Create DMG
719
+ DMG_PATH="MyApp.dmg"
720
+ hdiutil create -volname "MyApp" \
721
+ -srcfolder "$APP_PATH" \
722
+ -ov \
723
+ -format UDZO \
724
+ "$DMG_PATH"
725
+
726
+ # Submit for notarization
727
+ echo "Submitting for notarization..."
728
+ xcrun notarytool submit "$DMG_PATH" \
729
+ --apple-id "$APPLE_ID" \
730
+ --team-id "$TEAM_ID" \
731
+ --password "$APP_SPECIFIC_PASSWORD" \
732
+ --wait
733
+
734
+ # Staple the ticket
735
+ echo "Stapling ticket..."
736
+ xcrun stapler staple "$DMG_PATH"
737
+
738
+ echo "Notarization completed!"
739
+ ```
740
+
741
+ ### Windows Code Signing
742
+
743
+ ```powershell
744
+ # scripts/sign-windows.ps1
745
+
746
+ param(
747
+ [string]$FilePath,
748
+ [string]$CertPath,
749
+ [string]$Password
750
+ )
751
+
752
+ # Sign executable
753
+ & "C:\Program Files (x86)\Windows Kits\10\bin\10.0.19041.0\x64\signtool.exe" sign `
754
+ /f $CertPath `
755
+ /p $Password `
756
+ /tr http://timestamp.digicert.com `
757
+ /td sha256 `
758
+ /fd sha256 `
759
+ $FilePath
760
+
761
+ # Verify signature
762
+ & "C:\Program Files (x86)\Windows Kits\10\bin\10.0.19041.0\x64\signtool.exe" verify /pa $FilePath
763
+
764
+ Write-Host "Signing completed!"
765
+ ```
766
+
767
+ ## CI/CD Pipelines
768
+
769
+ ### GitHub Actions Workflow
770
+
771
+ ```yaml
772
+ # .github/workflows/build.yml
773
+ name: Build and Release
774
+
775
+ on:
776
+ push:
777
+ tags:
778
+ - 'v*'
779
+
780
+ jobs:
781
+ build:
782
+ strategy:
783
+ fail-fast: false
784
+ matrix:
785
+ platform: [macos-latest, ubuntu-20.04, windows-latest]
786
+
787
+ runs-on: ${{ matrix.platform }}
788
+
789
+ steps:
790
+ - name: Checkout code
791
+ uses: actions/checkout@v4
792
+
793
+ - name: Setup Node.js
794
+ uses: actions/setup-node@v4
795
+ with:
796
+ node-version: 20
797
+
798
+ - name: Setup Rust
799
+ uses: dtolnay/rust-toolchain@stable
800
+
801
+ - name: Install dependencies (Ubuntu only)
802
+ if: matrix.platform == 'ubuntu-20.04'
803
+ run: |
804
+ sudo apt-get update
805
+ sudo apt-get install -y libgtk-3-dev libwebkit2gtk-4.0-dev libappindicator3-dev librsvg2-dev patchelf
806
+
807
+ - name: Install frontend dependencies
808
+ run: npm install
809
+
810
+ - name: Build application
811
+ run: npm run tauri build
812
+
813
+ - name: Upload artifacts
814
+ uses: actions/upload-artifact@v4
815
+ with:
816
+ name: ${{ matrix.platform }}-build
817
+ path: |
818
+ src-tauri/target/release/bundle/*
819
+
820
+ release:
821
+ needs: build
822
+ runs-on: ubuntu-latest
823
+ steps:
824
+ - name: Download artifacts
825
+ uses: actions/download-artifact@v4
826
+
827
+ - name: Create Release
828
+ uses: softprops/action-gh-release@v1
829
+ with:
830
+ files: |
831
+ macos-latest-build/**/*
832
+ ubuntu-20.04-build/**/*
833
+ windows-latest-build/**/*
834
+ env:
835
+ GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
836
+ ```
837
+
838
+ ### Multi-Platform Build Matrix
839
+
840
+ ```yaml
841
+ # .github/workflows/release.yml
842
+ name: Release
843
+
844
+ on:
845
+ workflow_dispatch:
846
+ push:
847
+ tags:
848
+ - 'v*'
849
+
850
+ jobs:
851
+ build:
852
+ strategy:
853
+ matrix:
854
+ include:
855
+ - os: ubuntu-20.04
856
+ target: x86_64-unknown-linux-gnu
857
+ artifact: deb, appimage
858
+ - os: macos-latest
859
+ target: x86_64-apple-darwin
860
+ artifact: dmg, app
861
+ - os: macos-latest
862
+ target: aarch64-apple-darwin
863
+ artifact: dmg, app
864
+ - os: windows-latest
865
+ target: x86_64-pc-windows-msvc
866
+ artifact: msi, exe
867
+
868
+ runs-on: ${{ matrix.os }}
869
+
870
+ steps:
871
+ - uses: actions/checkout@v4
872
+
873
+ - name: Setup Rust
874
+ uses: dtolnay/rust-toolchain@stable
875
+ with:
876
+ targets: ${{ matrix.target }}
877
+
878
+ - name: Build
879
+ run: cargo tauri build --target ${{ matrix.target }}
880
+
881
+ - name: Upload artifacts
882
+ uses: actions/upload-artifact@v4
883
+ with:
884
+ name: ${{ matrix.target }}
885
+ path: src-tauri/target/${{ matrix.target }}/release/bundle/
886
+ ```
887
+
888
+ ### Auto-Update Server
889
+
890
+ ```rust
891
+ // Simple update server
892
+ use actix_web::{web, App, HttpResponse, HttpServer};
893
+ use serde::Serialize;
894
+
895
+ #[derive(Serialize)]
896
+ struct UpdateManifest {
897
+ version: String,
898
+ notes: String,
899
+ pub_date: String,
900
+ platforms: Platforms,
901
+ }
902
+
903
+ #[derive(Serialize)]
904
+ struct Platforms {
905
+ #[serde(rename = "darwin-x86_64")]
906
+ darwin_x86_64: Platform,
907
+ #[serde(rename = "darwin-aarch64")]
908
+ darwin_aarch64: Platform,
909
+ #[serde(rename = "linux-x86_64")]
910
+ linux_x86_64: Platform,
911
+ #[serde(rename = "windows-x86_64")]
912
+ windows_x86_64: Platform,
913
+ }
914
+
915
+ #[derive(Serialize)]
916
+ struct Platform {
917
+ signature: String,
918
+ url: String,
919
+ }
920
+
921
+ async fn update_manifest() -> HttpResponse {
922
+ let manifest = UpdateManifest {
923
+ version: "1.0.1".to_string(),
924
+ notes: "Bug fixes and improvements".to_string(),
925
+ pub_date: "2024-01-15T00:00:00Z".to_string(),
926
+ platforms: Platforms {
927
+ darwin_x86_64: Platform {
928
+ signature: "BASE64_SIGNATURE".to_string(),
929
+ url: "https://releases.myapp.com/myapp-1.0.1-x64.dmg".to_string(),
930
+ },
931
+ darwin_aarch64: Platform {
932
+ signature: "BASE64_SIGNATURE".to_string(),
933
+ url: "https://releases.myapp.com/myapp-1.0.1-arm64.dmg".to_string(),
934
+ },
935
+ linux_x86_64: Platform {
936
+ signature: "BASE64_SIGNATURE".to_string(),
937
+ url: "https://releases.myapp.com/myapp-1.0.1-amd64.AppImage".to_string(),
938
+ },
939
+ windows_x86_64: Platform {
940
+ signature: "BASE64_SIGNATURE".to_string(),
941
+ url: "https://releases.myapp.com/myapp-1.0.1-x64-setup.exe".to_string(),
942
+ },
943
+ },
944
+ };
945
+
946
+ HttpResponse::Ok().json(manifest)
947
+ }
948
+
949
+ #[actix_web::main]
950
+ async fn main() -> std::io::Result<()> {
951
+ HttpServer::new(|| {
952
+ App::new()
953
+ .route("/update-manifest.json", web::get().to(update_manifest))
954
+ })
955
+ .bind(("127.0.0.1", 8080))?
956
+ .run()
957
+ .await
958
+ }
959
+ ```
960
+
961
+ This comprehensive testing and deployment guide covers everything from unit tests to production CI/CD pipelines, enabling reliable cross-platform distribution of Rust desktop applications.