claude-mpm 5.4.41__py3-none-any.whl → 5.6.72__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 (490) 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/auth/__init__.py +35 -0
  9. claude_mpm/auth/callback_server.py +328 -0
  10. claude_mpm/auth/models.py +104 -0
  11. claude_mpm/auth/oauth_manager.py +266 -0
  12. claude_mpm/auth/providers/__init__.py +12 -0
  13. claude_mpm/auth/providers/base.py +165 -0
  14. claude_mpm/auth/providers/google.py +261 -0
  15. claude_mpm/auth/token_storage.py +252 -0
  16. claude_mpm/cli/__init__.py +5 -1
  17. claude_mpm/cli/commands/agents.py +2 -4
  18. claude_mpm/cli/commands/agents_reconcile.py +197 -0
  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 +620 -21
  22. claude_mpm/cli/commands/configure_agent_display.py +3 -1
  23. claude_mpm/cli/commands/hook_errors.py +60 -60
  24. claude_mpm/cli/commands/mcp.py +29 -17
  25. claude_mpm/cli/commands/mcp_command_router.py +39 -0
  26. claude_mpm/cli/commands/mcp_service_commands.py +304 -0
  27. claude_mpm/cli/commands/monitor.py +2 -2
  28. claude_mpm/cli/commands/mpm_init/core.py +15 -8
  29. claude_mpm/cli/commands/oauth.py +481 -0
  30. claude_mpm/cli/commands/profile.py +9 -10
  31. claude_mpm/cli/commands/run.py +35 -3
  32. claude_mpm/cli/commands/skill_source.py +51 -2
  33. claude_mpm/cli/commands/skills.py +182 -32
  34. claude_mpm/cli/executor.py +129 -16
  35. claude_mpm/cli/helpers.py +1 -1
  36. claude_mpm/cli/interactive/__init__.py +10 -0
  37. claude_mpm/cli/interactive/agent_wizard.py +30 -50
  38. claude_mpm/cli/interactive/questionary_styles.py +65 -0
  39. claude_mpm/cli/interactive/skill_selector.py +481 -0
  40. claude_mpm/cli/parsers/base_parser.py +89 -1
  41. claude_mpm/cli/parsers/commander_parser.py +116 -0
  42. claude_mpm/cli/parsers/mcp_parser.py +79 -0
  43. claude_mpm/cli/parsers/oauth_parser.py +165 -0
  44. claude_mpm/cli/parsers/profile_parser.py +0 -1
  45. claude_mpm/cli/parsers/run_parser.py +10 -0
  46. claude_mpm/cli/parsers/skill_source_parser.py +4 -0
  47. claude_mpm/cli/parsers/skills_parser.py +2 -3
  48. claude_mpm/cli/startup.py +662 -524
  49. claude_mpm/cli/startup_display.py +76 -7
  50. claude_mpm/cli/startup_logging.py +2 -2
  51. claude_mpm/cli/utils.py +7 -3
  52. claude_mpm/commander/__init__.py +78 -0
  53. claude_mpm/commander/adapters/__init__.py +60 -0
  54. claude_mpm/commander/adapters/auggie.py +260 -0
  55. claude_mpm/commander/adapters/base.py +288 -0
  56. claude_mpm/commander/adapters/claude_code.py +392 -0
  57. claude_mpm/commander/adapters/codex.py +237 -0
  58. claude_mpm/commander/adapters/communication.py +366 -0
  59. claude_mpm/commander/adapters/example_usage.py +310 -0
  60. claude_mpm/commander/adapters/mpm.py +389 -0
  61. claude_mpm/commander/adapters/registry.py +204 -0
  62. claude_mpm/commander/api/__init__.py +16 -0
  63. claude_mpm/commander/api/app.py +121 -0
  64. claude_mpm/commander/api/errors.py +133 -0
  65. claude_mpm/commander/api/routes/__init__.py +8 -0
  66. claude_mpm/commander/api/routes/events.py +184 -0
  67. claude_mpm/commander/api/routes/inbox.py +171 -0
  68. claude_mpm/commander/api/routes/messages.py +148 -0
  69. claude_mpm/commander/api/routes/projects.py +271 -0
  70. claude_mpm/commander/api/routes/sessions.py +226 -0
  71. claude_mpm/commander/api/routes/work.py +296 -0
  72. claude_mpm/commander/api/schemas.py +186 -0
  73. claude_mpm/commander/chat/__init__.py +7 -0
  74. claude_mpm/commander/chat/cli.py +149 -0
  75. claude_mpm/commander/chat/commands.py +122 -0
  76. claude_mpm/commander/chat/repl.py +1821 -0
  77. claude_mpm/commander/config.py +51 -0
  78. claude_mpm/commander/config_loader.py +115 -0
  79. claude_mpm/commander/core/__init__.py +10 -0
  80. claude_mpm/commander/core/block_manager.py +325 -0
  81. claude_mpm/commander/core/response_manager.py +323 -0
  82. claude_mpm/commander/daemon.py +603 -0
  83. claude_mpm/commander/env_loader.py +59 -0
  84. claude_mpm/commander/events/__init__.py +26 -0
  85. claude_mpm/commander/events/manager.py +392 -0
  86. claude_mpm/commander/frameworks/__init__.py +12 -0
  87. claude_mpm/commander/frameworks/base.py +233 -0
  88. claude_mpm/commander/frameworks/claude_code.py +58 -0
  89. claude_mpm/commander/frameworks/mpm.py +57 -0
  90. claude_mpm/commander/git/__init__.py +5 -0
  91. claude_mpm/commander/git/worktree_manager.py +212 -0
  92. claude_mpm/commander/inbox/__init__.py +16 -0
  93. claude_mpm/commander/inbox/dedup.py +128 -0
  94. claude_mpm/commander/inbox/inbox.py +224 -0
  95. claude_mpm/commander/inbox/models.py +70 -0
  96. claude_mpm/commander/instance_manager.py +865 -0
  97. claude_mpm/commander/llm/__init__.py +6 -0
  98. claude_mpm/commander/llm/openrouter_client.py +167 -0
  99. claude_mpm/commander/llm/summarizer.py +70 -0
  100. claude_mpm/commander/memory/__init__.py +45 -0
  101. claude_mpm/commander/memory/compression.py +347 -0
  102. claude_mpm/commander/memory/embeddings.py +230 -0
  103. claude_mpm/commander/memory/entities.py +310 -0
  104. claude_mpm/commander/memory/example_usage.py +290 -0
  105. claude_mpm/commander/memory/integration.py +325 -0
  106. claude_mpm/commander/memory/search.py +381 -0
  107. claude_mpm/commander/memory/store.py +657 -0
  108. claude_mpm/commander/models/__init__.py +18 -0
  109. claude_mpm/commander/models/events.py +127 -0
  110. claude_mpm/commander/models/project.py +162 -0
  111. claude_mpm/commander/models/work.py +214 -0
  112. claude_mpm/commander/parsing/__init__.py +20 -0
  113. claude_mpm/commander/parsing/extractor.py +132 -0
  114. claude_mpm/commander/parsing/output_parser.py +270 -0
  115. claude_mpm/commander/parsing/patterns.py +100 -0
  116. claude_mpm/commander/persistence/__init__.py +11 -0
  117. claude_mpm/commander/persistence/event_store.py +274 -0
  118. claude_mpm/commander/persistence/state_store.py +403 -0
  119. claude_mpm/commander/persistence/work_store.py +164 -0
  120. claude_mpm/commander/polling/__init__.py +13 -0
  121. claude_mpm/commander/polling/event_detector.py +104 -0
  122. claude_mpm/commander/polling/output_buffer.py +49 -0
  123. claude_mpm/commander/polling/output_poller.py +153 -0
  124. claude_mpm/commander/project_session.py +268 -0
  125. claude_mpm/commander/proxy/__init__.py +12 -0
  126. claude_mpm/commander/proxy/formatter.py +89 -0
  127. claude_mpm/commander/proxy/output_handler.py +191 -0
  128. claude_mpm/commander/proxy/relay.py +155 -0
  129. claude_mpm/commander/registry.py +410 -0
  130. claude_mpm/commander/runtime/__init__.py +10 -0
  131. claude_mpm/commander/runtime/executor.py +191 -0
  132. claude_mpm/commander/runtime/monitor.py +346 -0
  133. claude_mpm/commander/session/__init__.py +6 -0
  134. claude_mpm/commander/session/context.py +81 -0
  135. claude_mpm/commander/session/manager.py +59 -0
  136. claude_mpm/commander/tmux_orchestrator.py +362 -0
  137. claude_mpm/commander/web/__init__.py +1 -0
  138. claude_mpm/commander/work/__init__.py +30 -0
  139. claude_mpm/commander/work/executor.py +207 -0
  140. claude_mpm/commander/work/queue.py +405 -0
  141. claude_mpm/commander/workflow/__init__.py +27 -0
  142. claude_mpm/commander/workflow/event_handler.py +241 -0
  143. claude_mpm/commander/workflow/notifier.py +146 -0
  144. claude_mpm/commands/mpm-config.md +8 -0
  145. claude_mpm/commands/mpm-doctor.md +8 -0
  146. claude_mpm/commands/mpm-help.md +8 -0
  147. claude_mpm/commands/mpm-init.md +8 -0
  148. claude_mpm/commands/mpm-monitor.md +8 -0
  149. claude_mpm/commands/mpm-organize.md +8 -0
  150. claude_mpm/commands/mpm-postmortem.md +8 -0
  151. claude_mpm/commands/mpm-session-resume.md +9 -1
  152. claude_mpm/commands/mpm-status.md +8 -0
  153. claude_mpm/commands/mpm-ticket-view.md +8 -0
  154. claude_mpm/commands/mpm-version.md +8 -0
  155. claude_mpm/commands/mpm.md +8 -0
  156. claude_mpm/config/agent_presets.py +8 -7
  157. claude_mpm/config/skill_sources.py +16 -0
  158. claude_mpm/constants.py +6 -0
  159. claude_mpm/core/claude_runner.py +154 -2
  160. claude_mpm/core/config.py +35 -22
  161. claude_mpm/core/config_constants.py +74 -9
  162. claude_mpm/core/constants.py +56 -12
  163. claude_mpm/core/hook_manager.py +53 -4
  164. claude_mpm/core/interactive_session.py +12 -11
  165. claude_mpm/core/logger.py +26 -9
  166. claude_mpm/core/logging_utils.py +39 -13
  167. claude_mpm/core/network_config.py +148 -0
  168. claude_mpm/core/oneshot_session.py +7 -6
  169. claude_mpm/core/optimized_startup.py +3 -1
  170. claude_mpm/core/output_style_manager.py +66 -18
  171. claude_mpm/core/shared/config_loader.py +3 -1
  172. claude_mpm/core/socketio_pool.py +47 -15
  173. claude_mpm/core/unified_config.py +54 -8
  174. claude_mpm/core/unified_paths.py +95 -90
  175. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/0.C33zOoyM.css +1 -0
  176. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/2.CW1J-YuA.css +1 -0
  177. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/1WZnGYqX.js +24 -0
  178. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/67pF3qNn.js +1 -0
  179. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/6RxdMKe4.js +1 -0
  180. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/8cZrfX0h.js +60 -0
  181. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/9a6T2nm-.js +7 -0
  182. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/B443AUzu.js +1 -0
  183. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/B8AwtY2H.js +1 -0
  184. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BF15LAsF.js +1 -0
  185. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BQaXIfA_.js +331 -0
  186. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BRcwIQNr.js +4 -0
  187. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{uj46x2Wr.js → BSNlmTZj.js} +1 -1
  188. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BV6nKitt.js +43 -0
  189. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BViJ8lZt.js +128 -0
  190. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BcQ-Q0FE.js +1 -0
  191. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Bpyvgze_.js +30 -0
  192. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BzTRqg-z.js +1 -0
  193. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C0Fr8dve.js +1 -0
  194. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C3rbW_a-.js +1 -0
  195. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C8WYN38h.js +1 -0
  196. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C9I8FlXH.js +61 -0
  197. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CIQcWgO2.js +36 -0
  198. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CIctN7YN.js +7 -0
  199. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CKrS_JZW.js +145 -0
  200. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CR6P9C4A.js +89 -0
  201. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CRRR9MD_.js +2 -0
  202. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CRcR2DqT.js +334 -0
  203. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CSXtMOf0.js +1 -0
  204. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CT-sbxSk.js +1 -0
  205. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CWm6DJsp.js +1 -0
  206. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CmKTTxBW.js +1 -0
  207. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CpqQ1Kzn.js +1 -0
  208. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Cu_Erd72.js +261 -0
  209. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/D2nGpDRe.js +1 -0
  210. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/D9iCMida.js +267 -0
  211. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/D9ykgMoY.js +10 -0
  212. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DL2Ldur1.js +1 -0
  213. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DPfltzjH.js +165 -0
  214. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{N4qtv3Hx.js → DR8nis88.js} +2 -2
  215. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DUliQN2b.js +1 -0
  216. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DVp1hx9R.js +1 -0
  217. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DXlhR01x.js +122 -0
  218. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/D_lyTybS.js +1 -0
  219. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DngoTTgh.js +1 -0
  220. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DqkmHtDC.js +220 -0
  221. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DsDh8EYs.js +1 -0
  222. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DypDmXgd.js +139 -0
  223. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Gi6I4Gst.js +1 -0
  224. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/IPYC-LnN.js +162 -0
  225. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/JTLiF7dt.js +24 -0
  226. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/JpevfAFt.js +68 -0
  227. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/{DjhvlsAc.js → NqQ1dWOy.js} +1 -1
  228. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/R8CEIRAd.js +2 -0
  229. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Zxy7qc-l.js +64 -0
  230. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/q9Hm6zAU.js +1 -0
  231. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/qtd3IeO4.js +15 -0
  232. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/ulBFON_C.js +65 -0
  233. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/wQVh1CoA.js +10 -0
  234. claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/app.Dr7t0z2J.js +2 -0
  235. claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/start.BGhZHUS3.js +1 -0
  236. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/{0.CAGBuiOw.js → 0.RgBboRvH.js} +1 -1
  237. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/1.DG-KkbDf.js +1 -0
  238. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/2.D_jnf-x6.js +1 -0
  239. claude_mpm/dashboard/static/svelte-build/_app/version.json +1 -1
  240. claude_mpm/dashboard/static/svelte-build/index.html +11 -11
  241. claude_mpm/dashboard-svelte/node_modules/katex/src/fonts/generate_fonts.py +58 -0
  242. claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/extract_tfms.py +114 -0
  243. claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/extract_ttfs.py +122 -0
  244. claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/format_json.py +28 -0
  245. claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/parse_tfm.py +211 -0
  246. claude_mpm/experimental/cli_enhancements.py +2 -1
  247. claude_mpm/hooks/claude_hooks/INTEGRATION_EXAMPLE.md +243 -0
  248. claude_mpm/hooks/claude_hooks/README_AUTO_PAUSE.md +403 -0
  249. claude_mpm/hooks/claude_hooks/__pycache__/auto_pause_handler.cpython-311.pyc +0 -0
  250. claude_mpm/hooks/claude_hooks/__pycache__/event_handlers.cpython-311.pyc +0 -0
  251. claude_mpm/hooks/claude_hooks/__pycache__/hook_handler.cpython-311.pyc +0 -0
  252. claude_mpm/hooks/claude_hooks/__pycache__/memory_integration.cpython-311.pyc +0 -0
  253. claude_mpm/hooks/claude_hooks/__pycache__/response_tracking.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 +466 -136
  256. claude_mpm/hooks/claude_hooks/hook_handler.py +204 -104
  257. claude_mpm/hooks/claude_hooks/hook_wrapper.sh +6 -11
  258. claude_mpm/hooks/claude_hooks/installer.py +291 -59
  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__/protocols.cpython-311.pyc +0 -0
  266. claude_mpm/hooks/claude_hooks/services/__pycache__/state_manager.cpython-311.pyc +0 -0
  267. claude_mpm/hooks/claude_hooks/services/__pycache__/subagent_processor.cpython-311.pyc +0 -0
  268. claude_mpm/hooks/claude_hooks/services/connection_manager.py +41 -26
  269. claude_mpm/hooks/claude_hooks/services/connection_manager_http.py +38 -105
  270. claude_mpm/hooks/claude_hooks/services/container.py +326 -0
  271. claude_mpm/hooks/claude_hooks/services/protocols.py +328 -0
  272. claude_mpm/hooks/claude_hooks/services/state_manager.py +25 -38
  273. claude_mpm/hooks/claude_hooks/services/subagent_processor.py +75 -77
  274. claude_mpm/hooks/kuzu_memory_hook.py +5 -5
  275. claude_mpm/hooks/session_resume_hook.py +89 -1
  276. claude_mpm/hooks/templates/pre_tool_use_simple.py +6 -6
  277. claude_mpm/hooks/templates/pre_tool_use_template.py +16 -8
  278. claude_mpm/init.py +224 -4
  279. claude_mpm/mcp/__init__.py +9 -0
  280. claude_mpm/mcp/google_workspace_server.py +610 -0
  281. claude_mpm/scripts/claude-hook-handler.sh +46 -19
  282. claude_mpm/services/agents/agent_recommendation_service.py +8 -8
  283. claude_mpm/services/agents/agent_selection_service.py +2 -2
  284. claude_mpm/services/agents/cache_git_manager.py +1 -1
  285. claude_mpm/services/agents/deployment/agent_discovery_service.py +3 -1
  286. claude_mpm/services/agents/deployment/agent_format_converter.py +25 -13
  287. claude_mpm/services/agents/deployment/agent_template_builder.py +37 -17
  288. claude_mpm/services/agents/deployment/async_agent_deployment.py +31 -27
  289. claude_mpm/services/agents/deployment/deployment_reconciler.py +577 -0
  290. claude_mpm/services/agents/deployment/local_template_deployment.py +3 -1
  291. claude_mpm/services/agents/deployment/multi_source_deployment_service.py +36 -8
  292. claude_mpm/services/agents/deployment/remote_agent_discovery_service.py +50 -26
  293. claude_mpm/services/agents/deployment/startup_reconciliation.py +138 -0
  294. claude_mpm/services/agents/git_source_manager.py +21 -2
  295. claude_mpm/services/agents/loading/framework_agent_loader.py +75 -2
  296. claude_mpm/services/agents/single_tier_deployment_service.py +4 -4
  297. claude_mpm/services/agents/sources/git_source_sync_service.py +116 -5
  298. claude_mpm/services/agents/startup_sync.py +5 -2
  299. claude_mpm/services/cli/__init__.py +3 -0
  300. claude_mpm/services/cli/incremental_pause_manager.py +561 -0
  301. claude_mpm/services/cli/session_resume_helper.py +10 -2
  302. claude_mpm/services/command_deployment_service.py +44 -26
  303. claude_mpm/services/delegation_detector.py +175 -0
  304. claude_mpm/services/diagnostics/checks/agent_sources_check.py +30 -0
  305. claude_mpm/services/diagnostics/checks/configuration_check.py +24 -0
  306. claude_mpm/services/diagnostics/checks/installation_check.py +22 -0
  307. claude_mpm/services/diagnostics/checks/mcp_services_check.py +23 -0
  308. claude_mpm/services/diagnostics/doctor_reporter.py +31 -1
  309. claude_mpm/services/diagnostics/models.py +14 -1
  310. claude_mpm/services/event_log.py +325 -0
  311. claude_mpm/services/hook_installer_service.py +77 -8
  312. claude_mpm/services/infrastructure/__init__.py +4 -0
  313. claude_mpm/services/infrastructure/context_usage_tracker.py +291 -0
  314. claude_mpm/services/infrastructure/resume_log_generator.py +24 -5
  315. claude_mpm/services/mcp_config_manager.py +99 -19
  316. claude_mpm/services/mcp_service_registry.py +294 -0
  317. claude_mpm/services/monitor/daemon_manager.py +15 -4
  318. claude_mpm/services/monitor/management/lifecycle.py +8 -3
  319. claude_mpm/services/monitor/server.py +111 -16
  320. claude_mpm/services/pm_skills_deployer.py +302 -94
  321. claude_mpm/services/profile_manager.py +10 -4
  322. claude_mpm/services/skills/git_skill_source_manager.py +192 -29
  323. claude_mpm/services/skills/selective_skill_deployer.py +211 -46
  324. claude_mpm/services/skills/skill_discovery_service.py +74 -4
  325. claude_mpm/services/skills_deployer.py +192 -70
  326. claude_mpm/services/socketio/handlers/hook.py +14 -7
  327. claude_mpm/services/socketio/server/main.py +12 -4
  328. claude_mpm/skills/__init__.py +2 -1
  329. claude_mpm/skills/bundled/collaboration/brainstorming/SKILL.md +79 -0
  330. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/SKILL.md +178 -0
  331. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/agent-prompts.md +577 -0
  332. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/coordination-patterns.md +467 -0
  333. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/examples.md +537 -0
  334. claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/troubleshooting.md +730 -0
  335. claude_mpm/skills/bundled/collaboration/git-worktrees.md +317 -0
  336. claude_mpm/skills/bundled/collaboration/requesting-code-review/SKILL.md +112 -0
  337. claude_mpm/skills/bundled/collaboration/requesting-code-review/references/code-reviewer-template.md +146 -0
  338. claude_mpm/skills/bundled/collaboration/requesting-code-review/references/review-examples.md +412 -0
  339. claude_mpm/skills/bundled/collaboration/stacked-prs.md +251 -0
  340. claude_mpm/skills/bundled/collaboration/writing-plans/SKILL.md +81 -0
  341. claude_mpm/skills/bundled/collaboration/writing-plans/references/best-practices.md +362 -0
  342. claude_mpm/skills/bundled/collaboration/writing-plans/references/plan-structure-templates.md +312 -0
  343. claude_mpm/skills/bundled/debugging/root-cause-tracing/SKILL.md +152 -0
  344. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/advanced-techniques.md +668 -0
  345. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/examples.md +587 -0
  346. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/integration.md +438 -0
  347. claude_mpm/skills/bundled/debugging/root-cause-tracing/references/tracing-techniques.md +391 -0
  348. claude_mpm/skills/bundled/debugging/systematic-debugging/CREATION-LOG.md +119 -0
  349. claude_mpm/skills/bundled/debugging/systematic-debugging/SKILL.md +148 -0
  350. claude_mpm/skills/bundled/debugging/systematic-debugging/references/anti-patterns.md +483 -0
  351. claude_mpm/skills/bundled/debugging/systematic-debugging/references/examples.md +452 -0
  352. claude_mpm/skills/bundled/debugging/systematic-debugging/references/troubleshooting.md +449 -0
  353. claude_mpm/skills/bundled/debugging/systematic-debugging/references/workflow.md +411 -0
  354. claude_mpm/skills/bundled/debugging/systematic-debugging/test-academic.md +14 -0
  355. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-1.md +58 -0
  356. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-2.md +68 -0
  357. claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-3.md +69 -0
  358. claude_mpm/skills/bundled/debugging/verification-before-completion/SKILL.md +131 -0
  359. claude_mpm/skills/bundled/debugging/verification-before-completion/references/gate-function.md +325 -0
  360. claude_mpm/skills/bundled/debugging/verification-before-completion/references/integration-and-workflows.md +490 -0
  361. claude_mpm/skills/bundled/debugging/verification-before-completion/references/red-flags-and-failures.md +425 -0
  362. claude_mpm/skills/bundled/debugging/verification-before-completion/references/verification-patterns.md +499 -0
  363. claude_mpm/skills/bundled/infrastructure/env-manager/INTEGRATION.md +611 -0
  364. claude_mpm/skills/bundled/infrastructure/env-manager/README.md +596 -0
  365. claude_mpm/skills/bundled/infrastructure/env-manager/SKILL.md +260 -0
  366. claude_mpm/skills/bundled/infrastructure/env-manager/examples/nextjs-env-structure.md +315 -0
  367. claude_mpm/skills/bundled/infrastructure/env-manager/references/frameworks.md +436 -0
  368. claude_mpm/skills/bundled/infrastructure/env-manager/references/security.md +433 -0
  369. claude_mpm/skills/bundled/infrastructure/env-manager/references/synchronization.md +452 -0
  370. claude_mpm/skills/bundled/infrastructure/env-manager/references/troubleshooting.md +404 -0
  371. claude_mpm/skills/bundled/infrastructure/env-manager/references/validation.md +420 -0
  372. claude_mpm/skills/bundled/main/artifacts-builder/SKILL.md +86 -0
  373. claude_mpm/skills/bundled/main/internal-comms/SKILL.md +43 -0
  374. claude_mpm/skills/bundled/main/internal-comms/examples/3p-updates.md +47 -0
  375. claude_mpm/skills/bundled/main/internal-comms/examples/company-newsletter.md +65 -0
  376. claude_mpm/skills/bundled/main/internal-comms/examples/faq-answers.md +30 -0
  377. claude_mpm/skills/bundled/main/internal-comms/examples/general-comms.md +16 -0
  378. claude_mpm/skills/bundled/main/mcp-builder/SKILL.md +160 -0
  379. claude_mpm/skills/bundled/main/mcp-builder/reference/design_principles.md +412 -0
  380. claude_mpm/skills/bundled/main/mcp-builder/reference/evaluation.md +602 -0
  381. claude_mpm/skills/bundled/main/mcp-builder/reference/mcp_best_practices.md +915 -0
  382. claude_mpm/skills/bundled/main/mcp-builder/reference/node_mcp_server.md +916 -0
  383. claude_mpm/skills/bundled/main/mcp-builder/reference/python_mcp_server.md +752 -0
  384. claude_mpm/skills/bundled/main/mcp-builder/reference/workflow.md +1237 -0
  385. claude_mpm/skills/bundled/main/skill-creator/SKILL.md +189 -0
  386. claude_mpm/skills/bundled/main/skill-creator/references/best-practices.md +500 -0
  387. claude_mpm/skills/bundled/main/skill-creator/references/creation-workflow.md +464 -0
  388. claude_mpm/skills/bundled/main/skill-creator/references/examples.md +619 -0
  389. claude_mpm/skills/bundled/main/skill-creator/references/progressive-disclosure.md +437 -0
  390. claude_mpm/skills/bundled/main/skill-creator/references/skill-structure.md +231 -0
  391. claude_mpm/skills/bundled/php/espocrm-development/SKILL.md +170 -0
  392. claude_mpm/skills/bundled/php/espocrm-development/references/architecture.md +602 -0
  393. claude_mpm/skills/bundled/php/espocrm-development/references/common-tasks.md +821 -0
  394. claude_mpm/skills/bundled/php/espocrm-development/references/development-workflow.md +742 -0
  395. claude_mpm/skills/bundled/php/espocrm-development/references/frontend-customization.md +726 -0
  396. claude_mpm/skills/bundled/php/espocrm-development/references/hooks-and-services.md +764 -0
  397. claude_mpm/skills/bundled/php/espocrm-development/references/testing-debugging.md +831 -0
  398. claude_mpm/skills/bundled/pm/mpm/SKILL.md +38 -0
  399. claude_mpm/skills/bundled/pm/mpm-agent-update-workflow/SKILL.md +75 -0
  400. claude_mpm/skills/bundled/pm/mpm-bug-reporting/SKILL.md +248 -0
  401. claude_mpm/skills/bundled/pm/mpm-circuit-breaker-enforcement/SKILL.md +476 -0
  402. claude_mpm/skills/bundled/pm/mpm-config/SKILL.md +29 -0
  403. claude_mpm/skills/bundled/pm/mpm-delegation-patterns/SKILL.md +167 -0
  404. claude_mpm/skills/bundled/pm/mpm-doctor/SKILL.md +53 -0
  405. claude_mpm/skills/bundled/pm/mpm-git-file-tracking/SKILL.md +113 -0
  406. claude_mpm/skills/bundled/pm/mpm-help/SKILL.md +35 -0
  407. claude_mpm/skills/bundled/pm/mpm-init/SKILL.md +125 -0
  408. claude_mpm/skills/bundled/pm/mpm-monitor/SKILL.md +32 -0
  409. claude_mpm/skills/bundled/pm/mpm-organize/SKILL.md +121 -0
  410. claude_mpm/skills/bundled/pm/mpm-postmortem/SKILL.md +22 -0
  411. claude_mpm/skills/bundled/pm/mpm-pr-workflow/SKILL.md +124 -0
  412. claude_mpm/skills/bundled/pm/mpm-session-management/SKILL.md +312 -0
  413. claude_mpm/skills/bundled/pm/mpm-session-pause/SKILL.md +170 -0
  414. claude_mpm/skills/bundled/pm/mpm-session-resume/SKILL.md +31 -0
  415. claude_mpm/skills/bundled/pm/mpm-status/SKILL.md +37 -0
  416. claude_mpm/skills/bundled/pm/mpm-teaching-mode/SKILL.md +657 -0
  417. claude_mpm/skills/bundled/pm/mpm-ticket-view/SKILL.md +110 -0
  418. claude_mpm/skills/bundled/pm/mpm-ticketing-integration/SKILL.md +154 -0
  419. claude_mpm/skills/bundled/pm/mpm-tool-usage-guide/SKILL.md +386 -0
  420. claude_mpm/skills/bundled/pm/mpm-verification-protocols/SKILL.md +198 -0
  421. claude_mpm/skills/bundled/pm/mpm-version/SKILL.md +21 -0
  422. claude_mpm/skills/bundled/react/flexlayout-react.md +742 -0
  423. claude_mpm/skills/bundled/rust/desktop-applications/SKILL.md +226 -0
  424. claude_mpm/skills/bundled/rust/desktop-applications/references/architecture-patterns.md +901 -0
  425. claude_mpm/skills/bundled/rust/desktop-applications/references/native-gui-frameworks.md +901 -0
  426. claude_mpm/skills/bundled/rust/desktop-applications/references/platform-integration.md +775 -0
  427. claude_mpm/skills/bundled/rust/desktop-applications/references/state-management.md +937 -0
  428. claude_mpm/skills/bundled/rust/desktop-applications/references/tauri-framework.md +770 -0
  429. claude_mpm/skills/bundled/rust/desktop-applications/references/testing-deployment.md +961 -0
  430. claude_mpm/skills/bundled/security-scanning.md +112 -0
  431. claude_mpm/skills/bundled/tauri/tauri-async-patterns.md +495 -0
  432. claude_mpm/skills/bundled/tauri/tauri-build-deploy.md +599 -0
  433. claude_mpm/skills/bundled/tauri/tauri-command-patterns.md +535 -0
  434. claude_mpm/skills/bundled/tauri/tauri-error-handling.md +613 -0
  435. claude_mpm/skills/bundled/tauri/tauri-event-system.md +648 -0
  436. claude_mpm/skills/bundled/tauri/tauri-file-system.md +673 -0
  437. claude_mpm/skills/bundled/tauri/tauri-frontend-integration.md +767 -0
  438. claude_mpm/skills/bundled/tauri/tauri-performance.md +669 -0
  439. claude_mpm/skills/bundled/tauri/tauri-state-management.md +573 -0
  440. claude_mpm/skills/bundled/tauri/tauri-testing.md +384 -0
  441. claude_mpm/skills/bundled/tauri/tauri-window-management.md +628 -0
  442. claude_mpm/skills/bundled/testing/condition-based-waiting/SKILL.md +119 -0
  443. claude_mpm/skills/bundled/testing/condition-based-waiting/references/patterns-and-implementation.md +253 -0
  444. claude_mpm/skills/bundled/testing/test-driven-development/SKILL.md +145 -0
  445. claude_mpm/skills/bundled/testing/test-driven-development/references/anti-patterns.md +543 -0
  446. claude_mpm/skills/bundled/testing/test-driven-development/references/examples.md +741 -0
  447. claude_mpm/skills/bundled/testing/test-driven-development/references/integration.md +470 -0
  448. claude_mpm/skills/bundled/testing/test-driven-development/references/philosophy.md +458 -0
  449. claude_mpm/skills/bundled/testing/test-driven-development/references/workflow.md +639 -0
  450. claude_mpm/skills/bundled/testing/test-quality-inspector/SKILL.md +458 -0
  451. claude_mpm/skills/bundled/testing/test-quality-inspector/examples/example-inspection-report.md +411 -0
  452. claude_mpm/skills/bundled/testing/test-quality-inspector/references/assertion-quality.md +317 -0
  453. claude_mpm/skills/bundled/testing/test-quality-inspector/references/inspection-checklist.md +270 -0
  454. claude_mpm/skills/bundled/testing/test-quality-inspector/references/red-flags.md +436 -0
  455. claude_mpm/skills/bundled/testing/testing-anti-patterns/SKILL.md +140 -0
  456. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/completeness-anti-patterns.md +572 -0
  457. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/core-anti-patterns.md +411 -0
  458. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/detection-guide.md +569 -0
  459. claude_mpm/skills/bundled/testing/testing-anti-patterns/references/tdd-connection.md +695 -0
  460. claude_mpm/skills/bundled/testing/webapp-testing/SKILL.md +184 -0
  461. claude_mpm/skills/bundled/testing/webapp-testing/decision-tree.md +459 -0
  462. claude_mpm/skills/bundled/testing/webapp-testing/playwright-patterns.md +479 -0
  463. claude_mpm/skills/bundled/testing/webapp-testing/reconnaissance-pattern.md +687 -0
  464. claude_mpm/skills/bundled/testing/webapp-testing/server-management.md +758 -0
  465. claude_mpm/skills/bundled/testing/webapp-testing/troubleshooting.md +868 -0
  466. claude_mpm/skills/registry.py +295 -90
  467. claude_mpm/skills/skill_manager.py +29 -23
  468. claude_mpm/templates/.pre-commit-config.yaml +112 -0
  469. claude_mpm/utils/agent_dependency_loader.py +103 -4
  470. claude_mpm/utils/robust_installer.py +45 -24
  471. claude_mpm-5.6.72.dist-info/METADATA +416 -0
  472. {claude_mpm-5.4.41.dist-info → claude_mpm-5.6.72.dist-info}/RECORD +477 -159
  473. {claude_mpm-5.4.41.dist-info → claude_mpm-5.6.72.dist-info}/WHEEL +1 -1
  474. {claude_mpm-5.4.41.dist-info → claude_mpm-5.6.72.dist-info}/entry_points.txt +2 -0
  475. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/0.B_FtCwCQ.css +0 -1
  476. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/2.Cl_eSA4x.css +0 -1
  477. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BgChzWQ1.js +0 -1
  478. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CIXEwuWe.js +0 -1
  479. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CWc5urbQ.js +0 -1
  480. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DMkZpdF2.js +0 -2
  481. claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/app.DTL5mJO-.js +0 -2
  482. claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/start.DzuEhzqh.js +0 -1
  483. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/1.DFLC8jdE.js +0 -1
  484. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/2.DPvEihJJ.js +0 -10
  485. claude_mpm/hooks/claude_hooks/__pycache__/installer.cpython-311.pyc +0 -0
  486. claude_mpm/hooks/claude_hooks/services/__pycache__/connection_manager.cpython-311.pyc +0 -0
  487. claude_mpm-5.4.41.dist-info/METADATA +0 -998
  488. {claude_mpm-5.4.41.dist-info → claude_mpm-5.6.72.dist-info}/licenses/LICENSE +0 -0
  489. {claude_mpm-5.4.41.dist-info → claude_mpm-5.6.72.dist-info}/licenses/LICENSE-FAQ.md +0 -0
  490. {claude_mpm-5.4.41.dist-info → claude_mpm-5.6.72.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,610 @@
1
+ """Google Workspace MCP server integrated with claude-mpm OAuth storage.
2
+
3
+ This MCP server provides tools for interacting with Google Workspace APIs
4
+ (Calendar, Gmail, Drive) using OAuth tokens managed by claude-mpm's
5
+ TokenStorage system.
6
+
7
+ The server automatically handles token refresh when tokens expire,
8
+ using the OAuthManager for seamless re-authentication.
9
+ """
10
+
11
+ import asyncio
12
+ import json
13
+ import logging
14
+ from typing import Any, Optional
15
+
16
+ import httpx
17
+ from mcp.server import Server
18
+ from mcp.server.stdio import stdio_server
19
+ from mcp.types import TextContent, Tool
20
+
21
+ from claude_mpm.auth import OAuthManager, TokenStatus, TokenStorage
22
+
23
+ # Configure logging
24
+ logging.basicConfig(level=logging.INFO)
25
+ logger = logging.getLogger(__name__)
26
+
27
+ # Service name for token storage - matches workspace-mcp convention
28
+ SERVICE_NAME = "workspace-mcp"
29
+
30
+ # Google API base URLs
31
+ CALENDAR_API_BASE = "https://www.googleapis.com/calendar/v3"
32
+ GMAIL_API_BASE = "https://gmail.googleapis.com/gmail/v1"
33
+ DRIVE_API_BASE = "https://www.googleapis.com/drive/v3"
34
+
35
+
36
+ class GoogleWorkspaceServer:
37
+ """MCP server for Google Workspace APIs.
38
+
39
+ Integrates with claude-mpm's TokenStorage for credential management
40
+ and provides tools for Calendar, Gmail, and Drive operations.
41
+
42
+ Attributes:
43
+ server: MCP Server instance.
44
+ storage: TokenStorage for retrieving OAuth tokens.
45
+ manager: OAuthManager for token refresh operations.
46
+ """
47
+
48
+ def __init__(self) -> None:
49
+ """Initialize the Google Workspace MCP server."""
50
+ self.server = Server("google-workspace-mpm")
51
+ self.storage = TokenStorage()
52
+ self.manager = OAuthManager(storage=self.storage)
53
+ self._setup_handlers()
54
+
55
+ def _setup_handlers(self) -> None:
56
+ """Register MCP tool handlers."""
57
+
58
+ @self.server.list_tools()
59
+ async def list_tools() -> list[Tool]:
60
+ """Return list of available tools."""
61
+ return [
62
+ Tool(
63
+ name="list_calendars",
64
+ description="List all calendars accessible by the authenticated user",
65
+ inputSchema={
66
+ "type": "object",
67
+ "properties": {},
68
+ "required": [],
69
+ },
70
+ ),
71
+ Tool(
72
+ name="get_events",
73
+ description="Get events from a calendar within a time range",
74
+ inputSchema={
75
+ "type": "object",
76
+ "properties": {
77
+ "calendar_id": {
78
+ "type": "string",
79
+ "description": "Calendar ID (default: 'primary')",
80
+ "default": "primary",
81
+ },
82
+ "time_min": {
83
+ "type": "string",
84
+ "description": "Start time in RFC3339 format (e.g., '2024-01-01T00:00:00Z')",
85
+ },
86
+ "time_max": {
87
+ "type": "string",
88
+ "description": "End time in RFC3339 format",
89
+ },
90
+ "max_results": {
91
+ "type": "integer",
92
+ "description": "Maximum number of events to return (default: 10)",
93
+ "default": 10,
94
+ },
95
+ },
96
+ "required": [],
97
+ },
98
+ ),
99
+ Tool(
100
+ name="search_gmail_messages",
101
+ description="Search Gmail messages using a query string",
102
+ inputSchema={
103
+ "type": "object",
104
+ "properties": {
105
+ "query": {
106
+ "type": "string",
107
+ "description": "Gmail search query (e.g., 'from:user@example.com subject:meeting')",
108
+ },
109
+ "max_results": {
110
+ "type": "integer",
111
+ "description": "Maximum number of messages to return (default: 10)",
112
+ "default": 10,
113
+ },
114
+ },
115
+ "required": ["query"],
116
+ },
117
+ ),
118
+ Tool(
119
+ name="get_gmail_message_content",
120
+ description="Get the full content of a Gmail message by ID",
121
+ inputSchema={
122
+ "type": "object",
123
+ "properties": {
124
+ "message_id": {
125
+ "type": "string",
126
+ "description": "Gmail message ID",
127
+ },
128
+ },
129
+ "required": ["message_id"],
130
+ },
131
+ ),
132
+ Tool(
133
+ name="search_drive_files",
134
+ description="Search Google Drive files using a query string",
135
+ inputSchema={
136
+ "type": "object",
137
+ "properties": {
138
+ "query": {
139
+ "type": "string",
140
+ "description": "Drive search query (e.g., 'name contains \"report\"')",
141
+ },
142
+ "max_results": {
143
+ "type": "integer",
144
+ "description": "Maximum number of files to return (default: 10)",
145
+ "default": 10,
146
+ },
147
+ },
148
+ "required": ["query"],
149
+ },
150
+ ),
151
+ Tool(
152
+ name="get_drive_file_content",
153
+ description="Get the content of a Google Drive file by ID (text files only)",
154
+ inputSchema={
155
+ "type": "object",
156
+ "properties": {
157
+ "file_id": {
158
+ "type": "string",
159
+ "description": "Google Drive file ID",
160
+ },
161
+ },
162
+ "required": ["file_id"],
163
+ },
164
+ ),
165
+ ]
166
+
167
+ @self.server.call_tool()
168
+ async def call_tool(name: str, arguments: dict[str, Any]) -> list[TextContent]:
169
+ """Handle tool calls."""
170
+ try:
171
+ result = await self._dispatch_tool(name, arguments)
172
+ return [TextContent(type="text", text=json.dumps(result, indent=2))]
173
+ except Exception as e:
174
+ logger.exception(f"Error calling tool {name}")
175
+ return [
176
+ TextContent(
177
+ type="text",
178
+ text=json.dumps({"error": str(e)}, indent=2),
179
+ )
180
+ ]
181
+
182
+ async def _get_access_token(self) -> str:
183
+ """Get a valid access token, refreshing if necessary.
184
+
185
+ Returns:
186
+ Valid access token string.
187
+
188
+ Raises:
189
+ RuntimeError: If no token is available or refresh fails.
190
+ """
191
+ status = self.storage.get_status(SERVICE_NAME)
192
+
193
+ if status == TokenStatus.MISSING:
194
+ raise RuntimeError(
195
+ f"No OAuth token found for service '{SERVICE_NAME}'. "
196
+ "Please authenticate first using: claude-mpm auth login google"
197
+ )
198
+
199
+ if status == TokenStatus.INVALID:
200
+ raise RuntimeError(
201
+ f"OAuth token for service '{SERVICE_NAME}' is invalid or corrupted. "
202
+ "Please re-authenticate using: claude-mpm auth login google"
203
+ )
204
+
205
+ # Try to refresh if expired
206
+ if status == TokenStatus.EXPIRED:
207
+ logger.info("Token expired, attempting refresh...")
208
+ token = await self.manager.refresh_if_needed(SERVICE_NAME)
209
+ if token is None:
210
+ raise RuntimeError(
211
+ "Token refresh failed. Please re-authenticate using: "
212
+ "claude-mpm auth login google"
213
+ )
214
+ return token.access_token
215
+
216
+ # Token is valid
217
+ stored = self.storage.retrieve(SERVICE_NAME)
218
+ if stored is None:
219
+ raise RuntimeError("Unexpected error: token retrieval failed")
220
+
221
+ return stored.token.access_token
222
+
223
+ async def _make_request(
224
+ self,
225
+ method: str,
226
+ url: str,
227
+ params: Optional[dict[str, Any]] = None,
228
+ json_data: Optional[dict[str, Any]] = None,
229
+ ) -> dict[str, Any]:
230
+ """Make an authenticated HTTP request to Google APIs.
231
+
232
+ Args:
233
+ method: HTTP method (GET, POST, etc.).
234
+ url: Full URL to request.
235
+ params: Optional query parameters.
236
+ json_data: Optional JSON body data.
237
+
238
+ Returns:
239
+ JSON response as a dictionary.
240
+
241
+ Raises:
242
+ httpx.HTTPStatusError: If the request fails.
243
+ """
244
+ access_token = await self._get_access_token()
245
+
246
+ async with httpx.AsyncClient() as client:
247
+ response = await client.request(
248
+ method=method,
249
+ url=url,
250
+ params=params,
251
+ json=json_data,
252
+ headers={
253
+ "Authorization": f"Bearer {access_token}",
254
+ "Accept": "application/json",
255
+ },
256
+ timeout=30.0,
257
+ )
258
+ response.raise_for_status()
259
+ result: dict[str, Any] = response.json()
260
+ return result
261
+
262
+ async def _dispatch_tool(
263
+ self, name: str, arguments: dict[str, Any]
264
+ ) -> dict[str, Any]:
265
+ """Dispatch tool call to appropriate handler.
266
+
267
+ Args:
268
+ name: Tool name.
269
+ arguments: Tool arguments.
270
+
271
+ Returns:
272
+ Tool result as dictionary.
273
+
274
+ Raises:
275
+ ValueError: If tool name is not recognized.
276
+ """
277
+ handlers = {
278
+ "list_calendars": self._list_calendars,
279
+ "get_events": self._get_events,
280
+ "search_gmail_messages": self._search_gmail_messages,
281
+ "get_gmail_message_content": self._get_gmail_message_content,
282
+ "search_drive_files": self._search_drive_files,
283
+ "get_drive_file_content": self._get_drive_file_content,
284
+ }
285
+
286
+ handler = handlers.get(name)
287
+ if handler is None:
288
+ raise ValueError(f"Unknown tool: {name}")
289
+
290
+ return await handler(arguments)
291
+
292
+ async def _list_calendars(self, arguments: dict[str, Any]) -> dict[str, Any]:
293
+ """List all calendars accessible by the user.
294
+
295
+ Args:
296
+ arguments: Tool arguments (not used).
297
+
298
+ Returns:
299
+ List of calendars with id, summary, and access role.
300
+ """
301
+ url = f"{CALENDAR_API_BASE}/users/me/calendarList"
302
+ response = await self._make_request("GET", url)
303
+
304
+ calendars = []
305
+ for item in response.get("items", []):
306
+ calendars.append(
307
+ {
308
+ "id": item.get("id"),
309
+ "summary": item.get("summary"),
310
+ "description": item.get("description"),
311
+ "access_role": item.get("accessRole"),
312
+ "primary": item.get("primary", False),
313
+ }
314
+ )
315
+
316
+ return {"calendars": calendars, "count": len(calendars)}
317
+
318
+ async def _get_events(self, arguments: dict[str, Any]) -> dict[str, Any]:
319
+ """Get events from a calendar.
320
+
321
+ Args:
322
+ arguments: Tool arguments with calendar_id, time_min, time_max, max_results.
323
+
324
+ Returns:
325
+ List of events with summary, start, end times.
326
+ """
327
+ calendar_id = arguments.get("calendar_id", "primary")
328
+ time_min = arguments.get("time_min")
329
+ time_max = arguments.get("time_max")
330
+ max_results = arguments.get("max_results", 10)
331
+
332
+ url = f"{CALENDAR_API_BASE}/calendars/{calendar_id}/events"
333
+ params: dict[str, Any] = {
334
+ "maxResults": max_results,
335
+ "singleEvents": True,
336
+ "orderBy": "startTime",
337
+ }
338
+
339
+ if time_min:
340
+ params["timeMin"] = time_min
341
+ if time_max:
342
+ params["timeMax"] = time_max
343
+
344
+ response = await self._make_request("GET", url, params=params)
345
+
346
+ events = []
347
+ for item in response.get("items", []):
348
+ start = item.get("start", {})
349
+ end = item.get("end", {})
350
+ events.append(
351
+ {
352
+ "id": item.get("id"),
353
+ "summary": item.get("summary"),
354
+ "description": item.get("description"),
355
+ "start": start.get("dateTime") or start.get("date"),
356
+ "end": end.get("dateTime") or end.get("date"),
357
+ "location": item.get("location"),
358
+ "attendees": [a.get("email") for a in item.get("attendees", [])],
359
+ }
360
+ )
361
+
362
+ return {"events": events, "count": len(events)}
363
+
364
+ async def _search_gmail_messages(self, arguments: dict[str, Any]) -> dict[str, Any]:
365
+ """Search Gmail messages.
366
+
367
+ Args:
368
+ arguments: Tool arguments with query and max_results.
369
+
370
+ Returns:
371
+ List of message snippets with id, thread_id, subject, from, date.
372
+ """
373
+ query = arguments.get("query", "")
374
+ max_results = arguments.get("max_results", 10)
375
+
376
+ url = f"{GMAIL_API_BASE}/users/me/messages"
377
+ params = {"q": query, "maxResults": max_results}
378
+
379
+ response = await self._make_request("GET", url, params=params)
380
+
381
+ messages = []
382
+ for msg in response.get("messages", []):
383
+ # Get message metadata
384
+ msg_url = f"{GMAIL_API_BASE}/users/me/messages/{msg['id']}"
385
+ msg_detail = await self._make_request(
386
+ "GET", msg_url, params={"format": "metadata"}
387
+ )
388
+
389
+ headers = {
390
+ h["name"]: h["value"]
391
+ for h in msg_detail.get("payload", {}).get("headers", [])
392
+ }
393
+
394
+ messages.append(
395
+ {
396
+ "id": msg["id"],
397
+ "thread_id": msg.get("threadId"),
398
+ "subject": headers.get("Subject"),
399
+ "from": headers.get("From"),
400
+ "to": headers.get("To"),
401
+ "date": headers.get("Date"),
402
+ "snippet": msg_detail.get("snippet"),
403
+ }
404
+ )
405
+
406
+ return {"messages": messages, "count": len(messages)}
407
+
408
+ async def _get_gmail_message_content(
409
+ self, arguments: dict[str, Any]
410
+ ) -> dict[str, Any]:
411
+ """Get full content of a Gmail message.
412
+
413
+ Args:
414
+ arguments: Tool arguments with message_id.
415
+
416
+ Returns:
417
+ Message content including headers and body.
418
+ """
419
+ message_id = arguments["message_id"]
420
+
421
+ url = f"{GMAIL_API_BASE}/users/me/messages/{message_id}"
422
+ response = await self._make_request("GET", url, params={"format": "full"})
423
+
424
+ headers = {
425
+ h["name"]: h["value"]
426
+ for h in response.get("payload", {}).get("headers", [])
427
+ }
428
+
429
+ # Extract body content
430
+ body = self._extract_message_body(response.get("payload", {}))
431
+
432
+ return {
433
+ "id": response.get("id"),
434
+ "thread_id": response.get("threadId"),
435
+ "subject": headers.get("Subject"),
436
+ "from": headers.get("From"),
437
+ "to": headers.get("To"),
438
+ "cc": headers.get("Cc"),
439
+ "date": headers.get("Date"),
440
+ "body": body,
441
+ "labels": response.get("labelIds", []),
442
+ }
443
+
444
+ def _extract_message_body(self, payload: dict[str, Any]) -> str:
445
+ """Extract message body from Gmail payload.
446
+
447
+ Handles both simple and multipart messages.
448
+
449
+ Args:
450
+ payload: Gmail message payload.
451
+
452
+ Returns:
453
+ Decoded message body text.
454
+ """
455
+ import base64
456
+
457
+ # Simple message with body data
458
+ if "body" in payload and payload["body"].get("data"):
459
+ data = payload["body"]["data"]
460
+ return base64.urlsafe_b64decode(data).decode("utf-8", errors="replace")
461
+
462
+ # Multipart message
463
+ parts = payload.get("parts", [])
464
+ for part in parts:
465
+ mime_type = part.get("mimeType", "")
466
+ if mime_type == "text/plain":
467
+ data = part.get("body", {}).get("data", "")
468
+ if data:
469
+ return base64.urlsafe_b64decode(data).decode(
470
+ "utf-8", errors="replace"
471
+ )
472
+ elif mime_type.startswith("multipart/"):
473
+ # Recursively extract from nested parts
474
+ result = self._extract_message_body(part)
475
+ if result:
476
+ return result
477
+
478
+ # Fallback to HTML if no plain text
479
+ for part in parts:
480
+ if part.get("mimeType") == "text/html":
481
+ data = part.get("body", {}).get("data", "")
482
+ if data:
483
+ return base64.urlsafe_b64decode(data).decode(
484
+ "utf-8", errors="replace"
485
+ )
486
+
487
+ return ""
488
+
489
+ async def _search_drive_files(self, arguments: dict[str, Any]) -> dict[str, Any]:
490
+ """Search Google Drive files.
491
+
492
+ Args:
493
+ arguments: Tool arguments with query and max_results.
494
+
495
+ Returns:
496
+ List of files with id, name, mimeType, modifiedTime.
497
+ """
498
+ query = arguments.get("query", "")
499
+ max_results = arguments.get("max_results", 10)
500
+
501
+ url = f"{DRIVE_API_BASE}/files"
502
+ params = {
503
+ "q": query,
504
+ "pageSize": max_results,
505
+ "fields": "files(id,name,mimeType,modifiedTime,size,webViewLink,owners)",
506
+ }
507
+
508
+ response = await self._make_request("GET", url, params=params)
509
+
510
+ files = []
511
+ for item in response.get("files", []):
512
+ files.append(
513
+ {
514
+ "id": item.get("id"),
515
+ "name": item.get("name"),
516
+ "mimeType": item.get("mimeType"),
517
+ "modifiedTime": item.get("modifiedTime"),
518
+ "size": item.get("size"),
519
+ "webViewLink": item.get("webViewLink"),
520
+ "owners": [o.get("emailAddress") for o in item.get("owners", [])],
521
+ }
522
+ )
523
+
524
+ return {"files": files, "count": len(files)}
525
+
526
+ async def _get_drive_file_content(
527
+ self, arguments: dict[str, Any]
528
+ ) -> dict[str, Any]:
529
+ """Get content of a Google Drive file.
530
+
531
+ Args:
532
+ arguments: Tool arguments with file_id.
533
+
534
+ Returns:
535
+ File metadata and content (for exportable types).
536
+ """
537
+ file_id = arguments["file_id"]
538
+
539
+ # First get file metadata
540
+ meta_url = f"{DRIVE_API_BASE}/files/{file_id}"
541
+ metadata = await self._make_request(
542
+ "GET", meta_url, params={"fields": "id,name,mimeType,size"}
543
+ )
544
+
545
+ mime_type = metadata.get("mimeType", "")
546
+
547
+ # Google Docs types need export
548
+ export_map = {
549
+ "application/vnd.google-apps.document": "text/plain",
550
+ "application/vnd.google-apps.spreadsheet": "text/csv",
551
+ "application/vnd.google-apps.presentation": "text/plain",
552
+ }
553
+
554
+ access_token = await self._get_access_token()
555
+
556
+ if mime_type in export_map:
557
+ # Export Google Workspace files
558
+ export_url = f"{DRIVE_API_BASE}/files/{file_id}/export"
559
+ async with httpx.AsyncClient() as client:
560
+ response = await client.get(
561
+ export_url,
562
+ params={"mimeType": export_map[mime_type]},
563
+ headers={"Authorization": f"Bearer {access_token}"},
564
+ timeout=30.0,
565
+ )
566
+ response.raise_for_status()
567
+ content = response.text
568
+ else:
569
+ # Download regular files
570
+ download_url = f"{DRIVE_API_BASE}/files/{file_id}"
571
+ async with httpx.AsyncClient() as client:
572
+ response = await client.get(
573
+ download_url,
574
+ params={"alt": "media"},
575
+ headers={"Authorization": f"Bearer {access_token}"},
576
+ timeout=30.0,
577
+ )
578
+ response.raise_for_status()
579
+
580
+ # Try to decode as text, otherwise indicate binary
581
+ try:
582
+ content = response.text
583
+ except UnicodeDecodeError:
584
+ content = f"[Binary file: {metadata.get('size', 'unknown')} bytes]"
585
+
586
+ return {
587
+ "id": metadata.get("id"),
588
+ "name": metadata.get("name"),
589
+ "mimeType": mime_type,
590
+ "content": content,
591
+ }
592
+
593
+ async def run(self) -> None:
594
+ """Run the MCP server using stdio transport."""
595
+ async with stdio_server() as (read_stream, write_stream):
596
+ await self.server.run(
597
+ read_stream,
598
+ write_stream,
599
+ self.server.create_initialization_options(),
600
+ )
601
+
602
+
603
+ def main() -> None:
604
+ """Entry point for the Google Workspace MCP server."""
605
+ server = GoogleWorkspaceServer()
606
+ asyncio.run(server.run())
607
+
608
+
609
+ if __name__ == "__main__":
610
+ main()