claude-mpm 5.4.41__py3-none-any.whl → 5.6.23__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 (460) hide show
  1. claude_mpm/VERSION +1 -1
  2. claude_mpm/agents/CLAUDE_MPM_OUTPUT_STYLE.md +66 -241
  3. claude_mpm/agents/CLAUDE_MPM_RESEARCH_OUTPUT_STYLE.md +413 -0
  4. claude_mpm/agents/CLAUDE_MPM_TEACHER_OUTPUT_STYLE.md +109 -1925
  5. claude_mpm/agents/PM_INSTRUCTIONS.md +161 -298
  6. claude_mpm/agents/WORKFLOW.md +2 -0
  7. claude_mpm/agents/templates/circuit-breakers.md +26 -17
  8. claude_mpm/cli/__init__.py +5 -1
  9. claude_mpm/cli/commands/agents.py +2 -4
  10. claude_mpm/cli/commands/agents_reconcile.py +197 -0
  11. claude_mpm/cli/commands/autotodos.py +566 -0
  12. claude_mpm/cli/commands/commander.py +216 -0
  13. claude_mpm/cli/commands/configure.py +620 -21
  14. claude_mpm/cli/commands/configure_agent_display.py +3 -1
  15. claude_mpm/cli/commands/hook_errors.py +60 -60
  16. claude_mpm/cli/commands/monitor.py +2 -2
  17. claude_mpm/cli/commands/mpm_init/core.py +15 -8
  18. claude_mpm/cli/commands/profile.py +9 -10
  19. claude_mpm/cli/commands/run.py +35 -3
  20. claude_mpm/cli/commands/skill_source.py +51 -2
  21. claude_mpm/cli/commands/skills.py +182 -32
  22. claude_mpm/cli/executor.py +120 -16
  23. claude_mpm/cli/interactive/__init__.py +10 -0
  24. claude_mpm/cli/interactive/agent_wizard.py +30 -50
  25. claude_mpm/cli/interactive/questionary_styles.py +65 -0
  26. claude_mpm/cli/interactive/skill_selector.py +481 -0
  27. claude_mpm/cli/parsers/base_parser.py +76 -1
  28. claude_mpm/cli/parsers/commander_parser.py +116 -0
  29. claude_mpm/cli/parsers/profile_parser.py +0 -1
  30. claude_mpm/cli/parsers/run_parser.py +10 -0
  31. claude_mpm/cli/parsers/skill_source_parser.py +4 -0
  32. claude_mpm/cli/parsers/skills_parser.py +2 -3
  33. claude_mpm/cli/startup.py +527 -506
  34. claude_mpm/cli/startup_display.py +74 -6
  35. claude_mpm/cli/startup_logging.py +2 -2
  36. claude_mpm/cli/utils.py +7 -3
  37. claude_mpm/commander/__init__.py +78 -0
  38. claude_mpm/commander/adapters/__init__.py +60 -0
  39. claude_mpm/commander/adapters/auggie.py +260 -0
  40. claude_mpm/commander/adapters/base.py +288 -0
  41. claude_mpm/commander/adapters/claude_code.py +392 -0
  42. claude_mpm/commander/adapters/codex.py +237 -0
  43. claude_mpm/commander/adapters/communication.py +366 -0
  44. claude_mpm/commander/adapters/example_usage.py +310 -0
  45. claude_mpm/commander/adapters/mpm.py +389 -0
  46. claude_mpm/commander/adapters/registry.py +204 -0
  47. claude_mpm/commander/api/__init__.py +16 -0
  48. claude_mpm/commander/api/app.py +121 -0
  49. claude_mpm/commander/api/errors.py +133 -0
  50. claude_mpm/commander/api/routes/__init__.py +8 -0
  51. claude_mpm/commander/api/routes/events.py +184 -0
  52. claude_mpm/commander/api/routes/inbox.py +171 -0
  53. claude_mpm/commander/api/routes/messages.py +148 -0
  54. claude_mpm/commander/api/routes/projects.py +271 -0
  55. claude_mpm/commander/api/routes/sessions.py +226 -0
  56. claude_mpm/commander/api/routes/work.py +296 -0
  57. claude_mpm/commander/api/schemas.py +186 -0
  58. claude_mpm/commander/chat/__init__.py +7 -0
  59. claude_mpm/commander/chat/cli.py +146 -0
  60. claude_mpm/commander/chat/commands.py +96 -0
  61. claude_mpm/commander/chat/repl.py +310 -0
  62. claude_mpm/commander/config.py +51 -0
  63. claude_mpm/commander/config_loader.py +115 -0
  64. claude_mpm/commander/core/__init__.py +10 -0
  65. claude_mpm/commander/core/block_manager.py +325 -0
  66. claude_mpm/commander/core/response_manager.py +323 -0
  67. claude_mpm/commander/daemon.py +603 -0
  68. claude_mpm/commander/env_loader.py +59 -0
  69. claude_mpm/commander/events/__init__.py +26 -0
  70. claude_mpm/commander/events/manager.py +332 -0
  71. claude_mpm/commander/frameworks/__init__.py +12 -0
  72. claude_mpm/commander/frameworks/base.py +146 -0
  73. claude_mpm/commander/frameworks/claude_code.py +58 -0
  74. claude_mpm/commander/frameworks/mpm.py +62 -0
  75. claude_mpm/commander/inbox/__init__.py +16 -0
  76. claude_mpm/commander/inbox/dedup.py +128 -0
  77. claude_mpm/commander/inbox/inbox.py +224 -0
  78. claude_mpm/commander/inbox/models.py +70 -0
  79. claude_mpm/commander/instance_manager.py +450 -0
  80. claude_mpm/commander/llm/__init__.py +6 -0
  81. claude_mpm/commander/llm/openrouter_client.py +167 -0
  82. claude_mpm/commander/llm/summarizer.py +70 -0
  83. claude_mpm/commander/memory/__init__.py +45 -0
  84. claude_mpm/commander/memory/compression.py +347 -0
  85. claude_mpm/commander/memory/embeddings.py +230 -0
  86. claude_mpm/commander/memory/entities.py +310 -0
  87. claude_mpm/commander/memory/example_usage.py +290 -0
  88. claude_mpm/commander/memory/integration.py +325 -0
  89. claude_mpm/commander/memory/search.py +381 -0
  90. claude_mpm/commander/memory/store.py +657 -0
  91. claude_mpm/commander/models/__init__.py +18 -0
  92. claude_mpm/commander/models/events.py +121 -0
  93. claude_mpm/commander/models/project.py +162 -0
  94. claude_mpm/commander/models/work.py +214 -0
  95. claude_mpm/commander/parsing/__init__.py +20 -0
  96. claude_mpm/commander/parsing/extractor.py +132 -0
  97. claude_mpm/commander/parsing/output_parser.py +270 -0
  98. claude_mpm/commander/parsing/patterns.py +100 -0
  99. claude_mpm/commander/persistence/__init__.py +11 -0
  100. claude_mpm/commander/persistence/event_store.py +274 -0
  101. claude_mpm/commander/persistence/state_store.py +309 -0
  102. claude_mpm/commander/persistence/work_store.py +164 -0
  103. claude_mpm/commander/polling/__init__.py +13 -0
  104. claude_mpm/commander/polling/event_detector.py +104 -0
  105. claude_mpm/commander/polling/output_buffer.py +49 -0
  106. claude_mpm/commander/polling/output_poller.py +153 -0
  107. claude_mpm/commander/project_session.py +268 -0
  108. claude_mpm/commander/proxy/__init__.py +12 -0
  109. claude_mpm/commander/proxy/formatter.py +89 -0
  110. claude_mpm/commander/proxy/output_handler.py +191 -0
  111. claude_mpm/commander/proxy/relay.py +155 -0
  112. claude_mpm/commander/registry.py +410 -0
  113. claude_mpm/commander/runtime/__init__.py +10 -0
  114. claude_mpm/commander/runtime/executor.py +191 -0
  115. claude_mpm/commander/runtime/monitor.py +346 -0
  116. claude_mpm/commander/session/__init__.py +6 -0
  117. claude_mpm/commander/session/context.py +81 -0
  118. claude_mpm/commander/session/manager.py +59 -0
  119. claude_mpm/commander/tmux_orchestrator.py +361 -0
  120. claude_mpm/commander/web/__init__.py +1 -0
  121. claude_mpm/commander/work/__init__.py +30 -0
  122. claude_mpm/commander/work/executor.py +207 -0
  123. claude_mpm/commander/work/queue.py +405 -0
  124. claude_mpm/commander/workflow/__init__.py +27 -0
  125. claude_mpm/commander/workflow/event_handler.py +241 -0
  126. claude_mpm/commander/workflow/notifier.py +146 -0
  127. claude_mpm/commands/mpm-config.md +8 -0
  128. claude_mpm/commands/mpm-doctor.md +8 -0
  129. claude_mpm/commands/mpm-help.md +8 -0
  130. claude_mpm/commands/mpm-init.md +8 -0
  131. claude_mpm/commands/mpm-monitor.md +8 -0
  132. claude_mpm/commands/mpm-organize.md +8 -0
  133. claude_mpm/commands/mpm-postmortem.md +8 -0
  134. claude_mpm/commands/mpm-session-resume.md +9 -1
  135. claude_mpm/commands/mpm-status.md +8 -0
  136. claude_mpm/commands/mpm-ticket-view.md +8 -0
  137. claude_mpm/commands/mpm-version.md +8 -0
  138. claude_mpm/commands/mpm.md +8 -0
  139. claude_mpm/config/agent_presets.py +8 -7
  140. claude_mpm/config/skill_sources.py +16 -0
  141. claude_mpm/constants.py +1 -0
  142. claude_mpm/core/claude_runner.py +154 -2
  143. claude_mpm/core/config.py +35 -22
  144. claude_mpm/core/config_constants.py +74 -9
  145. claude_mpm/core/constants.py +56 -12
  146. claude_mpm/core/hook_manager.py +51 -3
  147. claude_mpm/core/interactive_session.py +12 -11
  148. claude_mpm/core/logger.py +26 -9
  149. claude_mpm/core/logging_utils.py +35 -11
  150. claude_mpm/core/network_config.py +148 -0
  151. claude_mpm/core/oneshot_session.py +7 -6
  152. claude_mpm/core/optimized_startup.py +3 -1
  153. claude_mpm/core/output_style_manager.py +63 -18
  154. claude_mpm/core/shared/config_loader.py +3 -1
  155. claude_mpm/core/socketio_pool.py +13 -5
  156. claude_mpm/core/unified_config.py +54 -8
  157. claude_mpm/core/unified_paths.py +95 -90
  158. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/0.C33zOoyM.css +1 -0
  159. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/2.CW1J-YuA.css +1 -0
  160. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/1WZnGYqX.js +24 -0
  161. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/67pF3qNn.js +1 -0
  162. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/6RxdMKe4.js +1 -0
  163. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/8cZrfX0h.js +60 -0
  164. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/9a6T2nm-.js +7 -0
  165. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/B443AUzu.js +1 -0
  166. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/B8AwtY2H.js +1 -0
  167. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BF15LAsF.js +1 -0
  168. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BQaXIfA_.js +331 -0
  169. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BRcwIQNr.js +4 -0
  170. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{uj46x2Wr.js → BSNlmTZj.js} +1 -1
  171. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BV6nKitt.js +43 -0
  172. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BViJ8lZt.js +128 -0
  173. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BcQ-Q0FE.js +1 -0
  174. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Bpyvgze_.js +30 -0
  175. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BzTRqg-z.js +1 -0
  176. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C0Fr8dve.js +1 -0
  177. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C3rbW_a-.js +1 -0
  178. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C8WYN38h.js +1 -0
  179. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C9I8FlXH.js +61 -0
  180. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CIQcWgO2.js +36 -0
  181. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CIctN7YN.js +7 -0
  182. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CKrS_JZW.js +145 -0
  183. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CR6P9C4A.js +89 -0
  184. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CRRR9MD_.js +2 -0
  185. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CRcR2DqT.js +334 -0
  186. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CSXtMOf0.js +1 -0
  187. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CT-sbxSk.js +1 -0
  188. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CWm6DJsp.js +1 -0
  189. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CmKTTxBW.js +1 -0
  190. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CpqQ1Kzn.js +1 -0
  191. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Cu_Erd72.js +261 -0
  192. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/D2nGpDRe.js +1 -0
  193. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/D9iCMida.js +267 -0
  194. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/D9ykgMoY.js +10 -0
  195. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DL2Ldur1.js +1 -0
  196. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DPfltzjH.js +165 -0
  197. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{N4qtv3Hx.js → DR8nis88.js} +2 -2
  198. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DUliQN2b.js +1 -0
  199. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DVp1hx9R.js +1 -0
  200. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DXlhR01x.js +122 -0
  201. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/D_lyTybS.js +1 -0
  202. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DngoTTgh.js +1 -0
  203. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DqkmHtDC.js +220 -0
  204. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DsDh8EYs.js +1 -0
  205. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DypDmXgd.js +139 -0
  206. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Gi6I4Gst.js +1 -0
  207. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/IPYC-LnN.js +162 -0
  208. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/JTLiF7dt.js +24 -0
  209. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/JpevfAFt.js +68 -0
  210. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{DjhvlsAc.js → NqQ1dWOy.js} +1 -1
  211. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/R8CEIRAd.js +2 -0
  212. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Zxy7qc-l.js +64 -0
  213. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/q9Hm6zAU.js +1 -0
  214. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/qtd3IeO4.js +15 -0
  215. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/ulBFON_C.js +65 -0
  216. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/wQVh1CoA.js +10 -0
  217. claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/app.Dr7t0z2J.js +2 -0
  218. claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/start.BGhZHUS3.js +1 -0
  219. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/{0.CAGBuiOw.js → 0.RgBboRvH.js} +1 -1
  220. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/1.DG-KkbDf.js +1 -0
  221. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/2.D_jnf-x6.js +1 -0
  222. claude_mpm/dashboard/static/svelte-build/_app/version.json +1 -1
  223. claude_mpm/dashboard/static/svelte-build/index.html +11 -11
  224. claude_mpm/dashboard-svelte/node_modules/katex/src/fonts/generate_fonts.py +58 -0
  225. claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/extract_tfms.py +114 -0
  226. claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/extract_ttfs.py +122 -0
  227. claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/format_json.py +28 -0
  228. claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/parse_tfm.py +211 -0
  229. claude_mpm/experimental/cli_enhancements.py +2 -1
  230. claude_mpm/hooks/claude_hooks/INTEGRATION_EXAMPLE.md +243 -0
  231. claude_mpm/hooks/claude_hooks/README_AUTO_PAUSE.md +403 -0
  232. claude_mpm/hooks/claude_hooks/__pycache__/auto_pause_handler.cpython-311.pyc +0 -0
  233. claude_mpm/hooks/claude_hooks/__pycache__/event_handlers.cpython-311.pyc +0 -0
  234. claude_mpm/hooks/claude_hooks/__pycache__/hook_handler.cpython-311.pyc +0 -0
  235. claude_mpm/hooks/claude_hooks/__pycache__/memory_integration.cpython-311.pyc +0 -0
  236. claude_mpm/hooks/claude_hooks/__pycache__/response_tracking.cpython-311.pyc +0 -0
  237. claude_mpm/hooks/claude_hooks/auto_pause_handler.py +485 -0
  238. claude_mpm/hooks/claude_hooks/event_handlers.py +305 -87
  239. claude_mpm/hooks/claude_hooks/hook_handler.py +106 -89
  240. claude_mpm/hooks/claude_hooks/hook_wrapper.sh +6 -11
  241. claude_mpm/hooks/claude_hooks/installer.py +116 -8
  242. claude_mpm/hooks/claude_hooks/memory_integration.py +51 -31
  243. claude_mpm/hooks/claude_hooks/response_tracking.py +42 -59
  244. claude_mpm/hooks/claude_hooks/services/__pycache__/connection_manager_http.cpython-311.pyc +0 -0
  245. claude_mpm/hooks/claude_hooks/services/__pycache__/state_manager.cpython-311.pyc +0 -0
  246. claude_mpm/hooks/claude_hooks/services/__pycache__/subagent_processor.cpython-311.pyc +0 -0
  247. claude_mpm/hooks/claude_hooks/services/connection_manager.py +39 -24
  248. claude_mpm/hooks/claude_hooks/services/connection_manager_http.py +36 -103
  249. claude_mpm/hooks/claude_hooks/services/state_manager.py +23 -36
  250. claude_mpm/hooks/claude_hooks/services/subagent_processor.py +73 -75
  251. claude_mpm/hooks/kuzu_memory_hook.py +5 -5
  252. claude_mpm/hooks/session_resume_hook.py +89 -1
  253. claude_mpm/hooks/templates/pre_tool_use_template.py +10 -2
  254. claude_mpm/init.py +215 -2
  255. claude_mpm/scripts/claude-hook-handler.sh +43 -16
  256. claude_mpm/services/agents/agent_recommendation_service.py +8 -8
  257. claude_mpm/services/agents/agent_selection_service.py +2 -2
  258. claude_mpm/services/agents/cache_git_manager.py +1 -1
  259. claude_mpm/services/agents/deployment/agent_discovery_service.py +3 -1
  260. claude_mpm/services/agents/deployment/agent_format_converter.py +25 -13
  261. claude_mpm/services/agents/deployment/agent_template_builder.py +37 -17
  262. claude_mpm/services/agents/deployment/async_agent_deployment.py +31 -27
  263. claude_mpm/services/agents/deployment/deployment_reconciler.py +577 -0
  264. claude_mpm/services/agents/deployment/local_template_deployment.py +3 -1
  265. claude_mpm/services/agents/deployment/multi_source_deployment_service.py +36 -8
  266. claude_mpm/services/agents/deployment/remote_agent_discovery_service.py +50 -26
  267. claude_mpm/services/agents/deployment/startup_reconciliation.py +138 -0
  268. claude_mpm/services/agents/git_source_manager.py +21 -2
  269. claude_mpm/services/agents/loading/framework_agent_loader.py +75 -2
  270. claude_mpm/services/agents/single_tier_deployment_service.py +4 -4
  271. claude_mpm/services/agents/sources/git_source_sync_service.py +116 -5
  272. claude_mpm/services/agents/startup_sync.py +5 -2
  273. claude_mpm/services/cli/__init__.py +3 -0
  274. claude_mpm/services/cli/incremental_pause_manager.py +561 -0
  275. claude_mpm/services/cli/session_resume_helper.py +10 -2
  276. claude_mpm/services/delegation_detector.py +175 -0
  277. claude_mpm/services/diagnostics/checks/agent_sources_check.py +30 -0
  278. claude_mpm/services/diagnostics/checks/configuration_check.py +24 -0
  279. claude_mpm/services/diagnostics/checks/installation_check.py +22 -0
  280. claude_mpm/services/diagnostics/checks/mcp_services_check.py +23 -0
  281. claude_mpm/services/diagnostics/doctor_reporter.py +31 -1
  282. claude_mpm/services/diagnostics/models.py +14 -1
  283. claude_mpm/services/event_log.py +325 -0
  284. claude_mpm/services/infrastructure/__init__.py +4 -0
  285. claude_mpm/services/infrastructure/context_usage_tracker.py +291 -0
  286. claude_mpm/services/infrastructure/resume_log_generator.py +24 -5
  287. claude_mpm/services/monitor/daemon_manager.py +15 -4
  288. claude_mpm/services/monitor/management/lifecycle.py +8 -3
  289. claude_mpm/services/monitor/server.py +106 -16
  290. claude_mpm/services/pm_skills_deployer.py +302 -94
  291. claude_mpm/services/profile_manager.py +10 -4
  292. claude_mpm/services/skills/git_skill_source_manager.py +192 -29
  293. claude_mpm/services/skills/selective_skill_deployer.py +211 -46
  294. claude_mpm/services/skills/skill_discovery_service.py +74 -4
  295. claude_mpm/services/skills_deployer.py +192 -70
  296. claude_mpm/services/socketio/handlers/hook.py +14 -7
  297. claude_mpm/services/socketio/server/main.py +12 -4
  298. claude_mpm/skills/__init__.py +2 -1
  299. claude_mpm/skills/bundled/collaboration/brainstorming/SKILL.md +79 -0
  300. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/SKILL.md +178 -0
  301. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/agent-prompts.md +577 -0
  302. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/coordination-patterns.md +467 -0
  303. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/examples.md +537 -0
  304. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/troubleshooting.md +730 -0
  305. claude_mpm/skills/bundled/collaboration/git-worktrees.md +317 -0
  306. claude_mpm/skills/bundled/collaboration/requesting-code-review/SKILL.md +112 -0
  307. claude_mpm/skills/bundled/collaboration/requesting-code-review/references/code-reviewer-template.md +146 -0
  308. claude_mpm/skills/bundled/collaboration/requesting-code-review/references/review-examples.md +412 -0
  309. claude_mpm/skills/bundled/collaboration/stacked-prs.md +251 -0
  310. claude_mpm/skills/bundled/collaboration/writing-plans/SKILL.md +81 -0
  311. claude_mpm/skills/bundled/collaboration/writing-plans/references/best-practices.md +362 -0
  312. claude_mpm/skills/bundled/collaboration/writing-plans/references/plan-structure-templates.md +312 -0
  313. claude_mpm/skills/bundled/debugging/root-cause-tracing/SKILL.md +152 -0
  314. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/advanced-techniques.md +668 -0
  315. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/examples.md +587 -0
  316. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/integration.md +438 -0
  317. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/tracing-techniques.md +391 -0
  318. claude_mpm/skills/bundled/debugging/systematic-debugging/CREATION-LOG.md +119 -0
  319. claude_mpm/skills/bundled/debugging/systematic-debugging/SKILL.md +148 -0
  320. claude_mpm/skills/bundled/debugging/systematic-debugging/references/anti-patterns.md +483 -0
  321. claude_mpm/skills/bundled/debugging/systematic-debugging/references/examples.md +452 -0
  322. claude_mpm/skills/bundled/debugging/systematic-debugging/references/troubleshooting.md +449 -0
  323. claude_mpm/skills/bundled/debugging/systematic-debugging/references/workflow.md +411 -0
  324. claude_mpm/skills/bundled/debugging/systematic-debugging/test-academic.md +14 -0
  325. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-1.md +58 -0
  326. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-2.md +68 -0
  327. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-3.md +69 -0
  328. claude_mpm/skills/bundled/debugging/verification-before-completion/SKILL.md +131 -0
  329. claude_mpm/skills/bundled/debugging/verification-before-completion/references/gate-function.md +325 -0
  330. claude_mpm/skills/bundled/debugging/verification-before-completion/references/integration-and-workflows.md +490 -0
  331. claude_mpm/skills/bundled/debugging/verification-before-completion/references/red-flags-and-failures.md +425 -0
  332. claude_mpm/skills/bundled/debugging/verification-before-completion/references/verification-patterns.md +499 -0
  333. claude_mpm/skills/bundled/infrastructure/env-manager/INTEGRATION.md +611 -0
  334. claude_mpm/skills/bundled/infrastructure/env-manager/README.md +596 -0
  335. claude_mpm/skills/bundled/infrastructure/env-manager/SKILL.md +260 -0
  336. claude_mpm/skills/bundled/infrastructure/env-manager/examples/nextjs-env-structure.md +315 -0
  337. claude_mpm/skills/bundled/infrastructure/env-manager/references/frameworks.md +436 -0
  338. claude_mpm/skills/bundled/infrastructure/env-manager/references/security.md +433 -0
  339. claude_mpm/skills/bundled/infrastructure/env-manager/references/synchronization.md +452 -0
  340. claude_mpm/skills/bundled/infrastructure/env-manager/references/troubleshooting.md +404 -0
  341. claude_mpm/skills/bundled/infrastructure/env-manager/references/validation.md +420 -0
  342. claude_mpm/skills/bundled/main/artifacts-builder/SKILL.md +86 -0
  343. claude_mpm/skills/bundled/main/internal-comms/SKILL.md +43 -0
  344. claude_mpm/skills/bundled/main/internal-comms/examples/3p-updates.md +47 -0
  345. claude_mpm/skills/bundled/main/internal-comms/examples/company-newsletter.md +65 -0
  346. claude_mpm/skills/bundled/main/internal-comms/examples/faq-answers.md +30 -0
  347. claude_mpm/skills/bundled/main/internal-comms/examples/general-comms.md +16 -0
  348. claude_mpm/skills/bundled/main/mcp-builder/SKILL.md +160 -0
  349. claude_mpm/skills/bundled/main/mcp-builder/reference/design_principles.md +412 -0
  350. claude_mpm/skills/bundled/main/mcp-builder/reference/evaluation.md +602 -0
  351. claude_mpm/skills/bundled/main/mcp-builder/reference/mcp_best_practices.md +915 -0
  352. claude_mpm/skills/bundled/main/mcp-builder/reference/node_mcp_server.md +916 -0
  353. claude_mpm/skills/bundled/main/mcp-builder/reference/python_mcp_server.md +752 -0
  354. claude_mpm/skills/bundled/main/mcp-builder/reference/workflow.md +1237 -0
  355. claude_mpm/skills/bundled/main/skill-creator/SKILL.md +189 -0
  356. claude_mpm/skills/bundled/main/skill-creator/references/best-practices.md +500 -0
  357. claude_mpm/skills/bundled/main/skill-creator/references/creation-workflow.md +464 -0
  358. claude_mpm/skills/bundled/main/skill-creator/references/examples.md +619 -0
  359. claude_mpm/skills/bundled/main/skill-creator/references/progressive-disclosure.md +437 -0
  360. claude_mpm/skills/bundled/main/skill-creator/references/skill-structure.md +231 -0
  361. claude_mpm/skills/bundled/php/espocrm-development/SKILL.md +170 -0
  362. claude_mpm/skills/bundled/php/espocrm-development/references/architecture.md +602 -0
  363. claude_mpm/skills/bundled/php/espocrm-development/references/common-tasks.md +821 -0
  364. claude_mpm/skills/bundled/php/espocrm-development/references/development-workflow.md +742 -0
  365. claude_mpm/skills/bundled/php/espocrm-development/references/frontend-customization.md +726 -0
  366. claude_mpm/skills/bundled/php/espocrm-development/references/hooks-and-services.md +764 -0
  367. claude_mpm/skills/bundled/php/espocrm-development/references/testing-debugging.md +831 -0
  368. claude_mpm/skills/bundled/pm/mpm/SKILL.md +38 -0
  369. claude_mpm/skills/bundled/pm/mpm-agent-update-workflow/SKILL.md +75 -0
  370. claude_mpm/skills/bundled/pm/mpm-bug-reporting/SKILL.md +248 -0
  371. claude_mpm/skills/bundled/pm/mpm-circuit-breaker-enforcement/SKILL.md +476 -0
  372. claude_mpm/skills/bundled/pm/mpm-config/SKILL.md +29 -0
  373. claude_mpm/skills/bundled/pm/mpm-delegation-patterns/SKILL.md +167 -0
  374. claude_mpm/skills/bundled/pm/mpm-doctor/SKILL.md +53 -0
  375. claude_mpm/skills/bundled/pm/mpm-git-file-tracking/SKILL.md +113 -0
  376. claude_mpm/skills/bundled/pm/mpm-help/SKILL.md +35 -0
  377. claude_mpm/skills/bundled/pm/mpm-init/SKILL.md +125 -0
  378. claude_mpm/skills/bundled/pm/mpm-monitor/SKILL.md +32 -0
  379. claude_mpm/skills/bundled/pm/mpm-organize/SKILL.md +121 -0
  380. claude_mpm/skills/bundled/pm/mpm-postmortem/SKILL.md +22 -0
  381. claude_mpm/skills/bundled/pm/mpm-pr-workflow/SKILL.md +124 -0
  382. claude_mpm/skills/bundled/pm/mpm-session-management/SKILL.md +312 -0
  383. claude_mpm/skills/bundled/pm/mpm-session-pause/SKILL.md +170 -0
  384. claude_mpm/skills/bundled/pm/mpm-session-resume/SKILL.md +31 -0
  385. claude_mpm/skills/bundled/pm/mpm-status/SKILL.md +37 -0
  386. claude_mpm/skills/bundled/pm/mpm-teaching-mode/SKILL.md +657 -0
  387. claude_mpm/skills/bundled/pm/mpm-ticket-view/SKILL.md +110 -0
  388. claude_mpm/skills/bundled/pm/mpm-ticketing-integration/SKILL.md +154 -0
  389. claude_mpm/skills/bundled/pm/mpm-tool-usage-guide/SKILL.md +386 -0
  390. claude_mpm/skills/bundled/pm/mpm-verification-protocols/SKILL.md +198 -0
  391. claude_mpm/skills/bundled/pm/mpm-version/SKILL.md +21 -0
  392. claude_mpm/skills/bundled/react/flexlayout-react.md +742 -0
  393. claude_mpm/skills/bundled/rust/desktop-applications/SKILL.md +226 -0
  394. claude_mpm/skills/bundled/rust/desktop-applications/references/architecture-patterns.md +901 -0
  395. claude_mpm/skills/bundled/rust/desktop-applications/references/native-gui-frameworks.md +901 -0
  396. claude_mpm/skills/bundled/rust/desktop-applications/references/platform-integration.md +775 -0
  397. claude_mpm/skills/bundled/rust/desktop-applications/references/state-management.md +937 -0
  398. claude_mpm/skills/bundled/rust/desktop-applications/references/tauri-framework.md +770 -0
  399. claude_mpm/skills/bundled/rust/desktop-applications/references/testing-deployment.md +961 -0
  400. claude_mpm/skills/bundled/security-scanning.md +112 -0
  401. claude_mpm/skills/bundled/tauri/tauri-async-patterns.md +495 -0
  402. claude_mpm/skills/bundled/tauri/tauri-build-deploy.md +599 -0
  403. claude_mpm/skills/bundled/tauri/tauri-command-patterns.md +535 -0
  404. claude_mpm/skills/bundled/tauri/tauri-error-handling.md +613 -0
  405. claude_mpm/skills/bundled/tauri/tauri-event-system.md +648 -0
  406. claude_mpm/skills/bundled/tauri/tauri-file-system.md +673 -0
  407. claude_mpm/skills/bundled/tauri/tauri-frontend-integration.md +767 -0
  408. claude_mpm/skills/bundled/tauri/tauri-performance.md +669 -0
  409. claude_mpm/skills/bundled/tauri/tauri-state-management.md +573 -0
  410. claude_mpm/skills/bundled/tauri/tauri-testing.md +384 -0
  411. claude_mpm/skills/bundled/tauri/tauri-window-management.md +628 -0
  412. claude_mpm/skills/bundled/testing/condition-based-waiting/SKILL.md +119 -0
  413. claude_mpm/skills/bundled/testing/condition-based-waiting/references/patterns-and-implementation.md +253 -0
  414. claude_mpm/skills/bundled/testing/test-driven-development/SKILL.md +145 -0
  415. claude_mpm/skills/bundled/testing/test-driven-development/references/anti-patterns.md +543 -0
  416. claude_mpm/skills/bundled/testing/test-driven-development/references/examples.md +741 -0
  417. claude_mpm/skills/bundled/testing/test-driven-development/references/integration.md +470 -0
  418. claude_mpm/skills/bundled/testing/test-driven-development/references/philosophy.md +458 -0
  419. claude_mpm/skills/bundled/testing/test-driven-development/references/workflow.md +639 -0
  420. claude_mpm/skills/bundled/testing/test-quality-inspector/SKILL.md +458 -0
  421. claude_mpm/skills/bundled/testing/test-quality-inspector/examples/example-inspection-report.md +411 -0
  422. claude_mpm/skills/bundled/testing/test-quality-inspector/references/assertion-quality.md +317 -0
  423. claude_mpm/skills/bundled/testing/test-quality-inspector/references/inspection-checklist.md +270 -0
  424. claude_mpm/skills/bundled/testing/test-quality-inspector/references/red-flags.md +436 -0
  425. claude_mpm/skills/bundled/testing/testing-anti-patterns/SKILL.md +140 -0
  426. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/completeness-anti-patterns.md +572 -0
  427. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/core-anti-patterns.md +411 -0
  428. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/detection-guide.md +569 -0
  429. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/tdd-connection.md +695 -0
  430. claude_mpm/skills/bundled/testing/webapp-testing/SKILL.md +184 -0
  431. claude_mpm/skills/bundled/testing/webapp-testing/decision-tree.md +459 -0
  432. claude_mpm/skills/bundled/testing/webapp-testing/playwright-patterns.md +479 -0
  433. claude_mpm/skills/bundled/testing/webapp-testing/reconnaissance-pattern.md +687 -0
  434. claude_mpm/skills/bundled/testing/webapp-testing/server-management.md +758 -0
  435. claude_mpm/skills/bundled/testing/webapp-testing/troubleshooting.md +868 -0
  436. claude_mpm/skills/registry.py +295 -90
  437. claude_mpm/skills/skill_manager.py +29 -23
  438. claude_mpm/templates/.pre-commit-config.yaml +112 -0
  439. claude_mpm/utils/agent_dependency_loader.py +103 -4
  440. claude_mpm/utils/robust_installer.py +45 -24
  441. claude_mpm-5.6.23.dist-info/METADATA +393 -0
  442. {claude_mpm-5.4.41.dist-info → claude_mpm-5.6.23.dist-info}/RECORD +447 -149
  443. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/0.B_FtCwCQ.css +0 -1
  444. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/2.Cl_eSA4x.css +0 -1
  445. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BgChzWQ1.js +0 -1
  446. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CIXEwuWe.js +0 -1
  447. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CWc5urbQ.js +0 -1
  448. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DMkZpdF2.js +0 -2
  449. claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/app.DTL5mJO-.js +0 -2
  450. claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/start.DzuEhzqh.js +0 -1
  451. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/1.DFLC8jdE.js +0 -1
  452. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/2.DPvEihJJ.js +0 -10
  453. claude_mpm/hooks/claude_hooks/__pycache__/installer.cpython-311.pyc +0 -0
  454. claude_mpm/hooks/claude_hooks/services/__pycache__/connection_manager.cpython-311.pyc +0 -0
  455. claude_mpm-5.4.41.dist-info/METADATA +0 -998
  456. {claude_mpm-5.4.41.dist-info → claude_mpm-5.6.23.dist-info}/WHEEL +0 -0
  457. {claude_mpm-5.4.41.dist-info → claude_mpm-5.6.23.dist-info}/entry_points.txt +0 -0
  458. {claude_mpm-5.4.41.dist-info → claude_mpm-5.6.23.dist-info}/licenses/LICENSE +0 -0
  459. {claude_mpm-5.4.41.dist-info → claude_mpm-5.6.23.dist-info}/licenses/LICENSE-FAQ.md +0 -0
  460. {claude_mpm-5.4.41.dist-info → claude_mpm-5.6.23.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,775 @@
1
+ # Platform Integration
2
+
3
+ Comprehensive guide to integrating with native platform features across Windows, macOS, and Linux in Rust desktop applications.
4
+
5
+ ## File System Access
6
+
7
+ ### File Dialogs
8
+
9
+ ```rust
10
+ use tauri::api::dialog::{FileDialogBuilder, MessageDialogBuilder, MessageDialogKind};
11
+
12
+ #[tauri::command]
13
+ async fn open_file_dialog() -> Result<Option<String>, String> {
14
+ let path = FileDialogBuilder::new()
15
+ .add_filter("Text Files", &["txt", "md"])
16
+ .add_filter("All Files", &["*"])
17
+ .set_title("Select a file")
18
+ .pick_file();
19
+
20
+ Ok(path.map(|p| p.to_string_lossy().to_string()))
21
+ }
22
+
23
+ #[tauri::command]
24
+ async fn open_folder_dialog() -> Result<Option<String>, String> {
25
+ let path = FileDialogBuilder::new()
26
+ .set_title("Select a folder")
27
+ .pick_folder();
28
+
29
+ Ok(path.map(|p| p.to_string_lossy().to_string()))
30
+ }
31
+
32
+ #[tauri::command]
33
+ async fn save_file_dialog() -> Result<Option<String>, String> {
34
+ let path = FileDialogBuilder::new()
35
+ .add_filter("JSON Files", &["json"])
36
+ .set_file_name("untitled.json")
37
+ .save_file();
38
+
39
+ Ok(path.map(|p| p.to_string_lossy().to_string()))
40
+ }
41
+
42
+ #[tauri::command]
43
+ async fn show_message(title: String, message: String) -> Result<(), String> {
44
+ MessageDialogBuilder::new(title, message)
45
+ .kind(MessageDialogKind::Info)
46
+ .show();
47
+
48
+ Ok(())
49
+ }
50
+
51
+ #[tauri::command]
52
+ async fn confirm_dialog(title: String, message: String) -> Result<bool, String> {
53
+ let confirmed = MessageDialogBuilder::new(title, message)
54
+ .kind(MessageDialogKind::Warning)
55
+ .buttons(tauri::api::dialog::MessageDialogButtons::OkCancel)
56
+ .show();
57
+
58
+ Ok(confirmed)
59
+ }
60
+ ```
61
+
62
+ ### Safe File System Operations
63
+
64
+ ```rust
65
+ use std::path::{Path, PathBuf};
66
+ use std::fs;
67
+
68
+ // Validate file paths to prevent directory traversal
69
+ fn validate_path(path: &str, base_dir: &Path) -> Result<PathBuf, String> {
70
+ let path = Path::new(path);
71
+
72
+ // Canonicalize to resolve .. and symlinks
73
+ let canonical = path
74
+ .canonicalize()
75
+ .map_err(|_| "Invalid path".to_string())?;
76
+
77
+ // Ensure path is within base directory
78
+ if !canonical.starts_with(base_dir) {
79
+ return Err("Path outside allowed directory".to_string());
80
+ }
81
+
82
+ Ok(canonical)
83
+ }
84
+
85
+ #[tauri::command]
86
+ async fn read_file_safe(app: tauri::AppHandle, relative_path: String) -> Result<String, String> {
87
+ let app_dir = app
88
+ .path()
89
+ .app_data_dir()
90
+ .map_err(|e| e.to_string())?;
91
+
92
+ let file_path = validate_path(&relative_path, &app_dir)?;
93
+
94
+ fs::read_to_string(file_path).map_err(|e| e.to_string())
95
+ }
96
+
97
+ #[tauri::command]
98
+ async fn write_file_safe(
99
+ app: tauri::AppHandle,
100
+ relative_path: String,
101
+ content: String,
102
+ ) -> Result<(), String> {
103
+ let app_dir = app
104
+ .path()
105
+ .app_data_dir()
106
+ .map_err(|e| e.to_string())?;
107
+
108
+ // Ensure directory exists
109
+ fs::create_dir_all(&app_dir).map_err(|e| e.to_string())?;
110
+
111
+ let file_path = app_dir.join(&relative_path);
112
+
113
+ // Security check
114
+ let canonical = file_path
115
+ .canonicalize()
116
+ .or_else(|_| {
117
+ // File doesn't exist yet, validate parent
118
+ file_path
119
+ .parent()
120
+ .ok_or("Invalid path")?
121
+ .canonicalize()
122
+ .map(|p| p.join(file_path.file_name().unwrap()))
123
+ })
124
+ .map_err(|_| "Invalid path".to_string())?;
125
+
126
+ if !canonical.starts_with(&app_dir) {
127
+ return Err("Path outside allowed directory".to_string());
128
+ }
129
+
130
+ fs::write(canonical, content).map_err(|e| e.to_string())
131
+ }
132
+ ```
133
+
134
+ ### File Watching
135
+
136
+ ```rust
137
+ use notify::{Watcher, RecursiveMode, Event};
138
+ use std::sync::mpsc::channel;
139
+ use std::time::Duration;
140
+
141
+ struct FileWatcher {
142
+ watcher: notify::RecommendedWatcher,
143
+ }
144
+
145
+ impl FileWatcher {
146
+ fn new(app_handle: tauri::AppHandle) -> Result<Self, String> {
147
+ let (tx, rx) = channel();
148
+
149
+ let mut watcher = notify::recommended_watcher(tx)
150
+ .map_err(|e| e.to_string())?;
151
+
152
+ // Spawn task to handle events
153
+ tokio::spawn(async move {
154
+ while let Ok(event) = rx.recv() {
155
+ if let Ok(Event { kind, paths, .. }) = event {
156
+ let _ = app_handle.emit("file-changed", FileChangeEvent {
157
+ kind: format!("{:?}", kind),
158
+ paths: paths.iter().map(|p| p.to_string_lossy().to_string()).collect(),
159
+ });
160
+ }
161
+ }
162
+ });
163
+
164
+ Ok(Self { watcher })
165
+ }
166
+
167
+ fn watch(&mut self, path: &str) -> Result<(), String> {
168
+ self.watcher
169
+ .watch(Path::new(path), RecursiveMode::Recursive)
170
+ .map_err(|e| e.to_string())
171
+ }
172
+
173
+ fn unwatch(&mut self, path: &str) -> Result<(), String> {
174
+ self.watcher
175
+ .unwatch(Path::new(path))
176
+ .map_err(|e| e.to_string())
177
+ }
178
+ }
179
+
180
+ #[derive(Clone, serde::Serialize)]
181
+ struct FileChangeEvent {
182
+ kind: String,
183
+ paths: Vec<String>,
184
+ }
185
+
186
+ #[tauri::command]
187
+ fn watch_directory(
188
+ watcher: tauri::State<FileWatcher>,
189
+ path: String,
190
+ ) -> Result<(), String> {
191
+ watcher.inner().lock().unwrap().watch(&path)
192
+ }
193
+ ```
194
+
195
+ ## System Tray Integration
196
+
197
+ ### Cross-Platform System Tray
198
+
199
+ ```rust
200
+ use tauri::{
201
+ menu::{Menu, MenuItem, Submenu},
202
+ tray::{TrayIconBuilder, TrayIconEvent},
203
+ Manager, Runtime,
204
+ };
205
+
206
+ fn create_tray<R: Runtime>(app: &tauri::AppHandle<R>) -> Result<(), Box<dyn std::error::Error>> {
207
+ // Create menu items
208
+ let show_item = MenuItem::with_id(app, "show", "Show Window", true, None::<&str>)?;
209
+ let hide_item = MenuItem::with_id(app, "hide", "Hide Window", true, None::<&str>)?;
210
+ let quit_item = MenuItem::with_id(app, "quit", "Quit", true, None::<&str>)?;
211
+
212
+ // Create submenu
213
+ let settings_menu = Submenu::with_items(
214
+ app,
215
+ "Settings",
216
+ true,
217
+ &[
218
+ &MenuItem::with_id(app, "preferences", "Preferences", true, None::<&str>)?,
219
+ &MenuItem::with_id(app, "about", "About", true, None::<&str>)?,
220
+ ],
221
+ )?;
222
+
223
+ // Create menu
224
+ let menu = Menu::with_items(app, &[&show_item, &hide_item, &settings_menu, &quit_item])?;
225
+
226
+ // Build tray icon
227
+ let _tray = TrayIconBuilder::new()
228
+ .icon(app.default_window_icon().unwrap().clone())
229
+ .menu(&menu)
230
+ .tooltip("My Application")
231
+ .on_menu_event(|app, event| match event.id.as_ref() {
232
+ "show" => {
233
+ if let Some(window) = app.get_webview_window("main") {
234
+ let _ = window.show();
235
+ let _ = window.set_focus();
236
+ }
237
+ }
238
+ "hide" => {
239
+ if let Some(window) = app.get_webview_window("main") {
240
+ let _ = window.hide();
241
+ }
242
+ }
243
+ "quit" => {
244
+ app.exit(0);
245
+ }
246
+ "preferences" => {
247
+ // Open preferences window
248
+ println!("Open preferences");
249
+ }
250
+ "about" => {
251
+ // Show about dialog
252
+ println!("Show about dialog");
253
+ }
254
+ _ => {}
255
+ })
256
+ .on_tray_icon_event(|tray, event| {
257
+ if let TrayIconEvent::Click { .. } = event {
258
+ // Handle tray icon click
259
+ let app = tray.app_handle();
260
+ if let Some(window) = app.get_webview_window("main") {
261
+ if window.is_visible().unwrap_or(false) {
262
+ let _ = window.hide();
263
+ } else {
264
+ let _ = window.show();
265
+ let _ = window.set_focus();
266
+ }
267
+ }
268
+ }
269
+ })
270
+ .build(app)?;
271
+
272
+ Ok(())
273
+ }
274
+
275
+ fn main() {
276
+ tauri::Builder::default()
277
+ .setup(|app| {
278
+ create_tray(app.handle())?;
279
+ Ok(())
280
+ })
281
+ .run(tauri::generate_context!())
282
+ .expect("error while running tauri application");
283
+ }
284
+ ```
285
+
286
+ ### Dynamic Tray Menu Updates
287
+
288
+ ```rust
289
+ use std::sync::Mutex;
290
+
291
+ struct TrayState {
292
+ is_recording: Mutex<bool>,
293
+ }
294
+
295
+ #[tauri::command]
296
+ fn toggle_recording(
297
+ app: tauri::AppHandle,
298
+ state: tauri::State<TrayState>,
299
+ ) -> Result<(), String> {
300
+ let mut is_recording = state.is_recording.lock().unwrap();
301
+ *is_recording = !*is_recording;
302
+
303
+ // Update tray menu
304
+ let tray = app.tray_by_id("main").ok_or("Tray not found")?;
305
+
306
+ let menu_item = tray
307
+ .get_item("toggle_recording")
308
+ .ok_or("Menu item not found")?;
309
+
310
+ menu_item
311
+ .set_text(if *is_recording {
312
+ "Stop Recording"
313
+ } else {
314
+ "Start Recording"
315
+ })
316
+ .map_err(|e| e.to_string())?;
317
+
318
+ Ok(())
319
+ }
320
+ ```
321
+
322
+ ## Native Notifications
323
+
324
+ ### Cross-Platform Notifications
325
+
326
+ ```rust
327
+ use tauri::Notification;
328
+
329
+ #[tauri::command]
330
+ fn send_notification(
331
+ app: tauri::AppHandle,
332
+ title: String,
333
+ body: String,
334
+ ) -> Result<(), String> {
335
+ Notification::new(&app.config().identifier)
336
+ .title(title)
337
+ .body(body)
338
+ .icon("icon")
339
+ .show()
340
+ .map_err(|e| e.to_string())
341
+ }
342
+
343
+ #[tauri::command]
344
+ fn send_notification_with_action(
345
+ app: tauri::AppHandle,
346
+ title: String,
347
+ body: String,
348
+ ) -> Result<(), String> {
349
+ // Note: Actions are platform-dependent
350
+ #[cfg(target_os = "macos")]
351
+ {
352
+ Notification::new(&app.config().identifier)
353
+ .title(title)
354
+ .body(body)
355
+ .sound("default")
356
+ .show()
357
+ .map_err(|e| e.to_string())
358
+ }
359
+
360
+ #[cfg(not(target_os = "macos"))]
361
+ {
362
+ Notification::new(&app.config().identifier)
363
+ .title(title)
364
+ .body(body)
365
+ .show()
366
+ .map_err(|e| e.to_string())
367
+ }
368
+ }
369
+ ```
370
+
371
+ ### Notification with User Interaction
372
+
373
+ ```rust
374
+ use tauri::{Emitter, Manager};
375
+
376
+ #[tauri::command]
377
+ async fn send_interactive_notification(
378
+ app: tauri::AppHandle,
379
+ title: String,
380
+ body: String,
381
+ ) -> Result<(), String> {
382
+ // Send notification
383
+ Notification::new(&app.config().identifier)
384
+ .title(&title)
385
+ .body(&body)
386
+ .show()
387
+ .map_err(|e| e.to_string())?;
388
+
389
+ // Listen for notification clicks (platform-dependent)
390
+ // This is a simplified example; real implementation needs platform-specific code
391
+
392
+ Ok(())
393
+ }
394
+ ```
395
+
396
+ ## Auto-Updates
397
+
398
+ ### Tauri Updater Integration
399
+
400
+ ```rust
401
+ use tauri_plugin_updater::UpdaterExt;
402
+
403
+ #[tauri::command]
404
+ async fn check_for_updates(app: tauri::AppHandle) -> Result<Option<String>, String> {
405
+ let update = app
406
+ .updater()
407
+ .check()
408
+ .await
409
+ .map_err(|e| e.to_string())?;
410
+
411
+ if let Some(update) = update {
412
+ Ok(Some(format!(
413
+ "Update available: {} (current: {})",
414
+ update.version,
415
+ update.current_version
416
+ )))
417
+ } else {
418
+ Ok(None)
419
+ }
420
+ }
421
+
422
+ #[tauri::command]
423
+ async fn install_update(app: tauri::AppHandle) -> Result<(), String> {
424
+ let update = app
425
+ .updater()
426
+ .check()
427
+ .await
428
+ .map_err(|e| e.to_string())?;
429
+
430
+ if let Some(update) = update {
431
+ // Download and install
432
+ update
433
+ .download_and_install(
434
+ |chunk_length, content_length| {
435
+ println!(
436
+ "Downloaded {} of {:?}",
437
+ chunk_length,
438
+ content_length
439
+ );
440
+ },
441
+ || {
442
+ println!("Download finished");
443
+ },
444
+ )
445
+ .await
446
+ .map_err(|e| e.to_string())?;
447
+
448
+ // Restart app
449
+ app.restart();
450
+ }
451
+
452
+ Ok(())
453
+ }
454
+
455
+ // Setup auto-update check
456
+ fn main() {
457
+ tauri::Builder::default()
458
+ .plugin(tauri_plugin_updater::Builder::new().build())
459
+ .setup(|app| {
460
+ let handle = app.handle().clone();
461
+
462
+ // Check for updates on startup
463
+ tauri::async_runtime::spawn(async move {
464
+ tokio::time::sleep(tokio::time::Duration::from_secs(5)).await;
465
+
466
+ if let Ok(Some(update)) = handle.updater().check().await {
467
+ println!("Update available: {}", update.version);
468
+
469
+ // Emit event to frontend
470
+ let _ = handle.emit("update-available", &update.version);
471
+ }
472
+ });
473
+
474
+ Ok(())
475
+ })
476
+ .run(tauri::generate_context!())
477
+ .expect("error while running tauri application");
478
+ }
479
+ ```
480
+
481
+ ## Deep Linking / Custom URL Schemes
482
+
483
+ ### Register URL Scheme
484
+
485
+ **tauri.conf.json:**
486
+ ```json
487
+ {
488
+ "bundle": {
489
+ "macOS": {
490
+ "associatedDomains": ["myapp://"],
491
+ "category": "public.app-category.developer-tools"
492
+ },
493
+ "windows": {
494
+ "webviewInstallMode": {
495
+ "type": "downloadBootstrapper"
496
+ },
497
+ "protocols": [
498
+ {
499
+ "name": "myapp",
500
+ "schemes": ["myapp"]
501
+ }
502
+ ]
503
+ }
504
+ }
505
+ }
506
+ ```
507
+
508
+ ### Handle Deep Links
509
+
510
+ ```rust
511
+ use tauri::{Emitter, Manager};
512
+
513
+ fn main() {
514
+ tauri::Builder::default()
515
+ .setup(|app| {
516
+ // Register URL handler
517
+ app.listen_any("deep-link://", |event| {
518
+ println!("Received deep link: {:?}", event.payload());
519
+ });
520
+
521
+ Ok(())
522
+ })
523
+ .plugin(tauri_plugin_deep_link::init())
524
+ .run(tauri::generate_context!())
525
+ .expect("error while running tauri application");
526
+ }
527
+
528
+ #[tauri::command]
529
+ fn handle_url(app: tauri::AppHandle, url: String) -> Result<(), String> {
530
+ println!("Handling URL: {}", url);
531
+
532
+ // Parse URL and navigate
533
+ if url.starts_with("myapp://open/") {
534
+ let file = url.strip_prefix("myapp://open/").unwrap();
535
+ app.emit("open-file", file).map_err(|e| e.to_string())?;
536
+ }
537
+
538
+ Ok(())
539
+ }
540
+ ```
541
+
542
+ ## Platform-Specific Features
543
+
544
+ ### Windows
545
+
546
+ ```rust
547
+ #[cfg(target_os = "windows")]
548
+ mod windows {
549
+ use winapi::um::winuser::{MessageBoxW, MB_OK};
550
+ use std::ffi::OsStr;
551
+ use std::os::windows::ffi::OsStrExt;
552
+
553
+ pub fn show_native_message_box(title: &str, message: &str) {
554
+ let title_wide: Vec<u16> = OsStr::new(title)
555
+ .encode_wide()
556
+ .chain(std::iter::once(0))
557
+ .collect();
558
+
559
+ let message_wide: Vec<u16> = OsStr::new(message)
560
+ .encode_wide()
561
+ .chain(std::iter::once(0))
562
+ .collect();
563
+
564
+ unsafe {
565
+ MessageBoxW(
566
+ std::ptr::null_mut(),
567
+ message_wide.as_ptr(),
568
+ title_wide.as_ptr(),
569
+ MB_OK,
570
+ );
571
+ }
572
+ }
573
+
574
+ // Windows Registry access
575
+ use winreg::enums::*;
576
+ use winreg::RegKey;
577
+
578
+ pub fn read_registry_value(key_path: &str, value_name: &str) -> Option<String> {
579
+ let hklm = RegKey::predef(HKEY_CURRENT_USER);
580
+ let key = hklm.open_subkey(key_path).ok()?;
581
+ key.get_value(value_name).ok()
582
+ }
583
+
584
+ pub fn write_registry_value(
585
+ key_path: &str,
586
+ value_name: &str,
587
+ value: &str,
588
+ ) -> Result<(), std::io::Error> {
589
+ let hklm = RegKey::predef(HKEY_CURRENT_USER);
590
+ let (key, _) = hklm.create_subkey(key_path)?;
591
+ key.set_value(value_name, &value)?;
592
+ Ok(())
593
+ }
594
+ }
595
+
596
+ #[tauri::command]
597
+ #[cfg(target_os = "windows")]
598
+ fn windows_specific_feature() -> Result<String, String> {
599
+ windows::show_native_message_box("Title", "Message");
600
+
601
+ let value = windows::read_registry_value(
602
+ "Software\\MyApp",
603
+ "Setting1",
604
+ )
605
+ .unwrap_or_default();
606
+
607
+ Ok(value)
608
+ }
609
+ ```
610
+
611
+ ### macOS
612
+
613
+ ```rust
614
+ #[cfg(target_os = "macos")]
615
+ mod macos {
616
+ use cocoa::base::nil;
617
+ use cocoa::foundation::NSString;
618
+ use objc::{class, msg_send, sel, sel_impl};
619
+
620
+ pub fn set_dock_badge(label: &str) {
621
+ unsafe {
622
+ let app = cocoa::appkit::NSApp();
623
+ let dock_tile: cocoa::base::id = msg_send![app, dockTile];
624
+
625
+ let badge_label = NSString::alloc(nil).init_str(label);
626
+ let _: () = msg_send![dock_tile, setBadgeLabel: badge_label];
627
+ }
628
+ }
629
+
630
+ pub fn clear_dock_badge() {
631
+ unsafe {
632
+ let app = cocoa::appkit::NSApp();
633
+ let dock_tile: cocoa::base::id = msg_send![app, dockTile];
634
+ let _: () = msg_send![dock_tile, setBadgeLabel: nil];
635
+ }
636
+ }
637
+
638
+ // Access macOS services
639
+ use std::process::Command;
640
+
641
+ pub fn trigger_notification_center(title: &str, message: &str) {
642
+ let script = format!(
643
+ r#"display notification "{}" with title "{}""#,
644
+ message, title
645
+ );
646
+
647
+ Command::new("osascript")
648
+ .arg("-e")
649
+ .arg(script)
650
+ .output()
651
+ .ok();
652
+ }
653
+ }
654
+
655
+ #[tauri::command]
656
+ #[cfg(target_os = "macos")]
657
+ fn macos_specific_feature(badge: String) -> Result<(), String> {
658
+ macos::set_dock_badge(&badge);
659
+ Ok(())
660
+ }
661
+
662
+ #[tauri::command]
663
+ #[cfg(target_os = "macos")]
664
+ fn clear_badge() -> Result<(), String> {
665
+ macos::clear_dock_badge();
666
+ Ok(())
667
+ }
668
+ ```
669
+
670
+ ### Linux
671
+
672
+ ```rust
673
+ #[cfg(target_os = "linux")]
674
+ mod linux {
675
+ use std::process::Command;
676
+
677
+ pub fn send_desktop_notification(title: &str, message: &str) -> Result<(), String> {
678
+ Command::new("notify-send")
679
+ .arg(title)
680
+ .arg(message)
681
+ .output()
682
+ .map_err(|e| e.to_string())?;
683
+
684
+ Ok(())
685
+ }
686
+
687
+ // D-Bus integration
688
+ use dbus::blocking::Connection;
689
+ use std::time::Duration;
690
+
691
+ pub fn get_desktop_environment() -> Result<String, Box<dyn std::error::Error>> {
692
+ let conn = Connection::new_session()?;
693
+ let proxy = conn.with_proxy(
694
+ "org.freedesktop.portal.Desktop",
695
+ "/org/freedesktop/portal/desktop",
696
+ Duration::from_millis(5000),
697
+ );
698
+
699
+ // Query desktop environment
700
+ // This is a simplified example
701
+ Ok("Unknown".to_string())
702
+ }
703
+ }
704
+
705
+ #[tauri::command]
706
+ #[cfg(target_os = "linux")]
707
+ fn linux_specific_feature(title: String, message: String) -> Result<(), String> {
708
+ linux::send_desktop_notification(&title, &message)
709
+ }
710
+ ```
711
+
712
+ ## Permissions and Security
713
+
714
+ ### Scope Configuration
715
+
716
+ ```rust
717
+ use tauri::Manager;
718
+
719
+ fn main() {
720
+ tauri::Builder::default()
721
+ .setup(|app| {
722
+ // Configure file system scope
723
+ let scope = app.fs_scope();
724
+
725
+ // Allow access to specific directories
726
+ let app_data_dir = app.path().app_data_dir()?;
727
+ scope.allow_directory(&app_data_dir, true)?;
728
+
729
+ let documents_dir = app.path().document_dir()?;
730
+ scope.allow_directory(&documents_dir, false)?;
731
+
732
+ Ok(())
733
+ })
734
+ .run(tauri::generate_context!())
735
+ .expect("error while running tauri application");
736
+ }
737
+ ```
738
+
739
+ ### Runtime Permission Checks
740
+
741
+ ```rust
742
+ use tauri::Manager;
743
+
744
+ #[tauri::command]
745
+ fn read_file_with_permission(
746
+ app: tauri::AppHandle,
747
+ path: String,
748
+ ) -> Result<String, String> {
749
+ let scope = app.fs_scope();
750
+
751
+ // Check if path is allowed
752
+ if !scope.is_allowed(&path) {
753
+ return Err("Access denied: path not in scope".to_string());
754
+ }
755
+
756
+ std::fs::read_to_string(&path).map_err(|e| e.to_string())
757
+ }
758
+
759
+ #[tauri::command]
760
+ fn request_file_access(
761
+ app: tauri::AppHandle,
762
+ path: String,
763
+ ) -> Result<(), String> {
764
+ let scope = app.fs_scope();
765
+
766
+ // Request access (user must approve via dialog)
767
+ scope
768
+ .allow_file(&path)
769
+ .map_err(|e| e.to_string())?;
770
+
771
+ Ok(())
772
+ }
773
+ ```
774
+
775
+ These platform integration patterns enable full access to native OS features while maintaining cross-platform compatibility and security.