claude-mpm 5.0.9__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.
- claude_mpm/VERSION +1 -1
- claude_mpm/__init__.py +4 -0
- claude_mpm/agents/BASE_AGENT.md +164 -0
- claude_mpm/agents/CLAUDE_MPM_OUTPUT_STYLE.md +115 -0
- claude_mpm/agents/CLAUDE_MPM_RESEARCH_OUTPUT_STYLE.md +413 -0
- claude_mpm/agents/CLAUDE_MPM_TEACHER_OUTPUT_STYLE.md +186 -0
- claude_mpm/agents/MEMORY.md +1 -1
- claude_mpm/agents/PM_INSTRUCTIONS.md +479 -616
- claude_mpm/agents/WORKFLOW.md +6 -253
- claude_mpm/agents/agent_loader.py +13 -44
- claude_mpm/agents/base_agent.json +1 -1
- claude_mpm/agents/frontmatter_validator.py +70 -2
- claude_mpm/agents/templates/circuit-breakers.md +457 -62
- claude_mpm/cli/__init__.py +5 -2
- claude_mpm/cli/__main__.py +4 -0
- claude_mpm/cli/chrome_devtools_installer.py +175 -0
- claude_mpm/cli/commands/agent_state_manager.py +18 -27
- claude_mpm/cli/commands/agents.py +177 -41
- claude_mpm/cli/commands/agents_reconcile.py +197 -0
- claude_mpm/cli/commands/auto_configure.py +723 -236
- claude_mpm/cli/commands/autotodos.py +566 -0
- claude_mpm/cli/commands/commander.py +216 -0
- claude_mpm/cli/commands/config.py +88 -2
- claude_mpm/cli/commands/configure.py +1874 -170
- claude_mpm/cli/commands/configure_agent_display.py +27 -6
- claude_mpm/cli/commands/hook_errors.py +60 -60
- claude_mpm/cli/commands/monitor.py +2 -2
- claude_mpm/cli/commands/mpm_init/core.py +232 -46
- claude_mpm/cli/commands/mpm_init/knowledge_extractor.py +481 -0
- claude_mpm/cli/commands/mpm_init/prompts.py +280 -0
- claude_mpm/cli/commands/postmortem.py +1 -1
- claude_mpm/cli/commands/profile.py +276 -0
- claude_mpm/cli/commands/run.py +35 -3
- claude_mpm/cli/commands/skill_source.py +51 -2
- claude_mpm/cli/commands/skills.py +379 -204
- claude_mpm/cli/commands/summarize.py +413 -0
- claude_mpm/cli/executor.py +141 -19
- claude_mpm/cli/interactive/__init__.py +10 -0
- claude_mpm/cli/interactive/agent_wizard.py +115 -60
- claude_mpm/cli/interactive/questionary_styles.py +65 -0
- claude_mpm/cli/interactive/skill_selector.py +481 -0
- claude_mpm/cli/parsers/agents_parser.py +54 -9
- claude_mpm/cli/parsers/auto_configure_parser.py +13 -138
- claude_mpm/cli/parsers/base_parser.py +88 -1
- claude_mpm/cli/parsers/commander_parser.py +116 -0
- claude_mpm/cli/parsers/config_parser.py +153 -83
- claude_mpm/cli/parsers/profile_parser.py +147 -0
- claude_mpm/cli/parsers/run_parser.py +10 -0
- claude_mpm/cli/parsers/skill_source_parser.py +4 -0
- claude_mpm/cli/parsers/skills_parser.py +1 -1
- claude_mpm/cli/startup.py +1017 -266
- claude_mpm/cli/startup_display.py +74 -6
- claude_mpm/cli/startup_logging.py +2 -2
- claude_mpm/cli/utils.py +7 -3
- claude_mpm/commander/__init__.py +78 -0
- claude_mpm/commander/adapters/__init__.py +60 -0
- claude_mpm/commander/adapters/auggie.py +260 -0
- claude_mpm/commander/adapters/base.py +288 -0
- claude_mpm/commander/adapters/claude_code.py +392 -0
- claude_mpm/commander/adapters/codex.py +237 -0
- claude_mpm/commander/adapters/communication.py +366 -0
- claude_mpm/commander/adapters/example_usage.py +310 -0
- claude_mpm/commander/adapters/mpm.py +389 -0
- claude_mpm/commander/adapters/registry.py +204 -0
- claude_mpm/commander/api/__init__.py +16 -0
- claude_mpm/commander/api/app.py +121 -0
- claude_mpm/commander/api/errors.py +133 -0
- claude_mpm/commander/api/routes/__init__.py +8 -0
- claude_mpm/commander/api/routes/events.py +184 -0
- claude_mpm/commander/api/routes/inbox.py +171 -0
- claude_mpm/commander/api/routes/messages.py +148 -0
- claude_mpm/commander/api/routes/projects.py +271 -0
- claude_mpm/commander/api/routes/sessions.py +226 -0
- claude_mpm/commander/api/routes/work.py +296 -0
- claude_mpm/commander/api/schemas.py +186 -0
- claude_mpm/commander/chat/__init__.py +7 -0
- claude_mpm/commander/chat/cli.py +146 -0
- claude_mpm/commander/chat/commands.py +96 -0
- claude_mpm/commander/chat/repl.py +310 -0
- claude_mpm/commander/config.py +51 -0
- claude_mpm/commander/config_loader.py +115 -0
- claude_mpm/commander/core/__init__.py +10 -0
- claude_mpm/commander/core/block_manager.py +325 -0
- claude_mpm/commander/core/response_manager.py +323 -0
- claude_mpm/commander/daemon.py +603 -0
- claude_mpm/commander/env_loader.py +59 -0
- claude_mpm/commander/events/__init__.py +26 -0
- claude_mpm/commander/events/manager.py +332 -0
- claude_mpm/commander/frameworks/__init__.py +12 -0
- claude_mpm/commander/frameworks/base.py +146 -0
- claude_mpm/commander/frameworks/claude_code.py +58 -0
- claude_mpm/commander/frameworks/mpm.py +62 -0
- claude_mpm/commander/inbox/__init__.py +16 -0
- claude_mpm/commander/inbox/dedup.py +128 -0
- claude_mpm/commander/inbox/inbox.py +224 -0
- claude_mpm/commander/inbox/models.py +70 -0
- claude_mpm/commander/instance_manager.py +450 -0
- claude_mpm/commander/llm/__init__.py +6 -0
- claude_mpm/commander/llm/openrouter_client.py +167 -0
- claude_mpm/commander/llm/summarizer.py +70 -0
- claude_mpm/commander/memory/__init__.py +45 -0
- claude_mpm/commander/memory/compression.py +347 -0
- claude_mpm/commander/memory/embeddings.py +230 -0
- claude_mpm/commander/memory/entities.py +310 -0
- claude_mpm/commander/memory/example_usage.py +290 -0
- claude_mpm/commander/memory/integration.py +325 -0
- claude_mpm/commander/memory/search.py +381 -0
- claude_mpm/commander/memory/store.py +657 -0
- claude_mpm/commander/models/__init__.py +18 -0
- claude_mpm/commander/models/events.py +121 -0
- claude_mpm/commander/models/project.py +162 -0
- claude_mpm/commander/models/work.py +214 -0
- claude_mpm/commander/parsing/__init__.py +20 -0
- claude_mpm/commander/parsing/extractor.py +132 -0
- claude_mpm/commander/parsing/output_parser.py +270 -0
- claude_mpm/commander/parsing/patterns.py +100 -0
- claude_mpm/commander/persistence/__init__.py +11 -0
- claude_mpm/commander/persistence/event_store.py +274 -0
- claude_mpm/commander/persistence/state_store.py +309 -0
- claude_mpm/commander/persistence/work_store.py +164 -0
- claude_mpm/commander/polling/__init__.py +13 -0
- claude_mpm/commander/polling/event_detector.py +104 -0
- claude_mpm/commander/polling/output_buffer.py +49 -0
- claude_mpm/commander/polling/output_poller.py +153 -0
- claude_mpm/commander/project_session.py +268 -0
- claude_mpm/commander/proxy/__init__.py +12 -0
- claude_mpm/commander/proxy/formatter.py +89 -0
- claude_mpm/commander/proxy/output_handler.py +191 -0
- claude_mpm/commander/proxy/relay.py +155 -0
- claude_mpm/commander/registry.py +410 -0
- claude_mpm/commander/runtime/__init__.py +10 -0
- claude_mpm/commander/runtime/executor.py +191 -0
- claude_mpm/commander/runtime/monitor.py +346 -0
- claude_mpm/commander/session/__init__.py +6 -0
- claude_mpm/commander/session/context.py +81 -0
- claude_mpm/commander/session/manager.py +59 -0
- claude_mpm/commander/tmux_orchestrator.py +361 -0
- claude_mpm/commander/web/__init__.py +1 -0
- claude_mpm/commander/work/__init__.py +30 -0
- claude_mpm/commander/work/executor.py +207 -0
- claude_mpm/commander/work/queue.py +405 -0
- claude_mpm/commander/workflow/__init__.py +27 -0
- claude_mpm/commander/workflow/event_handler.py +241 -0
- claude_mpm/commander/workflow/notifier.py +146 -0
- claude_mpm/commands/mpm-config.md +36 -0
- claude_mpm/commands/mpm-doctor.md +16 -21
- claude_mpm/commands/mpm-help.md +12 -286
- claude_mpm/commands/mpm-init.md +88 -506
- claude_mpm/commands/mpm-monitor.md +22 -401
- claude_mpm/commands/mpm-organize.md +128 -0
- claude_mpm/commands/mpm-postmortem.md +13 -107
- claude_mpm/commands/mpm-session-resume.md +20 -363
- claude_mpm/commands/mpm-status.md +13 -69
- claude_mpm/commands/mpm-ticket-view.md +60 -495
- claude_mpm/commands/mpm-version.md +13 -107
- claude_mpm/commands/mpm.md +8 -0
- claude_mpm/config/agent_presets.py +8 -7
- claude_mpm/config/agent_sources.py +27 -0
- claude_mpm/config/skill_sources.py +16 -0
- claude_mpm/constants.py +1 -0
- claude_mpm/core/claude_runner.py +154 -2
- claude_mpm/core/config.py +37 -26
- claude_mpm/core/config_constants.py +74 -9
- claude_mpm/core/constants.py +56 -12
- claude_mpm/core/framework/formatters/content_formatter.py +3 -13
- claude_mpm/core/framework/loaders/agent_loader.py +8 -5
- claude_mpm/core/framework/loaders/instruction_loader.py +52 -11
- claude_mpm/core/framework_loader.py +4 -2
- claude_mpm/core/hook_manager.py +51 -3
- claude_mpm/core/interactive_session.py +12 -11
- claude_mpm/core/logger.py +39 -9
- claude_mpm/core/logging_utils.py +35 -11
- claude_mpm/core/network_config.py +148 -0
- claude_mpm/core/oneshot_session.py +7 -6
- claude_mpm/core/optimized_startup.py +61 -0
- claude_mpm/core/output_style_manager.py +219 -44
- claude_mpm/core/shared/config_loader.py +3 -1
- claude_mpm/core/socketio_pool.py +16 -8
- claude_mpm/core/unified_agent_registry.py +134 -16
- claude_mpm/core/unified_config.py +76 -8
- claude_mpm/core/unified_paths.py +95 -90
- claude_mpm/dashboard/static/svelte-build/_app/env.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/0.C33zOoyM.css +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/2.CW1J-YuA.css +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/1WZnGYqX.js +24 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/67pF3qNn.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/6RxdMKe4.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/8cZrfX0h.js +60 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/9a6T2nm-.js +7 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/B443AUzu.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/B8AwtY2H.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BF15LAsF.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BQaXIfA_.js +331 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BRcwIQNr.js +4 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BSNlmTZj.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BV6nKitt.js +43 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BViJ8lZt.js +128 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BcQ-Q0FE.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Bpyvgze_.js +30 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BzTRqg-z.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C0Fr8dve.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C3rbW_a-.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C8WYN38h.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/C9I8FlXH.js +61 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CIQcWgO2.js +36 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CIctN7YN.js +7 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CKrS_JZW.js +145 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CR6P9C4A.js +89 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CRRR9MD_.js +2 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CRcR2DqT.js +334 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CSXtMOf0.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CT-sbxSk.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CWm6DJsp.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CmKTTxBW.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CpqQ1Kzn.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Cu_Erd72.js +261 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/D2nGpDRe.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/D9iCMida.js +267 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/D9ykgMoY.js +10 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DL2Ldur1.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DPfltzjH.js +165 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DR8nis88.js +2 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DUliQN2b.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DVp1hx9R.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DXlhR01x.js +122 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/D_lyTybS.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DngoTTgh.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DqkmHtDC.js +220 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DsDh8EYs.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DypDmXgd.js +139 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Gi6I4Gst.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/IPYC-LnN.js +162 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/JTLiF7dt.js +24 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/JpevfAFt.js +68 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/NqQ1dWOy.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/R8CEIRAd.js +2 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/Zxy7qc-l.js +64 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/q9Hm6zAU.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/qtd3IeO4.js +15 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/ulBFON_C.js +65 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/wQVh1CoA.js +10 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/app.Dr7t0z2J.js +2 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/start.BGhZHUS3.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/0.RgBboRvH.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/1.DG-KkbDf.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/2.D_jnf-x6.js +1 -0
- claude_mpm/dashboard/static/svelte-build/_app/version.json +1 -0
- claude_mpm/dashboard/static/svelte-build/favicon.svg +7 -0
- claude_mpm/dashboard/static/svelte-build/index.html +36 -0
- claude_mpm/dashboard-svelte/node_modules/katex/src/fonts/generate_fonts.py +58 -0
- claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/extract_tfms.py +114 -0
- claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/extract_ttfs.py +122 -0
- claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/format_json.py +28 -0
- claude_mpm/dashboard-svelte/node_modules/katex/src/metrics/parse_tfm.py +211 -0
- claude_mpm/experimental/cli_enhancements.py +2 -1
- claude_mpm/hooks/claude_hooks/INTEGRATION_EXAMPLE.md +243 -0
- claude_mpm/hooks/claude_hooks/README_AUTO_PAUSE.md +403 -0
- claude_mpm/hooks/claude_hooks/__pycache__/__init__.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/auto_pause_handler.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/correlation_manager.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/event_handlers.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/hook_handler.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/memory_integration.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/response_tracking.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/tool_analysis.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/auto_pause_handler.py +485 -0
- claude_mpm/hooks/claude_hooks/correlation_manager.py +60 -0
- claude_mpm/hooks/claude_hooks/event_handlers.py +479 -128
- claude_mpm/hooks/claude_hooks/hook_handler.py +254 -83
- claude_mpm/hooks/claude_hooks/hook_wrapper.sh +6 -11
- claude_mpm/hooks/claude_hooks/installer.py +149 -18
- claude_mpm/hooks/claude_hooks/memory_integration.py +67 -19
- claude_mpm/hooks/claude_hooks/response_tracking.py +44 -62
- claude_mpm/hooks/claude_hooks/services/__pycache__/__init__.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/connection_manager_http.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/duplicate_detector.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/state_manager.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/subagent_processor.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/connection_manager.py +69 -30
- claude_mpm/hooks/claude_hooks/services/connection_manager_http.py +36 -103
- claude_mpm/hooks/claude_hooks/services/state_manager.py +23 -36
- claude_mpm/hooks/claude_hooks/services/subagent_processor.py +73 -75
- claude_mpm/hooks/kuzu_memory_hook.py +5 -5
- claude_mpm/hooks/memory_integration_hook.py +46 -1
- claude_mpm/hooks/session_resume_hook.py +89 -1
- claude_mpm/hooks/templates/pre_tool_use_template.py +10 -2
- claude_mpm/init.py +276 -19
- claude_mpm/models/agent_definition.py +7 -0
- claude_mpm/models/git_repository.py +3 -3
- claude_mpm/scripts/claude-hook-handler.sh +87 -20
- claude_mpm/scripts/launch_monitor.py +93 -13
- claude_mpm/scripts/start_activity_logging.py +0 -0
- claude_mpm/services/agents/agent_builder.py +3 -3
- claude_mpm/services/agents/agent_recommendation_service.py +278 -0
- claude_mpm/services/agents/agent_review_service.py +280 -0
- claude_mpm/services/agents/agent_selection_service.py +2 -2
- claude_mpm/services/agents/cache_git_manager.py +7 -7
- claude_mpm/services/agents/deployment/agent_deployment.py +29 -7
- claude_mpm/services/agents/deployment/agent_discovery_service.py +6 -5
- claude_mpm/services/agents/deployment/agent_format_converter.py +25 -13
- claude_mpm/services/agents/deployment/agent_template_builder.py +42 -20
- claude_mpm/services/agents/deployment/agents_directory_resolver.py +2 -2
- claude_mpm/services/agents/deployment/async_agent_deployment.py +31 -27
- claude_mpm/services/agents/deployment/deployment_reconciler.py +577 -0
- claude_mpm/services/agents/deployment/local_template_deployment.py +3 -1
- claude_mpm/services/agents/deployment/multi_source_deployment_service.py +348 -29
- claude_mpm/services/agents/deployment/remote_agent_discovery_service.py +570 -68
- claude_mpm/services/agents/deployment/startup_reconciliation.py +138 -0
- claude_mpm/services/agents/git_source_manager.py +57 -4
- claude_mpm/services/agents/loading/base_agent_manager.py +1 -13
- claude_mpm/services/agents/loading/framework_agent_loader.py +75 -2
- claude_mpm/services/agents/recommender.py +5 -3
- claude_mpm/services/agents/single_tier_deployment_service.py +6 -6
- claude_mpm/services/agents/sources/git_source_sync_service.py +129 -11
- claude_mpm/services/agents/startup_sync.py +27 -4
- claude_mpm/services/agents/toolchain_detector.py +10 -6
- claude_mpm/services/analysis/__init__.py +11 -1
- claude_mpm/services/analysis/clone_detector.py +1030 -0
- claude_mpm/services/cli/__init__.py +3 -0
- claude_mpm/services/cli/incremental_pause_manager.py +561 -0
- claude_mpm/services/cli/session_resume_helper.py +10 -2
- claude_mpm/services/command_deployment_service.py +81 -10
- claude_mpm/services/delegation_detector.py +175 -0
- claude_mpm/services/diagnostics/checks/agent_check.py +2 -2
- claude_mpm/services/diagnostics/checks/agent_sources_check.py +31 -1
- claude_mpm/services/diagnostics/checks/configuration_check.py +24 -0
- claude_mpm/services/diagnostics/checks/installation_check.py +22 -0
- claude_mpm/services/diagnostics/checks/mcp_services_check.py +23 -0
- claude_mpm/services/diagnostics/doctor_reporter.py +31 -1
- claude_mpm/services/diagnostics/models.py +14 -1
- claude_mpm/services/event_bus/config.py +3 -1
- claude_mpm/services/event_log.py +325 -0
- claude_mpm/services/git/git_operations_service.py +101 -16
- claude_mpm/services/infrastructure/__init__.py +4 -0
- claude_mpm/services/infrastructure/context_usage_tracker.py +291 -0
- claude_mpm/services/infrastructure/resume_log_generator.py +24 -5
- claude_mpm/services/monitor/daemon.py +9 -2
- claude_mpm/services/monitor/daemon_manager.py +54 -7
- claude_mpm/services/monitor/management/lifecycle.py +15 -3
- claude_mpm/services/monitor/server.py +796 -30
- claude_mpm/services/pm_skills_deployer.py +884 -0
- claude_mpm/services/profile_manager.py +337 -0
- claude_mpm/services/project/project_organizer.py +4 -0
- claude_mpm/services/self_upgrade_service.py +120 -12
- claude_mpm/services/skills/__init__.py +3 -0
- claude_mpm/services/skills/git_skill_source_manager.py +303 -12
- claude_mpm/services/skills/selective_skill_deployer.py +869 -0
- claude_mpm/services/skills/skill_discovery_service.py +74 -4
- claude_mpm/services/skills/skill_to_agent_mapper.py +406 -0
- claude_mpm/services/skills_deployer.py +294 -55
- claude_mpm/services/socketio/dashboard_server.py +1 -0
- claude_mpm/services/socketio/event_normalizer.py +51 -6
- claude_mpm/services/socketio/handlers/hook.py +14 -7
- claude_mpm/services/socketio/server/core.py +386 -108
- claude_mpm/services/socketio/server/main.py +12 -4
- claude_mpm/services/version_control/git_operations.py +103 -0
- claude_mpm/skills/__init__.py +2 -1
- claude_mpm/skills/bundled/collaboration/brainstorming/SKILL.md +79 -0
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/SKILL.md +178 -0
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/agent-prompts.md +577 -0
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/coordination-patterns.md +467 -0
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/examples.md +537 -0
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/troubleshooting.md +730 -0
- claude_mpm/skills/bundled/collaboration/git-worktrees.md +317 -0
- claude_mpm/skills/bundled/collaboration/requesting-code-review/SKILL.md +112 -0
- claude_mpm/skills/bundled/collaboration/requesting-code-review/references/code-reviewer-template.md +146 -0
- claude_mpm/skills/bundled/collaboration/requesting-code-review/references/review-examples.md +412 -0
- claude_mpm/skills/bundled/collaboration/stacked-prs.md +251 -0
- claude_mpm/skills/bundled/collaboration/writing-plans/SKILL.md +81 -0
- claude_mpm/skills/bundled/collaboration/writing-plans/references/best-practices.md +362 -0
- claude_mpm/skills/bundled/collaboration/writing-plans/references/plan-structure-templates.md +312 -0
- claude_mpm/skills/bundled/debugging/root-cause-tracing/SKILL.md +152 -0
- claude_mpm/skills/bundled/debugging/root-cause-tracing/references/advanced-techniques.md +668 -0
- claude_mpm/skills/bundled/debugging/root-cause-tracing/references/examples.md +587 -0
- claude_mpm/skills/bundled/debugging/root-cause-tracing/references/integration.md +438 -0
- claude_mpm/skills/bundled/debugging/root-cause-tracing/references/tracing-techniques.md +391 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/CREATION-LOG.md +119 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/SKILL.md +148 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/references/anti-patterns.md +483 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/references/examples.md +452 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/references/troubleshooting.md +449 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/references/workflow.md +411 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/test-academic.md +14 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-1.md +58 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-2.md +68 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-3.md +69 -0
- claude_mpm/skills/bundled/debugging/verification-before-completion/SKILL.md +131 -0
- claude_mpm/skills/bundled/debugging/verification-before-completion/references/gate-function.md +325 -0
- claude_mpm/skills/bundled/debugging/verification-before-completion/references/integration-and-workflows.md +490 -0
- claude_mpm/skills/bundled/debugging/verification-before-completion/references/red-flags-and-failures.md +425 -0
- claude_mpm/skills/bundled/debugging/verification-before-completion/references/verification-patterns.md +499 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/INTEGRATION.md +611 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/README.md +596 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/SKILL.md +260 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/examples/nextjs-env-structure.md +315 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/references/frameworks.md +436 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/references/security.md +433 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/references/synchronization.md +452 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/references/troubleshooting.md +404 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/references/validation.md +420 -0
- claude_mpm/skills/bundled/main/artifacts-builder/SKILL.md +86 -0
- claude_mpm/skills/bundled/main/internal-comms/SKILL.md +43 -0
- claude_mpm/skills/bundled/main/internal-comms/examples/3p-updates.md +47 -0
- claude_mpm/skills/bundled/main/internal-comms/examples/company-newsletter.md +65 -0
- claude_mpm/skills/bundled/main/internal-comms/examples/faq-answers.md +30 -0
- claude_mpm/skills/bundled/main/internal-comms/examples/general-comms.md +16 -0
- claude_mpm/skills/bundled/main/mcp-builder/SKILL.md +160 -0
- claude_mpm/skills/bundled/main/mcp-builder/reference/design_principles.md +412 -0
- claude_mpm/skills/bundled/main/mcp-builder/reference/evaluation.md +602 -0
- claude_mpm/skills/bundled/main/mcp-builder/reference/mcp_best_practices.md +915 -0
- claude_mpm/skills/bundled/main/mcp-builder/reference/node_mcp_server.md +916 -0
- claude_mpm/skills/bundled/main/mcp-builder/reference/python_mcp_server.md +752 -0
- claude_mpm/skills/bundled/main/mcp-builder/reference/workflow.md +1237 -0
- claude_mpm/skills/bundled/main/skill-creator/SKILL.md +189 -0
- claude_mpm/skills/bundled/main/skill-creator/references/best-practices.md +500 -0
- claude_mpm/skills/bundled/main/skill-creator/references/creation-workflow.md +464 -0
- claude_mpm/skills/bundled/main/skill-creator/references/examples.md +619 -0
- claude_mpm/skills/bundled/main/skill-creator/references/progressive-disclosure.md +437 -0
- claude_mpm/skills/bundled/main/skill-creator/references/skill-structure.md +231 -0
- claude_mpm/skills/bundled/php/espocrm-development/SKILL.md +170 -0
- claude_mpm/skills/bundled/php/espocrm-development/references/architecture.md +602 -0
- claude_mpm/skills/bundled/php/espocrm-development/references/common-tasks.md +821 -0
- claude_mpm/skills/bundled/php/espocrm-development/references/development-workflow.md +742 -0
- claude_mpm/skills/bundled/php/espocrm-development/references/frontend-customization.md +726 -0
- claude_mpm/skills/bundled/php/espocrm-development/references/hooks-and-services.md +764 -0
- claude_mpm/skills/bundled/php/espocrm-development/references/testing-debugging.md +831 -0
- claude_mpm/skills/bundled/pm/mpm/SKILL.md +38 -0
- claude_mpm/skills/bundled/pm/mpm-agent-update-workflow/SKILL.md +75 -0
- claude_mpm/skills/bundled/pm/mpm-bug-reporting/SKILL.md +248 -0
- claude_mpm/skills/bundled/pm/mpm-circuit-breaker-enforcement/SKILL.md +476 -0
- claude_mpm/skills/bundled/pm/mpm-config/SKILL.md +29 -0
- claude_mpm/skills/bundled/pm/mpm-delegation-patterns/SKILL.md +167 -0
- claude_mpm/skills/bundled/pm/mpm-doctor/SKILL.md +53 -0
- claude_mpm/skills/bundled/pm/mpm-git-file-tracking/SKILL.md +113 -0
- claude_mpm/skills/bundled/pm/mpm-help/SKILL.md +35 -0
- claude_mpm/skills/bundled/pm/mpm-init/SKILL.md +125 -0
- claude_mpm/skills/bundled/pm/mpm-monitor/SKILL.md +32 -0
- claude_mpm/skills/bundled/pm/mpm-organize/SKILL.md +121 -0
- claude_mpm/skills/bundled/pm/mpm-postmortem/SKILL.md +22 -0
- claude_mpm/skills/bundled/pm/mpm-pr-workflow/SKILL.md +124 -0
- claude_mpm/skills/bundled/pm/mpm-session-management/SKILL.md +312 -0
- claude_mpm/skills/bundled/pm/mpm-session-pause/SKILL.md +170 -0
- claude_mpm/skills/bundled/pm/mpm-session-resume/SKILL.md +31 -0
- claude_mpm/skills/bundled/pm/mpm-status/SKILL.md +37 -0
- claude_mpm/skills/bundled/pm/mpm-teaching-mode/SKILL.md +657 -0
- claude_mpm/skills/bundled/pm/mpm-ticket-view/SKILL.md +110 -0
- claude_mpm/skills/bundled/pm/mpm-ticketing-integration/SKILL.md +154 -0
- claude_mpm/skills/bundled/pm/mpm-tool-usage-guide/SKILL.md +386 -0
- claude_mpm/skills/bundled/pm/mpm-verification-protocols/SKILL.md +198 -0
- claude_mpm/skills/bundled/pm/mpm-version/SKILL.md +21 -0
- claude_mpm/skills/bundled/react/flexlayout-react.md +742 -0
- claude_mpm/skills/bundled/rust/desktop-applications/SKILL.md +226 -0
- claude_mpm/skills/bundled/rust/desktop-applications/references/architecture-patterns.md +901 -0
- claude_mpm/skills/bundled/rust/desktop-applications/references/native-gui-frameworks.md +901 -0
- claude_mpm/skills/bundled/rust/desktop-applications/references/platform-integration.md +775 -0
- claude_mpm/skills/bundled/rust/desktop-applications/references/state-management.md +937 -0
- claude_mpm/skills/bundled/rust/desktop-applications/references/tauri-framework.md +770 -0
- claude_mpm/skills/bundled/rust/desktop-applications/references/testing-deployment.md +961 -0
- claude_mpm/skills/bundled/security-scanning.md +112 -0
- claude_mpm/skills/bundled/tauri/tauri-async-patterns.md +495 -0
- claude_mpm/skills/bundled/tauri/tauri-build-deploy.md +599 -0
- claude_mpm/skills/bundled/tauri/tauri-command-patterns.md +535 -0
- claude_mpm/skills/bundled/tauri/tauri-error-handling.md +613 -0
- claude_mpm/skills/bundled/tauri/tauri-event-system.md +648 -0
- claude_mpm/skills/bundled/tauri/tauri-file-system.md +673 -0
- claude_mpm/skills/bundled/tauri/tauri-frontend-integration.md +767 -0
- claude_mpm/skills/bundled/tauri/tauri-performance.md +669 -0
- claude_mpm/skills/bundled/tauri/tauri-state-management.md +573 -0
- claude_mpm/skills/bundled/tauri/tauri-testing.md +384 -0
- claude_mpm/skills/bundled/tauri/tauri-window-management.md +628 -0
- claude_mpm/skills/bundled/testing/condition-based-waiting/SKILL.md +119 -0
- claude_mpm/skills/bundled/testing/condition-based-waiting/references/patterns-and-implementation.md +253 -0
- claude_mpm/skills/bundled/testing/test-driven-development/SKILL.md +145 -0
- claude_mpm/skills/bundled/testing/test-driven-development/references/anti-patterns.md +543 -0
- claude_mpm/skills/bundled/testing/test-driven-development/references/examples.md +741 -0
- claude_mpm/skills/bundled/testing/test-driven-development/references/integration.md +470 -0
- claude_mpm/skills/bundled/testing/test-driven-development/references/philosophy.md +458 -0
- claude_mpm/skills/bundled/testing/test-driven-development/references/workflow.md +639 -0
- claude_mpm/skills/bundled/testing/test-quality-inspector/SKILL.md +458 -0
- claude_mpm/skills/bundled/testing/test-quality-inspector/examples/example-inspection-report.md +411 -0
- claude_mpm/skills/bundled/testing/test-quality-inspector/references/assertion-quality.md +317 -0
- claude_mpm/skills/bundled/testing/test-quality-inspector/references/inspection-checklist.md +270 -0
- claude_mpm/skills/bundled/testing/test-quality-inspector/references/red-flags.md +436 -0
- claude_mpm/skills/bundled/testing/testing-anti-patterns/SKILL.md +140 -0
- claude_mpm/skills/bundled/testing/testing-anti-patterns/references/completeness-anti-patterns.md +572 -0
- claude_mpm/skills/bundled/testing/testing-anti-patterns/references/core-anti-patterns.md +411 -0
- claude_mpm/skills/bundled/testing/testing-anti-patterns/references/detection-guide.md +569 -0
- claude_mpm/skills/bundled/testing/testing-anti-patterns/references/tdd-connection.md +695 -0
- claude_mpm/skills/bundled/testing/webapp-testing/SKILL.md +184 -0
- claude_mpm/skills/bundled/testing/webapp-testing/decision-tree.md +459 -0
- claude_mpm/skills/bundled/testing/webapp-testing/playwright-patterns.md +479 -0
- claude_mpm/skills/bundled/testing/webapp-testing/reconnaissance-pattern.md +687 -0
- claude_mpm/skills/bundled/testing/webapp-testing/server-management.md +758 -0
- claude_mpm/skills/bundled/testing/webapp-testing/troubleshooting.md +868 -0
- claude_mpm/skills/registry.py +295 -90
- claude_mpm/skills/skill_manager.py +98 -3
- claude_mpm/templates/.pre-commit-config.yaml +112 -0
- claude_mpm/utils/agent_dependency_loader.py +115 -4
- claude_mpm/utils/agent_filters.py +17 -44
- claude_mpm/utils/gitignore.py +3 -0
- claude_mpm/utils/migration.py +4 -4
- claude_mpm/utils/robust_installer.py +86 -21
- claude_mpm-5.6.23.dist-info/METADATA +393 -0
- {claude_mpm-5.0.9.dist-info → claude_mpm-5.6.23.dist-info}/RECORD +508 -261
- claude_mpm-5.6.23.dist-info/entry_points.txt +5 -0
- claude_mpm-5.6.23.dist-info/licenses/LICENSE +94 -0
- claude_mpm-5.6.23.dist-info/licenses/LICENSE-FAQ.md +153 -0
- claude_mpm/agents/BASE_AGENT_TEMPLATE.md +0 -292
- claude_mpm/agents/BASE_DOCUMENTATION.md +0 -53
- claude_mpm/agents/BASE_OPS.md +0 -219
- claude_mpm/agents/BASE_PM.md +0 -480
- claude_mpm/agents/BASE_PROMPT_ENGINEER.md +0 -787
- claude_mpm/agents/BASE_QA.md +0 -167
- claude_mpm/agents/BASE_RESEARCH.md +0 -53
- claude_mpm/agents/OUTPUT_STYLE.md +0 -290
- claude_mpm/agents/PM_INSTRUCTIONS_TEACH.md +0 -1322
- claude_mpm/agents/base_agent_loader.py +0 -601
- claude_mpm/cli/commands/agents_detect.py +0 -380
- claude_mpm/cli/commands/agents_recommend.py +0 -309
- claude_mpm/cli/ticket_cli.py +0 -35
- claude_mpm/commands/mpm-agents-auto-configure.md +0 -278
- claude_mpm/commands/mpm-agents-detect.md +0 -177
- claude_mpm/commands/mpm-agents-list.md +0 -131
- claude_mpm/commands/mpm-agents-recommend.md +0 -223
- claude_mpm/commands/mpm-config-view.md +0 -150
- claude_mpm/commands/mpm-ticket-organize.md +0 -304
- claude_mpm/dashboard/analysis_runner.py +0 -455
- claude_mpm/dashboard/index.html +0 -13
- claude_mpm/dashboard/open_dashboard.py +0 -66
- claude_mpm/dashboard/static/css/activity.css +0 -1958
- claude_mpm/dashboard/static/css/connection-status.css +0 -370
- claude_mpm/dashboard/static/css/dashboard.css +0 -4701
- claude_mpm/dashboard/static/js/components/activity-tree.js +0 -1871
- claude_mpm/dashboard/static/js/components/agent-hierarchy.js +0 -777
- claude_mpm/dashboard/static/js/components/agent-inference.js +0 -956
- claude_mpm/dashboard/static/js/components/build-tracker.js +0 -333
- claude_mpm/dashboard/static/js/components/code-simple.js +0 -857
- claude_mpm/dashboard/static/js/components/connection-debug.js +0 -654
- claude_mpm/dashboard/static/js/components/diff-viewer.js +0 -891
- claude_mpm/dashboard/static/js/components/event-processor.js +0 -542
- claude_mpm/dashboard/static/js/components/event-viewer.js +0 -1155
- claude_mpm/dashboard/static/js/components/export-manager.js +0 -368
- claude_mpm/dashboard/static/js/components/file-change-tracker.js +0 -443
- claude_mpm/dashboard/static/js/components/file-change-viewer.js +0 -690
- claude_mpm/dashboard/static/js/components/file-tool-tracker.js +0 -724
- claude_mpm/dashboard/static/js/components/file-viewer.js +0 -580
- claude_mpm/dashboard/static/js/components/hud-library-loader.js +0 -211
- claude_mpm/dashboard/static/js/components/hud-manager.js +0 -671
- claude_mpm/dashboard/static/js/components/hud-visualizer.js +0 -1718
- claude_mpm/dashboard/static/js/components/module-viewer.js +0 -2764
- claude_mpm/dashboard/static/js/components/session-manager.js +0 -579
- claude_mpm/dashboard/static/js/components/socket-manager.js +0 -368
- claude_mpm/dashboard/static/js/components/ui-state-manager.js +0 -749
- claude_mpm/dashboard/static/js/components/unified-data-viewer.js +0 -1824
- claude_mpm/dashboard/static/js/components/working-directory.js +0 -920
- claude_mpm/dashboard/static/js/connection-manager.js +0 -536
- claude_mpm/dashboard/static/js/dashboard.js +0 -1914
- claude_mpm/dashboard/static/js/extension-error-handler.js +0 -164
- claude_mpm/dashboard/static/js/socket-client.js +0 -1474
- claude_mpm/dashboard/static/js/tab-isolation-fix.js +0 -185
- claude_mpm/dashboard/static/socket.io.min.js +0 -7
- claude_mpm/dashboard/static/socket.io.v4.8.1.backup.js +0 -7
- claude_mpm/dashboard/templates/code_simple.html +0 -153
- claude_mpm/dashboard/templates/index.html +0 -606
- claude_mpm/dashboard/test_dashboard.html +0 -372
- claude_mpm/hooks/claude_hooks/__pycache__/__init__.cpython-313.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/event_handlers.cpython-313.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/hook_handler.cpython-313.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/memory_integration.cpython-313.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/response_tracking.cpython-313.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/tool_analysis.cpython-313.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/__init__.cpython-313.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/connection_manager_http.cpython-313.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/duplicate_detector.cpython-313.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/state_manager.cpython-313.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/subagent_processor.cpython-313.pyc +0 -0
- claude_mpm/scripts/mcp_server.py +0 -75
- claude_mpm/scripts/mcp_wrapper.py +0 -39
- claude_mpm/services/mcp_gateway/__init__.py +0 -159
- claude_mpm/services/mcp_gateway/auto_configure.py +0 -369
- claude_mpm/services/mcp_gateway/config/__init__.py +0 -17
- claude_mpm/services/mcp_gateway/config/config_loader.py +0 -296
- claude_mpm/services/mcp_gateway/config/config_schema.py +0 -243
- claude_mpm/services/mcp_gateway/config/configuration.py +0 -429
- claude_mpm/services/mcp_gateway/core/__init__.py +0 -43
- claude_mpm/services/mcp_gateway/core/base.py +0 -312
- claude_mpm/services/mcp_gateway/core/exceptions.py +0 -253
- claude_mpm/services/mcp_gateway/core/interfaces.py +0 -443
- claude_mpm/services/mcp_gateway/core/process_pool.py +0 -977
- claude_mpm/services/mcp_gateway/core/singleton_manager.py +0 -315
- claude_mpm/services/mcp_gateway/core/startup_verification.py +0 -316
- claude_mpm/services/mcp_gateway/main.py +0 -589
- claude_mpm/services/mcp_gateway/registry/__init__.py +0 -12
- claude_mpm/services/mcp_gateway/registry/service_registry.py +0 -412
- claude_mpm/services/mcp_gateway/registry/tool_registry.py +0 -489
- claude_mpm/services/mcp_gateway/server/__init__.py +0 -15
- claude_mpm/services/mcp_gateway/server/mcp_gateway.py +0 -414
- claude_mpm/services/mcp_gateway/server/stdio_handler.py +0 -372
- claude_mpm/services/mcp_gateway/server/stdio_server.py +0 -712
- claude_mpm/services/mcp_gateway/tools/__init__.py +0 -36
- claude_mpm/services/mcp_gateway/tools/base_adapter.py +0 -485
- claude_mpm/services/mcp_gateway/tools/document_summarizer.py +0 -789
- claude_mpm/services/mcp_gateway/tools/external_mcp_services.py +0 -654
- claude_mpm/services/mcp_gateway/tools/health_check_tool.py +0 -456
- claude_mpm/services/mcp_gateway/tools/hello_world.py +0 -551
- claude_mpm/services/mcp_gateway/tools/kuzu_memory_service.py +0 -555
- claude_mpm/services/mcp_gateway/utils/__init__.py +0 -14
- claude_mpm/services/mcp_gateway/utils/package_version_checker.py +0 -160
- claude_mpm/services/mcp_gateway/utils/update_preferences.py +0 -170
- claude_mpm-5.0.9.dist-info/METADATA +0 -1028
- claude_mpm-5.0.9.dist-info/entry_points.txt +0 -10
- claude_mpm-5.0.9.dist-info/licenses/LICENSE +0 -21
- {claude_mpm-5.0.9.dist-info → claude_mpm-5.6.23.dist-info}/WHEEL +0 -0
- {claude_mpm-5.0.9.dist-info → claude_mpm-5.6.23.dist-info}/top_level.txt +0 -0
claude_mpm/init.py
CHANGED
|
@@ -141,25 +141,6 @@ class ProjectInitializer:
|
|
|
141
141
|
if not gitignore.exists():
|
|
142
142
|
gitignore.write_text("logs/\n*.log\n*.pyc\n__pycache__/\n")
|
|
143
143
|
|
|
144
|
-
# Also ensure MCP directories are in main project .gitignore
|
|
145
|
-
try:
|
|
146
|
-
from claude_mpm.services.project.project_organizer import (
|
|
147
|
-
ProjectOrganizer,
|
|
148
|
-
)
|
|
149
|
-
|
|
150
|
-
# Check if we're in a git repository
|
|
151
|
-
if (project_root / ".git").exists():
|
|
152
|
-
organizer = ProjectOrganizer(project_root)
|
|
153
|
-
# This will add MCP directories and other standard patterns
|
|
154
|
-
organizer.update_gitignore()
|
|
155
|
-
self.logger.debug(
|
|
156
|
-
"Updated project .gitignore with MCP and standard patterns"
|
|
157
|
-
)
|
|
158
|
-
except Exception as e:
|
|
159
|
-
self.logger.debug(
|
|
160
|
-
f"Could not update project gitignore with MCP patterns: {e}"
|
|
161
|
-
)
|
|
162
|
-
|
|
163
144
|
# Log successful creation with details
|
|
164
145
|
self.logger.info(f"Initialized project directory at {self.project_dir}")
|
|
165
146
|
self.logger.debug("Created directories: agents, config, responses, logs")
|
|
@@ -182,6 +163,15 @@ class ProjectInitializer:
|
|
|
182
163
|
f"✓ Found {agent_count} project agent(s) in .claude-mpm/agents/"
|
|
183
164
|
)
|
|
184
165
|
|
|
166
|
+
# Verify and deploy PM skills (non-blocking)
|
|
167
|
+
self._verify_and_deploy_pm_skills(project_root, is_mcp_mode)
|
|
168
|
+
|
|
169
|
+
# Setup security hooks (auto-install pre-commit, detect-secrets)
|
|
170
|
+
self._setup_security_hooks(project_root, is_mcp_mode)
|
|
171
|
+
|
|
172
|
+
# Perform security checks (non-blocking)
|
|
173
|
+
self._check_security_risks(project_root, is_mcp_mode)
|
|
174
|
+
|
|
185
175
|
return True
|
|
186
176
|
|
|
187
177
|
except Exception as e:
|
|
@@ -189,6 +179,68 @@ class ProjectInitializer:
|
|
|
189
179
|
print(f"✗ Failed to create .claude-mpm/ directory: {e}")
|
|
190
180
|
return False
|
|
191
181
|
|
|
182
|
+
def _verify_and_deploy_pm_skills(
|
|
183
|
+
self, project_root: Path, is_mcp_mode: bool = False
|
|
184
|
+
) -> None:
|
|
185
|
+
"""Verify PM skills are deployed and auto-deploy if missing.
|
|
186
|
+
|
|
187
|
+
Non-blocking operation that gracefully handles errors.
|
|
188
|
+
|
|
189
|
+
Args:
|
|
190
|
+
project_root: Project root directory
|
|
191
|
+
is_mcp_mode: Whether running in MCP mode (suppress console output)
|
|
192
|
+
"""
|
|
193
|
+
try:
|
|
194
|
+
from claude_mpm.services.pm_skills_deployer import PMSkillsDeployerService
|
|
195
|
+
|
|
196
|
+
deployer = PMSkillsDeployerService()
|
|
197
|
+
result = deployer.verify_pm_skills(project_root)
|
|
198
|
+
|
|
199
|
+
if not result.verified:
|
|
200
|
+
# Log warnings
|
|
201
|
+
for warning in result.warnings:
|
|
202
|
+
self.logger.warning(warning)
|
|
203
|
+
|
|
204
|
+
# Auto-deploy PM skills
|
|
205
|
+
self.logger.info("Auto-deploying PM skills...")
|
|
206
|
+
deploy_result = deployer.deploy_pm_skills(project_root)
|
|
207
|
+
|
|
208
|
+
if deploy_result.success:
|
|
209
|
+
self.logger.info(
|
|
210
|
+
f"PM skills deployed: {len(deploy_result.deployed)} deployed, "
|
|
211
|
+
f"{len(deploy_result.skipped)} skipped"
|
|
212
|
+
)
|
|
213
|
+
|
|
214
|
+
# Print to console if not in MCP mode
|
|
215
|
+
if not is_mcp_mode:
|
|
216
|
+
if deploy_result.deployed:
|
|
217
|
+
print(
|
|
218
|
+
f"✓ Deployed {len(deploy_result.deployed)} PM skill(s) "
|
|
219
|
+
f"to .claude/skills/"
|
|
220
|
+
)
|
|
221
|
+
else:
|
|
222
|
+
self.logger.warning(
|
|
223
|
+
f"PM skills deployment had errors: {len(deploy_result.errors)}"
|
|
224
|
+
)
|
|
225
|
+
if not is_mcp_mode and deploy_result.errors:
|
|
226
|
+
print(
|
|
227
|
+
f"⚠ PM skills deployment had {len(deploy_result.errors)} error(s)"
|
|
228
|
+
)
|
|
229
|
+
else:
|
|
230
|
+
# Skills verified successfully
|
|
231
|
+
registry = deployer._load_registry(project_root)
|
|
232
|
+
skill_count = len(registry.get("skills", []))
|
|
233
|
+
self.logger.debug(f"PM skills verified: {skill_count} skills")
|
|
234
|
+
|
|
235
|
+
if not is_mcp_mode and skill_count > 0:
|
|
236
|
+
print(f"✓ Verified {skill_count} PM skill(s)")
|
|
237
|
+
|
|
238
|
+
except ImportError:
|
|
239
|
+
self.logger.debug("PM skills deployer not available")
|
|
240
|
+
except Exception as e:
|
|
241
|
+
self.logger.warning(f"PM skills verification failed: {e}")
|
|
242
|
+
# Don't print to console - this is a non-critical failure
|
|
243
|
+
|
|
192
244
|
def _migrate_project_agents(self):
|
|
193
245
|
"""Migrate agents from old subdirectory structure to direct agents directory.
|
|
194
246
|
|
|
@@ -338,6 +390,211 @@ class ProjectInitializer:
|
|
|
338
390
|
if not dst_file.exists():
|
|
339
391
|
shutil.copy2(template_file, dst_file)
|
|
340
392
|
|
|
393
|
+
def _setup_security_hooks(
|
|
394
|
+
self, project_root: Path, is_mcp_mode: bool = False
|
|
395
|
+
) -> None:
|
|
396
|
+
"""Automatically install pre-commit hooks for secret scanning.
|
|
397
|
+
|
|
398
|
+
This method:
|
|
399
|
+
1. Installs pre-commit and detect-secrets if missing
|
|
400
|
+
2. Copies .pre-commit-config.yaml to project root
|
|
401
|
+
3. Runs pre-commit install to set up git hooks
|
|
402
|
+
4. Creates .secrets.baseline for detect-secrets
|
|
403
|
+
|
|
404
|
+
Args:
|
|
405
|
+
project_root: Project root directory
|
|
406
|
+
is_mcp_mode: Whether running in MCP mode (suppress console output)
|
|
407
|
+
"""
|
|
408
|
+
try:
|
|
409
|
+
import subprocess
|
|
410
|
+
|
|
411
|
+
# Only set up hooks if this is a git repository
|
|
412
|
+
if not (project_root / ".git").exists():
|
|
413
|
+
self.logger.debug("Not a git repository, skipping security hooks setup")
|
|
414
|
+
return
|
|
415
|
+
|
|
416
|
+
# Check/install pre-commit
|
|
417
|
+
try:
|
|
418
|
+
subprocess.run(
|
|
419
|
+
["pre-commit", "--version"],
|
|
420
|
+
capture_output=True,
|
|
421
|
+
text=True,
|
|
422
|
+
timeout=2,
|
|
423
|
+
check=True,
|
|
424
|
+
)
|
|
425
|
+
except (
|
|
426
|
+
subprocess.CalledProcessError,
|
|
427
|
+
subprocess.TimeoutExpired,
|
|
428
|
+
FileNotFoundError,
|
|
429
|
+
):
|
|
430
|
+
self.logger.info("Installing pre-commit...")
|
|
431
|
+
try:
|
|
432
|
+
subprocess.run(
|
|
433
|
+
[sys.executable, "-m", "pip", "install", "pre-commit"],
|
|
434
|
+
capture_output=True,
|
|
435
|
+
text=True,
|
|
436
|
+
timeout=60,
|
|
437
|
+
check=True,
|
|
438
|
+
)
|
|
439
|
+
self.logger.info("pre-commit installed successfully")
|
|
440
|
+
except subprocess.CalledProcessError as e:
|
|
441
|
+
self.logger.warning(f"Failed to install pre-commit: {e}")
|
|
442
|
+
return
|
|
443
|
+
|
|
444
|
+
# Check/install detect-secrets
|
|
445
|
+
try:
|
|
446
|
+
subprocess.run(
|
|
447
|
+
["detect-secrets", "--version"],
|
|
448
|
+
capture_output=True,
|
|
449
|
+
text=True,
|
|
450
|
+
timeout=2,
|
|
451
|
+
check=True,
|
|
452
|
+
)
|
|
453
|
+
except (
|
|
454
|
+
subprocess.CalledProcessError,
|
|
455
|
+
subprocess.TimeoutExpired,
|
|
456
|
+
FileNotFoundError,
|
|
457
|
+
):
|
|
458
|
+
self.logger.info("Installing detect-secrets...")
|
|
459
|
+
try:
|
|
460
|
+
subprocess.run(
|
|
461
|
+
[sys.executable, "-m", "pip", "install", "detect-secrets"],
|
|
462
|
+
capture_output=True,
|
|
463
|
+
text=True,
|
|
464
|
+
timeout=60,
|
|
465
|
+
check=True,
|
|
466
|
+
)
|
|
467
|
+
self.logger.info("detect-secrets installed successfully")
|
|
468
|
+
except subprocess.CalledProcessError as e:
|
|
469
|
+
self.logger.warning(f"Failed to install detect-secrets: {e}")
|
|
470
|
+
return
|
|
471
|
+
|
|
472
|
+
# Copy .pre-commit-config.yaml to project root if it doesn't exist
|
|
473
|
+
precommit_config = project_root / ".pre-commit-config.yaml"
|
|
474
|
+
if not precommit_config.exists():
|
|
475
|
+
template_dir = Path(__file__).parent / "templates"
|
|
476
|
+
template_config = template_dir / ".pre-commit-config.yaml"
|
|
477
|
+
|
|
478
|
+
if template_config.exists():
|
|
479
|
+
shutil.copy2(template_config, precommit_config)
|
|
480
|
+
self.logger.info("Copied .pre-commit-config.yaml to project root")
|
|
481
|
+
else:
|
|
482
|
+
self.logger.warning("Template .pre-commit-config.yaml not found")
|
|
483
|
+
return
|
|
484
|
+
|
|
485
|
+
# Create .secrets.baseline if it doesn't exist
|
|
486
|
+
secrets_baseline = project_root / ".secrets.baseline"
|
|
487
|
+
if not secrets_baseline.exists():
|
|
488
|
+
try:
|
|
489
|
+
subprocess.run(
|
|
490
|
+
["detect-secrets", "scan", "--baseline", ".secrets.baseline"],
|
|
491
|
+
cwd=str(project_root),
|
|
492
|
+
capture_output=True,
|
|
493
|
+
text=True,
|
|
494
|
+
timeout=30,
|
|
495
|
+
check=True,
|
|
496
|
+
)
|
|
497
|
+
self.logger.info("Created .secrets.baseline")
|
|
498
|
+
except subprocess.CalledProcessError as e:
|
|
499
|
+
self.logger.warning(f"Failed to create .secrets.baseline: {e}")
|
|
500
|
+
|
|
501
|
+
# Install git hooks
|
|
502
|
+
try:
|
|
503
|
+
subprocess.run(
|
|
504
|
+
["pre-commit", "install"],
|
|
505
|
+
cwd=str(project_root),
|
|
506
|
+
capture_output=True,
|
|
507
|
+
text=True,
|
|
508
|
+
timeout=30,
|
|
509
|
+
check=True,
|
|
510
|
+
)
|
|
511
|
+
self.logger.info("Pre-commit hooks installed in git repository")
|
|
512
|
+
|
|
513
|
+
if not is_mcp_mode:
|
|
514
|
+
print("✓ Security hooks installed (pre-commit + detect-secrets)")
|
|
515
|
+
|
|
516
|
+
except subprocess.CalledProcessError as e:
|
|
517
|
+
self.logger.warning(f"Failed to install pre-commit hooks: {e}")
|
|
518
|
+
|
|
519
|
+
except Exception as e:
|
|
520
|
+
self.logger.debug(f"Security hooks setup failed: {e}")
|
|
521
|
+
# Don't print to console - this is a non-critical failure
|
|
522
|
+
|
|
523
|
+
def _check_security_risks(
|
|
524
|
+
self, project_root: Path, is_mcp_mode: bool = False
|
|
525
|
+
) -> None:
|
|
526
|
+
"""Check for potential security risks like exposed config files.
|
|
527
|
+
|
|
528
|
+
Non-blocking operation that warns about security issues.
|
|
529
|
+
|
|
530
|
+
Args:
|
|
531
|
+
project_root: Project root directory
|
|
532
|
+
is_mcp_mode: Whether running in MCP mode (suppress console output)
|
|
533
|
+
"""
|
|
534
|
+
try:
|
|
535
|
+
import subprocess
|
|
536
|
+
|
|
537
|
+
security_issues = []
|
|
538
|
+
|
|
539
|
+
# Common secret file patterns to check
|
|
540
|
+
secret_patterns = [
|
|
541
|
+
".mcp-vector-search/config.json",
|
|
542
|
+
".mcp/config.json",
|
|
543
|
+
"openrouter.json",
|
|
544
|
+
"anthropic-config.json",
|
|
545
|
+
"credentials.json",
|
|
546
|
+
"secrets.json",
|
|
547
|
+
"api-keys.json",
|
|
548
|
+
]
|
|
549
|
+
|
|
550
|
+
for pattern in secret_patterns:
|
|
551
|
+
file_path = project_root / pattern
|
|
552
|
+
if file_path.exists():
|
|
553
|
+
# Check if file is tracked by git
|
|
554
|
+
try:
|
|
555
|
+
result = subprocess.run(
|
|
556
|
+
["git", "ls-files", str(file_path)],
|
|
557
|
+
check=False,
|
|
558
|
+
cwd=str(project_root),
|
|
559
|
+
capture_output=True,
|
|
560
|
+
text=True,
|
|
561
|
+
timeout=2,
|
|
562
|
+
)
|
|
563
|
+
if result.stdout.strip():
|
|
564
|
+
security_issues.append(
|
|
565
|
+
f"⚠️ SECURITY: {pattern} is tracked by git (may contain secrets)"
|
|
566
|
+
)
|
|
567
|
+
except (subprocess.TimeoutExpired, FileNotFoundError):
|
|
568
|
+
pass
|
|
569
|
+
|
|
570
|
+
# Check if file is ignored by .gitignore
|
|
571
|
+
try:
|
|
572
|
+
result = subprocess.run(
|
|
573
|
+
["git", "check-ignore", str(file_path)],
|
|
574
|
+
check=False,
|
|
575
|
+
cwd=str(project_root),
|
|
576
|
+
capture_output=True,
|
|
577
|
+
text=True,
|
|
578
|
+
timeout=2,
|
|
579
|
+
)
|
|
580
|
+
if result.returncode != 0: # File NOT ignored
|
|
581
|
+
security_issues.append(
|
|
582
|
+
f"⚠️ WARNING: {pattern} exists but not in .gitignore"
|
|
583
|
+
)
|
|
584
|
+
except (subprocess.TimeoutExpired, FileNotFoundError):
|
|
585
|
+
pass
|
|
586
|
+
|
|
587
|
+
# Print security warnings if not in MCP mode
|
|
588
|
+
if security_issues and not is_mcp_mode:
|
|
589
|
+
print("\n🔒 Security Check:")
|
|
590
|
+
for issue in security_issues:
|
|
591
|
+
print(f" {issue}")
|
|
592
|
+
print()
|
|
593
|
+
|
|
594
|
+
except Exception as e:
|
|
595
|
+
self.logger.debug(f"Security check failed: {e}")
|
|
596
|
+
# Don't print to console - this is a non-critical failure
|
|
597
|
+
|
|
341
598
|
def validate_dependencies(self) -> Dict[str, bool]:
|
|
342
599
|
"""Validate that all required dependencies are available."""
|
|
343
600
|
dependencies = {}
|
|
@@ -103,6 +103,10 @@ class AgentMetadata:
|
|
|
103
103
|
author: Optional[str] = None
|
|
104
104
|
tags: List[str] = field(default_factory=list)
|
|
105
105
|
specializations: List[str] = field(default_factory=list)
|
|
106
|
+
# NEW: Collection metadata for enhanced agent matching
|
|
107
|
+
collection_id: Optional[str] = None # Format: owner/repo-name
|
|
108
|
+
source_path: Optional[str] = None # Relative path in repository
|
|
109
|
+
canonical_id: Optional[str] = None # Format: collection_id:agent_id
|
|
106
110
|
|
|
107
111
|
def increment_serial_version(self) -> None:
|
|
108
112
|
"""Increment the patch version number.
|
|
@@ -181,6 +185,9 @@ class AgentDefinition:
|
|
|
181
185
|
"author": self.metadata.author,
|
|
182
186
|
"tags": self.metadata.tags,
|
|
183
187
|
"specializations": self.metadata.specializations,
|
|
188
|
+
"collection_id": self.metadata.collection_id,
|
|
189
|
+
"source_path": self.metadata.source_path,
|
|
190
|
+
"canonical_id": self.metadata.canonical_id,
|
|
184
191
|
},
|
|
185
192
|
"primary_role": self.primary_role,
|
|
186
193
|
"when_to_use": self.when_to_use,
|
|
@@ -34,7 +34,7 @@ class GitRepository:
|
|
|
34
34
|
def cache_path(self) -> Path:
|
|
35
35
|
"""Return cache directory path for this repository.
|
|
36
36
|
|
|
37
|
-
Cache structure: ~/.claude-mpm/cache/
|
|
37
|
+
Cache structure: ~/.claude-mpm/cache/agents/{owner}/{repo}/{subdirectory}/
|
|
38
38
|
|
|
39
39
|
Returns:
|
|
40
40
|
Absolute path to cache directory for this repository
|
|
@@ -45,10 +45,10 @@ class GitRepository:
|
|
|
45
45
|
... subdirectory="agents"
|
|
46
46
|
... )
|
|
47
47
|
>>> repo.cache_path
|
|
48
|
-
Path('/Users/user/.claude-mpm/cache/
|
|
48
|
+
Path('/Users/user/.claude-mpm/cache/agents/bobmatnyc/claude-mpm-agents/agents')
|
|
49
49
|
"""
|
|
50
50
|
home = Path.home()
|
|
51
|
-
base_cache = home / ".claude-mpm" / "cache" / "
|
|
51
|
+
base_cache = home / ".claude-mpm" / "cache" / "agents"
|
|
52
52
|
|
|
53
53
|
# Extract owner and repo from URL
|
|
54
54
|
owner, repo = self._parse_github_url(self.url)
|
|
@@ -62,8 +62,13 @@ set -e
|
|
|
62
62
|
SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
|
|
63
63
|
|
|
64
64
|
# Determine the claude-mpm root based on installation type
|
|
65
|
+
# Check if we're in a UV tools installation
|
|
66
|
+
if [[ "$SCRIPT_DIR" == *"/.local/share/uv/tools/"* ]]; then
|
|
67
|
+
# UV tools installation - script is at lib/python*/site-packages/claude_mpm/scripts/
|
|
68
|
+
# The tool root is what we need for Python detection
|
|
69
|
+
CLAUDE_MPM_ROOT="$(echo "$SCRIPT_DIR" | sed 's|/lib/python.*/site-packages.*||')"
|
|
65
70
|
# Check if we're in a pipx installation
|
|
66
|
-
|
|
71
|
+
elif [[ "$SCRIPT_DIR" == *"/.local/pipx/venvs/claude-mpm/"* ]]; then
|
|
67
72
|
# pipx installation - script is at lib/python*/site-packages/claude_mpm/scripts/
|
|
68
73
|
# The venv root is what we need for Python detection
|
|
69
74
|
CLAUDE_MPM_ROOT="$(echo "$SCRIPT_DIR" | sed 's|/lib/python.*/site-packages/.*||')"
|
|
@@ -88,18 +93,22 @@ fi
|
|
|
88
93
|
#
|
|
89
94
|
# STRATEGY:
|
|
90
95
|
# This function implements a fallback chain to find Python with claude-mpm dependencies:
|
|
91
|
-
# 1.
|
|
92
|
-
# 2.
|
|
93
|
-
# 3.
|
|
94
|
-
# 4.
|
|
96
|
+
# 1. UV-managed projects (uv.lock detected) - uses "uv run python"
|
|
97
|
+
# 2. UV tools installations (~/.local/share/uv/tools/) - uses tool's venv Python
|
|
98
|
+
# 3. pipx installations - uses pipx venv Python
|
|
99
|
+
# 4. Project-specific virtual environments (venv, .venv)
|
|
100
|
+
# 5. Currently active virtual environment ($VIRTUAL_ENV)
|
|
101
|
+
# 6. System python3 (may lack dependencies)
|
|
102
|
+
# 7. System python (last resort)
|
|
95
103
|
#
|
|
96
104
|
# WHY THIS APPROACH:
|
|
97
105
|
# - Claude MPM requires specific packages (socketio, eventlet) not in system Python
|
|
98
|
-
# -
|
|
106
|
+
# - UV and virtual environments ensure dependency isolation and availability
|
|
99
107
|
# - Multiple naming conventions supported (venv vs .venv)
|
|
100
108
|
# - Graceful degradation to system Python if no venv found
|
|
101
109
|
#
|
|
102
110
|
# ACTIVATION STRATEGY:
|
|
111
|
+
# - UV projects: use "uv run python" to execute in UV-managed environment
|
|
103
112
|
# - Sources activate script to set up environment variables
|
|
104
113
|
# - Returns specific Python path for exec (not just 'python')
|
|
105
114
|
# - Maintains environment in same shell process
|
|
@@ -110,10 +119,32 @@ fi
|
|
|
110
119
|
# - Caches result in process environment
|
|
111
120
|
#
|
|
112
121
|
# RETURNS:
|
|
113
|
-
# Absolute path to Python executable with claude-mpm dependencies
|
|
122
|
+
# Absolute path to Python executable with claude-mpm dependencies, or "uv run python" for UV projects
|
|
114
123
|
#
|
|
115
124
|
find_python_command() {
|
|
116
|
-
# 1. Check
|
|
125
|
+
# 1. Check for UV project first (uv.lock or pyproject.toml with uv)
|
|
126
|
+
if [ -f "$CLAUDE_MPM_ROOT/uv.lock" ]; then
|
|
127
|
+
if command -v uv &> /dev/null; then
|
|
128
|
+
echo "uv run --directory \"$CLAUDE_MPM_ROOT\" python"
|
|
129
|
+
return
|
|
130
|
+
fi
|
|
131
|
+
fi
|
|
132
|
+
|
|
133
|
+
# 2. Check if we're in a UV tools installation
|
|
134
|
+
if [[ "$SCRIPT_DIR" == *"/.local/share/uv/tools/"* ]]; then
|
|
135
|
+
# UV tools installation - extract the tool root directory
|
|
136
|
+
CLAUDE_MPM_ROOT="$(echo "$SCRIPT_DIR" | sed 's|/lib/python.*/site-packages.*||')"
|
|
137
|
+
local uv_python="$CLAUDE_MPM_ROOT/bin/python"
|
|
138
|
+
if [ -x "$uv_python" ]; then
|
|
139
|
+
if [ "${CLAUDE_MPM_HOOK_DEBUG}" = "true" ]; then
|
|
140
|
+
echo "[$(date -u +%Y-%m-%dT%H:%M:%S.%3NZ)] UV tools Python found: $uv_python" >> /tmp/claude-mpm-hook.log
|
|
141
|
+
fi
|
|
142
|
+
echo "$uv_python"
|
|
143
|
+
return
|
|
144
|
+
fi
|
|
145
|
+
fi
|
|
146
|
+
|
|
147
|
+
# 3. Check if we're in a pipx installation
|
|
117
148
|
if [[ "$SCRIPT_DIR" == *"/.local/pipx/venvs/claude-mpm/"* ]]; then
|
|
118
149
|
# pipx installation - use the pipx venv's Python directly
|
|
119
150
|
if [ -f "$CLAUDE_MPM_ROOT/bin/python" ]; then
|
|
@@ -122,7 +153,7 @@ find_python_command() {
|
|
|
122
153
|
fi
|
|
123
154
|
fi
|
|
124
155
|
|
|
125
|
-
#
|
|
156
|
+
# 4. Check for project-local virtual environment (common in development)
|
|
126
157
|
if [ -f "$CLAUDE_MPM_ROOT/venv/bin/activate" ]; then
|
|
127
158
|
source "$CLAUDE_MPM_ROOT/venv/bin/activate"
|
|
128
159
|
echo "$CLAUDE_MPM_ROOT/venv/bin/python"
|
|
@@ -143,7 +174,13 @@ find_python_command() {
|
|
|
143
174
|
PYTHON_CMD=$(find_python_command)
|
|
144
175
|
|
|
145
176
|
# Check installation type and set PYTHONPATH accordingly
|
|
146
|
-
if [[ "$SCRIPT_DIR" == *"/.local/
|
|
177
|
+
if [[ "$SCRIPT_DIR" == *"/.local/share/uv/tools/"* ]]; then
|
|
178
|
+
# UV tools installation - claude_mpm is already in the tool's site-packages
|
|
179
|
+
# No need to modify PYTHONPATH
|
|
180
|
+
if [ "${CLAUDE_MPM_HOOK_DEBUG}" = "true" ]; then
|
|
181
|
+
echo "[$(date -u +%Y-%m-%dT%H:%M:%S.%3NZ)] UV tools installation detected" >> /tmp/claude-mpm-hook.log
|
|
182
|
+
fi
|
|
183
|
+
elif [[ "$SCRIPT_DIR" == *"/.local/pipx/venvs/claude-mpm/"* ]]; then
|
|
147
184
|
# pipx installation - claude_mpm is already in the venv's site-packages
|
|
148
185
|
# No need to modify PYTHONPATH
|
|
149
186
|
if [ "${CLAUDE_MPM_HOOK_DEBUG}" = "true" ]; then
|
|
@@ -173,15 +210,45 @@ fi
|
|
|
173
210
|
# Set Socket.IO configuration for hook events
|
|
174
211
|
export CLAUDE_MPM_SOCKETIO_PORT="${CLAUDE_MPM_SOCKETIO_PORT:-8765}"
|
|
175
212
|
|
|
176
|
-
#
|
|
177
|
-
|
|
178
|
-
if ! exec "$PYTHON_CMD" -m claude_mpm.hooks.claude_hooks.hook_handler "$@" 2>/tmp/claude-mpm-hook-error.log; then
|
|
179
|
-
# If the Python handler fails, always return continue to not block Claude
|
|
213
|
+
# Function for debug logging
|
|
214
|
+
log_debug() {
|
|
180
215
|
if [ "${CLAUDE_MPM_HOOK_DEBUG}" = "true" ]; then
|
|
181
|
-
echo "[$(date -u +%Y-%m-%dT%H:%M:%S.%3NZ)]
|
|
182
|
-
echo "[$(date -u +%Y-%m-%dT%H:%M:%S.%3NZ)] Error: $(cat /tmp/claude-mpm-hook-error.log 2>/dev/null | head -5)" >> /tmp/claude-mpm-hook.log
|
|
216
|
+
echo "[$(date -u +%Y-%m-%dT%H:%M:%S.%3NZ)] $1" >> /tmp/claude-mpm-hook.log
|
|
183
217
|
fi
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
218
|
+
}
|
|
219
|
+
|
|
220
|
+
# Test Python works and module exists
|
|
221
|
+
# Handle UV's multi-word command specially
|
|
222
|
+
if [[ "$PYTHON_CMD" == "uv run"* ]]; then
|
|
223
|
+
if ! uv run --directory "$CLAUDE_MPM_ROOT" python -c "import claude_mpm" 2>/dev/null; then
|
|
224
|
+
log_debug "claude_mpm module not available, continuing without hook"
|
|
225
|
+
echo '{"action": "continue"}'
|
|
226
|
+
exit 0
|
|
227
|
+
fi
|
|
228
|
+
else
|
|
229
|
+
if ! $PYTHON_CMD -c "import claude_mpm" 2>/dev/null; then
|
|
230
|
+
log_debug "claude_mpm module not available, continuing without hook"
|
|
231
|
+
echo '{"action": "continue"}'
|
|
232
|
+
exit 0
|
|
233
|
+
fi
|
|
234
|
+
fi
|
|
235
|
+
|
|
236
|
+
# Run the Python hook handler with all input
|
|
237
|
+
# Use exec to replace the shell process with Python
|
|
238
|
+
# Handle UV's multi-word command specially
|
|
239
|
+
# Suppress RuntimeWarning to prevent stderr output (which causes hook errors)
|
|
240
|
+
if [[ "$PYTHON_CMD" == "uv run"* ]]; then
|
|
241
|
+
exec uv run --directory "$CLAUDE_MPM_ROOT" python -W ignore::RuntimeWarning -m claude_mpm.hooks.claude_hooks.hook_handler "$@" 2>/tmp/claude-mpm-hook-error.log
|
|
242
|
+
else
|
|
243
|
+
exec "$PYTHON_CMD" -W ignore::RuntimeWarning -m claude_mpm.hooks.claude_hooks.hook_handler "$@" 2>/tmp/claude-mpm-hook-error.log
|
|
244
|
+
fi
|
|
245
|
+
|
|
246
|
+
# Note: exec replaces the shell process, so code below only runs if exec fails
|
|
247
|
+
# If we reach here, the Python handler failed
|
|
248
|
+
if [ "${CLAUDE_MPM_HOOK_DEBUG}" = "true" ]; then
|
|
249
|
+
echo "[$(date -u +%Y-%m-%dT%H:%M:%S.%3NZ)] Hook handler failed, see /tmp/claude-mpm-hook-error.log" >> /tmp/claude-mpm-hook.log
|
|
250
|
+
echo "[$(date -u +%Y-%m-%dT%H:%M:%S.%3NZ)] Error: $(cat /tmp/claude-mpm-hook-error.log 2>/dev/null | head -5)" >> /tmp/claude-mpm-hook.log
|
|
251
|
+
fi
|
|
252
|
+
# Return continue action to prevent blocking Claude Code
|
|
253
|
+
echo '{"action": "continue"}'
|
|
254
|
+
exit 0
|
|
@@ -7,6 +7,12 @@ dashboard, which includes both the Socket.IO server and web interface.
|
|
|
7
7
|
|
|
8
8
|
WHY: Provides a simple command to start the monitoring dashboard that tracks
|
|
9
9
|
Claude MPM events and agent activity in real-time.
|
|
10
|
+
|
|
11
|
+
SINGLE INSTANCE ENFORCEMENT:
|
|
12
|
+
- Only ONE monitor instance runs at a time on port 8765 (default)
|
|
13
|
+
- If monitor already running on default port: reuse existing, open browser
|
|
14
|
+
- If user specifies --port explicitly: use that port, fail if busy
|
|
15
|
+
- No auto-increment port selection (prevents multiple instances)
|
|
10
16
|
"""
|
|
11
17
|
|
|
12
18
|
import argparse
|
|
@@ -15,12 +21,36 @@ import webbrowser
|
|
|
15
21
|
|
|
16
22
|
from claude_mpm.core.logging_config import get_logger
|
|
17
23
|
from claude_mpm.services.monitor.daemon import UnifiedMonitorDaemon
|
|
18
|
-
from claude_mpm.services.
|
|
24
|
+
from claude_mpm.services.monitor.daemon_manager import DaemonManager
|
|
19
25
|
|
|
20
26
|
DEFAULT_PORT = 8765
|
|
21
27
|
logger = get_logger(__name__)
|
|
22
28
|
|
|
23
29
|
|
|
30
|
+
def check_existing_monitor(host: str, port: int) -> bool:
|
|
31
|
+
"""Check if monitor is already running on the specified port.
|
|
32
|
+
|
|
33
|
+
Args:
|
|
34
|
+
host: Host to check
|
|
35
|
+
port: Port to check
|
|
36
|
+
|
|
37
|
+
Returns:
|
|
38
|
+
True if monitor is running, False otherwise
|
|
39
|
+
"""
|
|
40
|
+
try:
|
|
41
|
+
import requests
|
|
42
|
+
|
|
43
|
+
response = requests.get(f"http://{host}:{port}/health", timeout=2)
|
|
44
|
+
if response.status_code == 200:
|
|
45
|
+
data = response.json()
|
|
46
|
+
# Check if it's our claude-mpm-monitor service
|
|
47
|
+
if data.get("service") == "claude-mpm-monitor":
|
|
48
|
+
return True
|
|
49
|
+
except Exception:
|
|
50
|
+
pass
|
|
51
|
+
return False
|
|
52
|
+
|
|
53
|
+
|
|
24
54
|
def main():
|
|
25
55
|
"""Main entry point for monitor launcher."""
|
|
26
56
|
parser = argparse.ArgumentParser(
|
|
@@ -30,8 +60,8 @@ def main():
|
|
|
30
60
|
parser.add_argument(
|
|
31
61
|
"--port",
|
|
32
62
|
type=int,
|
|
33
|
-
default=DEFAULT_PORT
|
|
34
|
-
help=f"Port to run on (default: {DEFAULT_PORT})",
|
|
63
|
+
default=None, # Changed: None means use DEFAULT_PORT with single-instance check
|
|
64
|
+
help=f"Port to run on (default: {DEFAULT_PORT}). If specified, fails if port is busy.",
|
|
35
65
|
)
|
|
36
66
|
|
|
37
67
|
parser.add_argument(
|
|
@@ -46,20 +76,70 @@ def main():
|
|
|
46
76
|
"--background", action="store_true", help="Run in background daemon mode"
|
|
47
77
|
)
|
|
48
78
|
|
|
79
|
+
parser.add_argument(
|
|
80
|
+
"--dev",
|
|
81
|
+
action="store_true",
|
|
82
|
+
help="Enable development mode with hot reload for Svelte changes",
|
|
83
|
+
)
|
|
84
|
+
|
|
49
85
|
args = parser.parse_args()
|
|
50
86
|
|
|
51
|
-
#
|
|
52
|
-
|
|
53
|
-
|
|
87
|
+
# Determine target port
|
|
88
|
+
user_specified_port = args.port is not None
|
|
89
|
+
target_port = args.port if user_specified_port else DEFAULT_PORT
|
|
90
|
+
|
|
91
|
+
# SINGLE INSTANCE ENFORCEMENT:
|
|
92
|
+
# Check if monitor already running on target port
|
|
93
|
+
if check_existing_monitor(args.host, target_port):
|
|
94
|
+
logger.info(f"Monitor already running at http://{args.host}:{target_port}")
|
|
54
95
|
|
|
55
|
-
|
|
56
|
-
|
|
96
|
+
# Open browser to existing instance if requested
|
|
97
|
+
if not args.no_browser:
|
|
98
|
+
url = f"http://{args.host}:{target_port}"
|
|
99
|
+
logger.info(f"Opening browser to existing instance: {url}")
|
|
100
|
+
webbrowser.open(url)
|
|
101
|
+
|
|
102
|
+
# Success - reusing existing instance
|
|
103
|
+
return
|
|
104
|
+
|
|
105
|
+
# Port selection logic:
|
|
106
|
+
# - If user specified --port: Use that exact port, fail if busy
|
|
107
|
+
# - If no --port: Use DEFAULT_PORT (8765), fail if busy
|
|
108
|
+
# - Never auto-increment to find free port
|
|
109
|
+
|
|
110
|
+
# Create daemon manager for port checking
|
|
111
|
+
daemon_manager = DaemonManager(port=target_port, host=args.host)
|
|
112
|
+
|
|
113
|
+
if not daemon_manager._is_port_available():
|
|
114
|
+
if user_specified_port:
|
|
115
|
+
# User explicitly requested a port - fail with clear message
|
|
116
|
+
logger.error(
|
|
117
|
+
f"Port {target_port} is already in use by another service. "
|
|
118
|
+
f"Please stop the existing service or choose a different port."
|
|
119
|
+
)
|
|
120
|
+
sys.exit(1)
|
|
121
|
+
else:
|
|
122
|
+
# Default port is busy - fail with helpful message
|
|
123
|
+
logger.error(
|
|
124
|
+
f"Default port {DEFAULT_PORT} is already in use by another service. "
|
|
125
|
+
f"Please stop the existing service with 'claude-mpm monitor stop' "
|
|
126
|
+
f"or specify a different port with --port."
|
|
127
|
+
)
|
|
128
|
+
sys.exit(1)
|
|
57
129
|
|
|
58
130
|
# Start the monitor daemon
|
|
59
|
-
|
|
131
|
+
if args.dev:
|
|
132
|
+
logger.info(
|
|
133
|
+
f"Starting Claude MPM monitor on {args.host}:{target_port} (DEV MODE - hot reload enabled)"
|
|
134
|
+
)
|
|
135
|
+
else:
|
|
136
|
+
logger.info(f"Starting Claude MPM monitor on {args.host}:{target_port}")
|
|
60
137
|
|
|
61
138
|
daemon = UnifiedMonitorDaemon(
|
|
62
|
-
host=args.host,
|
|
139
|
+
host=args.host,
|
|
140
|
+
port=target_port,
|
|
141
|
+
daemon_mode=args.background,
|
|
142
|
+
enable_hot_reload=args.dev,
|
|
63
143
|
)
|
|
64
144
|
|
|
65
145
|
success = daemon.start()
|
|
@@ -67,14 +147,14 @@ def main():
|
|
|
67
147
|
if success:
|
|
68
148
|
# Open browser if requested
|
|
69
149
|
if not args.no_browser:
|
|
70
|
-
url = f"http://{args.host}:{
|
|
150
|
+
url = f"http://{args.host}:{target_port}"
|
|
71
151
|
logger.info(f"Opening browser to {url}")
|
|
72
152
|
webbrowser.open(url)
|
|
73
153
|
|
|
74
154
|
if args.background:
|
|
75
|
-
logger.info(f"Monitor daemon started in background on port {
|
|
155
|
+
logger.info(f"Monitor daemon started in background on port {target_port}")
|
|
76
156
|
else:
|
|
77
|
-
logger.info(f"Monitor running on port {
|
|
157
|
+
logger.info(f"Monitor running on port {target_port}")
|
|
78
158
|
logger.info("Press Ctrl+C to stop")
|
|
79
159
|
else:
|
|
80
160
|
logger.error("Failed to start monitor")
|