claude-mpm 5.4.96__py3-none-any.whl → 5.6.17__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/agents/{CLAUDE_MPM_FOUNDERS_OUTPUT_STYLE.md → CLAUDE_MPM_RESEARCH_OUTPUT_STYLE.md} +14 -6
- claude_mpm/agents/PM_INSTRUCTIONS.md +44 -10
- claude_mpm/agents/WORKFLOW.md +2 -0
- claude_mpm/agents/templates/circuit-breakers.md +26 -17
- claude_mpm/cli/commands/autotodos.py +45 -5
- claude_mpm/cli/commands/commander.py +216 -0
- claude_mpm/cli/commands/hook_errors.py +60 -60
- claude_mpm/cli/commands/run.py +35 -3
- claude_mpm/cli/commands/skill_source.py +51 -2
- claude_mpm/cli/commands/skills.py +5 -3
- claude_mpm/cli/executor.py +32 -17
- claude_mpm/cli/parsers/base_parser.py +17 -0
- claude_mpm/cli/parsers/commander_parser.py +116 -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 +5 -0
- claude_mpm/cli/startup.py +124 -3
- claude_mpm/cli/startup_display.py +2 -1
- 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 +111 -0
- claude_mpm/commander/chat/commands.py +96 -0
- claude_mpm/commander/chat/repl.py +310 -0
- claude_mpm/commander/config.py +49 -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 +594 -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 +143 -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 +337 -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 +8 -0
- claude_mpm/commands/mpm-doctor.md +8 -0
- claude_mpm/commands/mpm-help.md +8 -0
- claude_mpm/commands/mpm-init.md +8 -0
- claude_mpm/commands/mpm-monitor.md +8 -0
- claude_mpm/commands/mpm-organize.md +8 -0
- claude_mpm/commands/mpm-postmortem.md +8 -0
- claude_mpm/commands/mpm-session-resume.md +8 -0
- claude_mpm/commands/mpm-status.md +8 -0
- claude_mpm/commands/mpm-ticket-view.md +8 -0
- claude_mpm/commands/mpm-version.md +8 -0
- claude_mpm/commands/mpm.md +8 -0
- claude_mpm/config/agent_presets.py +8 -7
- claude_mpm/config/skill_sources.py +16 -0
- claude_mpm/core/claude_runner.py +143 -0
- claude_mpm/core/config.py +32 -19
- claude_mpm/core/logger.py +26 -9
- claude_mpm/core/logging_utils.py +35 -11
- claude_mpm/core/output_style_manager.py +49 -12
- claude_mpm/core/unified_config.py +10 -6
- claude_mpm/core/unified_paths.py +68 -80
- claude_mpm/experimental/cli_enhancements.py +2 -1
- claude_mpm/hooks/claude_hooks/__pycache__/__init__.cpython-312.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/__init__.cpython-314.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/auto_pause_handler.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/auto_pause_handler.cpython-312.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/auto_pause_handler.cpython-314.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/event_handlers.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/event_handlers.cpython-312.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/event_handlers.cpython-314.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/hook_handler.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/hook_handler.cpython-312.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/hook_handler.cpython-314.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/installer.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/installer.cpython-314.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/memory_integration.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/memory_integration.cpython-312.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/memory_integration.cpython-314.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/response_tracking.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/response_tracking.cpython-312.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/response_tracking.cpython-314.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/tool_analysis.cpython-312.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/tool_analysis.cpython-314.pyc +0 -0
- claude_mpm/hooks/claude_hooks/auto_pause_handler.py +29 -30
- claude_mpm/hooks/claude_hooks/event_handlers.py +112 -99
- claude_mpm/hooks/claude_hooks/hook_handler.py +81 -88
- claude_mpm/hooks/claude_hooks/hook_wrapper.sh +6 -11
- claude_mpm/hooks/claude_hooks/installer.py +116 -8
- claude_mpm/hooks/claude_hooks/memory_integration.py +51 -31
- claude_mpm/hooks/claude_hooks/response_tracking.py +39 -58
- claude_mpm/hooks/claude_hooks/services/__pycache__/__init__.cpython-312.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/__init__.cpython-314.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/connection_manager.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__/connection_manager_http.cpython-312.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/connection_manager_http.cpython-314.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/duplicate_detector.cpython-312.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/duplicate_detector.cpython-314.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/state_manager.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/state_manager.cpython-312.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/state_manager.cpython-314.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/subagent_processor.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/subagent_processor.cpython-312.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/subagent_processor.cpython-314.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/connection_manager.py +23 -28
- 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 +47 -73
- claude_mpm/hooks/session_resume_hook.py +22 -18
- claude_mpm/hooks/templates/pre_tool_use_template.py +10 -2
- claude_mpm/scripts/claude-hook-handler.sh +43 -16
- claude_mpm/scripts/start_activity_logging.py +0 -0
- claude_mpm/services/agents/agent_recommendation_service.py +8 -8
- claude_mpm/services/agents/agent_selection_service.py +2 -2
- claude_mpm/services/agents/loading/framework_agent_loader.py +75 -2
- claude_mpm/services/agents/single_tier_deployment_service.py +4 -4
- claude_mpm/services/event_log.py +8 -0
- claude_mpm/services/pm_skills_deployer.py +84 -6
- claude_mpm/services/skills/git_skill_source_manager.py +130 -10
- claude_mpm/services/skills/selective_skill_deployer.py +28 -0
- claude_mpm/services/skills/skill_discovery_service.py +74 -4
- claude_mpm/services/skills_deployer.py +31 -5
- claude_mpm/skills/__init__.py +2 -1
- claude_mpm/skills/bundled/pm/mpm/SKILL.md +38 -0
- claude_mpm/skills/bundled/pm/mpm-config/SKILL.md +29 -0
- claude_mpm/skills/bundled/pm/mpm-doctor/SKILL.md +53 -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-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-ticket-view/SKILL.md +110 -0
- claude_mpm/skills/bundled/pm/mpm-version/SKILL.md +21 -0
- claude_mpm/skills/registry.py +295 -90
- {claude_mpm-5.4.96.dist-info → claude_mpm-5.6.17.dist-info}/METADATA +22 -6
- {claude_mpm-5.4.96.dist-info → claude_mpm-5.6.17.dist-info}/RECORD +213 -83
- {claude_mpm-5.4.96.dist-info → claude_mpm-5.6.17.dist-info}/WHEEL +0 -0
- {claude_mpm-5.4.96.dist-info → claude_mpm-5.6.17.dist-info}/entry_points.txt +0 -0
- {claude_mpm-5.4.96.dist-info → claude_mpm-5.6.17.dist-info}/licenses/LICENSE +0 -0
- {claude_mpm-5.4.96.dist-info → claude_mpm-5.6.17.dist-info}/licenses/LICENSE-FAQ.md +0 -0
- {claude_mpm-5.4.96.dist-info → claude_mpm-5.6.17.dist-info}/top_level.txt +0 -0
|
@@ -7,14 +7,27 @@ This service manages:
|
|
|
7
7
|
DESIGN DECISION: Use stateless HTTP POST instead of persistent SocketIO
|
|
8
8
|
connections because hook handlers are ephemeral processes (< 1 second lifetime).
|
|
9
9
|
This eliminates disconnection issues and matches the process lifecycle.
|
|
10
|
+
|
|
11
|
+
DESIGN DECISION: Synchronous HTTP POST only (no async)
|
|
12
|
+
Hook handlers are too short-lived (~25ms lifecycle) to benefit from async.
|
|
13
|
+
Using asyncio.run() creates event loops that close before HTTP operations complete,
|
|
14
|
+
causing "Event loop is closed" errors. Synchronous HTTP POST in a thread pool
|
|
15
|
+
is simpler and more reliable for ephemeral processes.
|
|
10
16
|
"""
|
|
11
17
|
|
|
12
|
-
import asyncio
|
|
13
18
|
import os
|
|
14
|
-
import sys
|
|
15
19
|
from concurrent.futures import ThreadPoolExecutor
|
|
16
20
|
from datetime import datetime, timezone
|
|
17
21
|
|
|
22
|
+
# Try to import _log from hook_handler, fall back to no-op
|
|
23
|
+
try:
|
|
24
|
+
from claude_mpm.hooks.claude_hooks.hook_handler import _log
|
|
25
|
+
except ImportError:
|
|
26
|
+
|
|
27
|
+
def _log(msg: str) -> None:
|
|
28
|
+
pass # Silent fallback
|
|
29
|
+
|
|
30
|
+
|
|
18
31
|
# Debug mode is enabled by default for better visibility into hook processing
|
|
19
32
|
DEBUG = os.environ.get("CLAUDE_MPM_HOOK_DEBUG", "true").lower() != "false"
|
|
20
33
|
|
|
@@ -27,9 +40,6 @@ except ImportError:
|
|
|
27
40
|
REQUESTS_AVAILABLE = False
|
|
28
41
|
requests = None
|
|
29
42
|
|
|
30
|
-
# Import high-performance event emitter - lazy loaded in _async_emit()
|
|
31
|
-
# to reduce hook handler initialization time by ~85% (792ms -> minimal)
|
|
32
|
-
|
|
33
43
|
# Import EventNormalizer for consistent event formatting
|
|
34
44
|
try:
|
|
35
45
|
from claude_mpm.services.socketio.event_normalizer import EventNormalizer
|
|
@@ -55,10 +65,6 @@ except ImportError:
|
|
|
55
65
|
)
|
|
56
66
|
|
|
57
67
|
|
|
58
|
-
# EventBus removed - using direct HTTP POST only
|
|
59
|
-
# This eliminates duplicate events and simplifies the architecture
|
|
60
|
-
|
|
61
|
-
|
|
62
68
|
class ConnectionManagerService:
|
|
63
69
|
"""Manages connections for the Claude hook handler using HTTP POST."""
|
|
64
70
|
|
|
@@ -72,35 +78,26 @@ class ConnectionManagerService:
|
|
|
72
78
|
self.server_port = int(os.environ.get("CLAUDE_MPM_SERVER_PORT", "8765"))
|
|
73
79
|
self.http_endpoint = f"http://{self.server_host}:{self.server_port}/api/events"
|
|
74
80
|
|
|
75
|
-
# EventBus removed - using direct HTTP POST only
|
|
76
|
-
|
|
77
|
-
# For backward compatibility with tests
|
|
78
|
-
self.connection_pool = None # No longer used
|
|
79
|
-
|
|
80
|
-
# Track async emit tasks to prevent garbage collection
|
|
81
|
-
self._emit_tasks: set = set()
|
|
82
|
-
|
|
83
81
|
# Thread pool for non-blocking HTTP requests
|
|
84
82
|
# WHY: Prevents HTTP POST from blocking hook processing (2s timeout → 0ms blocking)
|
|
85
|
-
# max_workers=2: Sufficient for low-frequency
|
|
83
|
+
# max_workers=2: Sufficient for low-frequency hook events
|
|
86
84
|
self._http_executor = ThreadPoolExecutor(
|
|
87
85
|
max_workers=2, thread_name_prefix="http-emit"
|
|
88
86
|
)
|
|
89
87
|
|
|
90
88
|
if DEBUG:
|
|
91
|
-
|
|
92
|
-
f"✅ HTTP connection manager initialized - endpoint: {self.http_endpoint}"
|
|
93
|
-
file=sys.stderr,
|
|
89
|
+
_log(
|
|
90
|
+
f"✅ HTTP connection manager initialized - endpoint: {self.http_endpoint}"
|
|
94
91
|
)
|
|
95
92
|
|
|
96
93
|
def emit_event(self, namespace: str, event: str, data: dict):
|
|
97
|
-
"""Emit event using
|
|
94
|
+
"""Emit event using HTTP POST.
|
|
98
95
|
|
|
99
|
-
WHY
|
|
100
|
-
-
|
|
101
|
-
-
|
|
102
|
-
-
|
|
103
|
-
-
|
|
96
|
+
WHY HTTP POST only:
|
|
97
|
+
- Hook handlers are ephemeral (~25ms lifecycle)
|
|
98
|
+
- Async emission causes "Event loop is closed" errors
|
|
99
|
+
- HTTP POST in thread pool is simpler and more reliable
|
|
100
|
+
- Completes in 20-50ms, which is acceptable for hook handlers
|
|
104
101
|
"""
|
|
105
102
|
# Create event data for normalization
|
|
106
103
|
raw_event = {
|
|
@@ -120,74 +117,17 @@ class ConnectionManagerService:
|
|
|
120
117
|
if DEBUG and event in ["subagent_stop", "pre_tool"]:
|
|
121
118
|
if event == "subagent_stop":
|
|
122
119
|
agent_type = data.get("agent_type", "unknown")
|
|
123
|
-
|
|
124
|
-
f"Hook handler: Publishing SubagentStop for agent '{agent_type}'",
|
|
125
|
-
file=sys.stderr,
|
|
126
|
-
)
|
|
120
|
+
_log(f"Hook handler: Publishing SubagentStop for agent '{agent_type}'")
|
|
127
121
|
elif event == "pre_tool" and data.get("tool_name") == "Task":
|
|
128
122
|
delegation = data.get("delegation_details", {})
|
|
129
123
|
agent_type = delegation.get("agent_type", "unknown")
|
|
130
|
-
|
|
131
|
-
f"Hook handler: Publishing Task delegation to agent '{agent_type}'"
|
|
132
|
-
file=sys.stderr,
|
|
124
|
+
_log(
|
|
125
|
+
f"Hook handler: Publishing Task delegation to agent '{agent_type}'"
|
|
133
126
|
)
|
|
134
127
|
|
|
135
|
-
#
|
|
136
|
-
success = self._try_async_emit(namespace, event, claude_event_data)
|
|
137
|
-
if success:
|
|
138
|
-
return
|
|
139
|
-
|
|
140
|
-
# Fallback to HTTP POST for cross-process communication
|
|
128
|
+
# Emit via HTTP POST (non-blocking, runs in thread pool)
|
|
141
129
|
self._try_http_emit(namespace, event, claude_event_data)
|
|
142
130
|
|
|
143
|
-
def _try_async_emit(self, namespace: str, event: str, data: dict) -> bool:
|
|
144
|
-
"""Try to emit event using high-performance async emitter."""
|
|
145
|
-
try:
|
|
146
|
-
# Run async emission in the current event loop or create one
|
|
147
|
-
loop = None
|
|
148
|
-
try:
|
|
149
|
-
loop = asyncio.get_running_loop()
|
|
150
|
-
except RuntimeError:
|
|
151
|
-
# No running loop, create a new one
|
|
152
|
-
pass
|
|
153
|
-
|
|
154
|
-
if loop:
|
|
155
|
-
# We're in an async context, create a task with tracking
|
|
156
|
-
task = loop.create_task(self._async_emit(namespace, event, data))
|
|
157
|
-
self._emit_tasks.add(task)
|
|
158
|
-
task.add_done_callback(self._emit_tasks.discard)
|
|
159
|
-
# Don't wait for completion to maintain low latency
|
|
160
|
-
if DEBUG:
|
|
161
|
-
print(f"✅ Async emit scheduled: {event}", file=sys.stderr)
|
|
162
|
-
return True
|
|
163
|
-
# No event loop, run synchronously
|
|
164
|
-
success = asyncio.run(self._async_emit(namespace, event, data))
|
|
165
|
-
if DEBUG and success:
|
|
166
|
-
print(f"✅ Async emit successful: {event}", file=sys.stderr)
|
|
167
|
-
return success
|
|
168
|
-
|
|
169
|
-
except Exception as e:
|
|
170
|
-
if DEBUG:
|
|
171
|
-
print(f"⚠️ Async emit failed: {e}", file=sys.stderr)
|
|
172
|
-
return False
|
|
173
|
-
|
|
174
|
-
async def _async_emit(self, namespace: str, event: str, data: dict) -> bool:
|
|
175
|
-
"""Async helper for event emission."""
|
|
176
|
-
try:
|
|
177
|
-
# Lazy load event emitter to reduce initialization overhead
|
|
178
|
-
from claude_mpm.services.monitor.event_emitter import get_event_emitter
|
|
179
|
-
|
|
180
|
-
emitter = await get_event_emitter()
|
|
181
|
-
return await emitter.emit_event(namespace, "claude_event", data)
|
|
182
|
-
except ImportError:
|
|
183
|
-
if DEBUG:
|
|
184
|
-
print("⚠️ Event emitter not available", file=sys.stderr)
|
|
185
|
-
return False
|
|
186
|
-
except Exception as e:
|
|
187
|
-
if DEBUG:
|
|
188
|
-
print(f"⚠️ Async emitter error: {e}", file=sys.stderr)
|
|
189
|
-
return False
|
|
190
|
-
|
|
191
131
|
def _try_http_emit(self, namespace: str, event: str, data: dict):
|
|
192
132
|
"""Try to emit event using HTTP POST fallback (non-blocking).
|
|
193
133
|
|
|
@@ -196,10 +136,7 @@ class ConnectionManagerService:
|
|
|
196
136
|
"""
|
|
197
137
|
if not REQUESTS_AVAILABLE:
|
|
198
138
|
if DEBUG:
|
|
199
|
-
|
|
200
|
-
"⚠️ requests module not available - cannot emit via HTTP",
|
|
201
|
-
file=sys.stderr,
|
|
202
|
-
)
|
|
139
|
+
_log("⚠️ requests module not available - cannot emit via HTTP")
|
|
203
140
|
return
|
|
204
141
|
|
|
205
142
|
# Submit to thread pool - don't wait for result (fire-and-forget)
|
|
@@ -225,25 +162,21 @@ class ConnectionManagerService:
|
|
|
225
162
|
|
|
226
163
|
if response.status_code in [200, 204]:
|
|
227
164
|
if DEBUG:
|
|
228
|
-
|
|
165
|
+
_log(f"✅ HTTP POST successful: {event}")
|
|
229
166
|
elif DEBUG:
|
|
230
|
-
|
|
231
|
-
f"⚠️ HTTP POST failed with status {response.status_code}: {event}",
|
|
232
|
-
file=sys.stderr,
|
|
233
|
-
)
|
|
167
|
+
_log(f"⚠️ HTTP POST failed with status {response.status_code}: {event}")
|
|
234
168
|
|
|
235
169
|
except requests.exceptions.Timeout:
|
|
236
170
|
if DEBUG:
|
|
237
|
-
|
|
171
|
+
_log(f"⚠️ HTTP POST timeout for: {event}")
|
|
238
172
|
except requests.exceptions.ConnectionError:
|
|
239
173
|
if DEBUG:
|
|
240
|
-
|
|
241
|
-
f"⚠️ HTTP POST connection failed for: {event} (server not running?)"
|
|
242
|
-
file=sys.stderr,
|
|
174
|
+
_log(
|
|
175
|
+
f"⚠️ HTTP POST connection failed for: {event} (server not running?)"
|
|
243
176
|
)
|
|
244
177
|
except Exception as e:
|
|
245
178
|
if DEBUG:
|
|
246
|
-
|
|
179
|
+
_log(f"⚠️ HTTP POST error for {event}: {e}")
|
|
247
180
|
|
|
248
181
|
def cleanup(self):
|
|
249
182
|
"""Cleanup connections on service destruction."""
|
|
@@ -251,4 +184,4 @@ class ConnectionManagerService:
|
|
|
251
184
|
if hasattr(self, "_http_executor"):
|
|
252
185
|
self._http_executor.shutdown(wait=False)
|
|
253
186
|
if DEBUG:
|
|
254
|
-
|
|
187
|
+
_log("✅ HTTP executor shutdown")
|
|
@@ -8,13 +8,22 @@ This service manages:
|
|
|
8
8
|
"""
|
|
9
9
|
|
|
10
10
|
import os
|
|
11
|
-
import subprocess
|
|
11
|
+
import subprocess # nosec B404
|
|
12
12
|
import time
|
|
13
13
|
from collections import deque
|
|
14
14
|
from datetime import datetime, timezone
|
|
15
15
|
from pathlib import Path
|
|
16
16
|
from typing import Optional
|
|
17
17
|
|
|
18
|
+
# Try to import _log from hook_handler, fall back to no-op
|
|
19
|
+
try:
|
|
20
|
+
from claude_mpm.hooks.claude_hooks.hook_handler import _log
|
|
21
|
+
except ImportError:
|
|
22
|
+
|
|
23
|
+
def _log(msg: str) -> None:
|
|
24
|
+
pass # Silent fallback
|
|
25
|
+
|
|
26
|
+
|
|
18
27
|
# Import constants for configuration
|
|
19
28
|
try:
|
|
20
29
|
from claude_mpm.core.constants import TimeoutConfig
|
|
@@ -63,17 +72,11 @@ class StateManagerService:
|
|
|
63
72
|
):
|
|
64
73
|
"""Track a new agent delegation with optional request data for response correlation."""
|
|
65
74
|
if DEBUG:
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
)
|
|
72
|
-
print(f" - agent_type: {agent_type}", file=sys.stderr)
|
|
73
|
-
print(f" - request_data provided: {bool(request_data)}", file=sys.stderr)
|
|
74
|
-
print(
|
|
75
|
-
f" - delegation_requests size before: {len(self.delegation_requests)}",
|
|
76
|
-
file=sys.stderr,
|
|
75
|
+
_log(f" - session_id: {session_id[:16] if session_id else 'None'}...")
|
|
76
|
+
_log(f" - agent_type: {agent_type}")
|
|
77
|
+
_log(f" - request_data provided: {bool(request_data)}")
|
|
78
|
+
_log(
|
|
79
|
+
f" - delegation_requests size before: {len(self.delegation_requests)}"
|
|
77
80
|
)
|
|
78
81
|
|
|
79
82
|
if session_id and agent_type and agent_type != "unknown":
|
|
@@ -89,15 +92,9 @@ class StateManagerService:
|
|
|
89
92
|
"timestamp": datetime.now(timezone.utc).isoformat(),
|
|
90
93
|
}
|
|
91
94
|
if DEBUG:
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
f" - ✅ Stored in delegation_requests[{session_id[:16]}...]",
|
|
96
|
-
file=sys.stderr,
|
|
97
|
-
)
|
|
98
|
-
print(
|
|
99
|
-
f" - delegation_requests size after: {len(self.delegation_requests)}",
|
|
100
|
-
file=sys.stderr,
|
|
95
|
+
_log(f" - ✅ Stored in delegation_requests[{session_id[:16]}...]")
|
|
96
|
+
_log(
|
|
97
|
+
f" - delegation_requests size after: {len(self.delegation_requests)}"
|
|
101
98
|
)
|
|
102
99
|
|
|
103
100
|
# Clean up old delegations (older than 5 minutes)
|
|
@@ -197,7 +194,7 @@ class StateManagerService:
|
|
|
197
194
|
os.chdir(working_dir)
|
|
198
195
|
|
|
199
196
|
# Run git command to get current branch
|
|
200
|
-
result = subprocess.run(
|
|
197
|
+
result = subprocess.run( # nosec B603 B607
|
|
201
198
|
["git", "branch", "--show-current"],
|
|
202
199
|
capture_output=True,
|
|
203
200
|
text=True,
|
|
@@ -233,17 +230,12 @@ class StateManagerService:
|
|
|
233
230
|
def find_matching_request(self, session_id: str) -> Optional[dict]:
|
|
234
231
|
"""Find matching request data for a session, with fuzzy matching fallback."""
|
|
235
232
|
# First try exact match
|
|
236
|
-
request_info = self.delegation_requests.get(session_id)
|
|
233
|
+
request_info = self.delegation_requests.get(session_id) # nosec B113
|
|
237
234
|
|
|
238
235
|
# If exact match fails, try partial matching
|
|
239
236
|
if not request_info and session_id:
|
|
240
237
|
if DEBUG:
|
|
241
|
-
|
|
242
|
-
|
|
243
|
-
print(
|
|
244
|
-
f" - Trying fuzzy match for session {session_id[:16]}...",
|
|
245
|
-
file=sys.stderr,
|
|
246
|
-
)
|
|
238
|
+
_log(f" - Trying fuzzy match for session {session_id[:16]}...")
|
|
247
239
|
# Try to find a session that matches the first 8-16 characters
|
|
248
240
|
for stored_sid in list(self.delegation_requests.keys()):
|
|
249
241
|
if (
|
|
@@ -256,13 +248,8 @@ class StateManagerService:
|
|
|
256
248
|
)
|
|
257
249
|
):
|
|
258
250
|
if DEBUG:
|
|
259
|
-
|
|
260
|
-
|
|
261
|
-
print(
|
|
262
|
-
f" - ✅ Fuzzy match found: {stored_sid[:16]}...",
|
|
263
|
-
file=sys.stderr,
|
|
264
|
-
)
|
|
265
|
-
request_info = self.delegation_requests.get(stored_sid)
|
|
251
|
+
_log(f" - ✅ Fuzzy match found: {stored_sid[:16]}...")
|
|
252
|
+
request_info = self.delegation_requests.get(stored_sid) # nosec B113
|
|
266
253
|
# Update the key to use the current session_id for consistency
|
|
267
254
|
if request_info:
|
|
268
255
|
self.delegation_requests[session_id] = request_info
|
|
@@ -10,10 +10,18 @@ This service handles:
|
|
|
10
10
|
import json
|
|
11
11
|
import os
|
|
12
12
|
import re
|
|
13
|
-
import sys
|
|
14
13
|
from datetime import datetime, timezone
|
|
15
14
|
from typing import Optional, Tuple
|
|
16
15
|
|
|
16
|
+
# Try to import _log from hook_handler, fall back to no-op
|
|
17
|
+
try:
|
|
18
|
+
from claude_mpm.hooks.claude_hooks.hook_handler import _log
|
|
19
|
+
except ImportError:
|
|
20
|
+
|
|
21
|
+
def _log(msg: str) -> None:
|
|
22
|
+
pass # Silent fallback
|
|
23
|
+
|
|
24
|
+
|
|
17
25
|
# Debug mode is enabled by default for better visibility into hook processing
|
|
18
26
|
DEBUG = os.environ.get("CLAUDE_MPM_HOOK_DEBUG", "true").lower() != "false"
|
|
19
27
|
|
|
@@ -45,26 +53,21 @@ class SubagentResponseProcessor:
|
|
|
45
53
|
# Enhanced debug logging for session correlation
|
|
46
54
|
session_id = event.get("session_id", "")
|
|
47
55
|
if DEBUG:
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
print(f" - event keys: {list(event.keys())}", file=sys.stderr)
|
|
53
|
-
print(
|
|
54
|
-
f" - delegation_requests size: {len(self.state_manager.delegation_requests)}",
|
|
55
|
-
file=sys.stderr,
|
|
56
|
+
_log(f" - session_id: {session_id[:16] if session_id else 'None'}...")
|
|
57
|
+
_log(f" - event keys: {list(event.keys())}")
|
|
58
|
+
_log(
|
|
59
|
+
f" - delegation_requests size: {len(self.state_manager.delegation_requests)}"
|
|
56
60
|
)
|
|
57
61
|
# Show all stored session IDs for comparison
|
|
58
62
|
all_sessions = list(self.state_manager.delegation_requests.keys())
|
|
59
63
|
if all_sessions:
|
|
60
|
-
|
|
64
|
+
_log(" - Stored sessions (first 16 chars):")
|
|
61
65
|
for sid in all_sessions[:10]: # Show up to 10
|
|
62
|
-
|
|
63
|
-
f" - {sid[:16]}... (agent: {self.state_manager.delegation_requests[sid].get('agent_type', 'unknown')})"
|
|
64
|
-
file=sys.stderr,
|
|
66
|
+
_log(
|
|
67
|
+
f" - {sid[:16]}... (agent: {self.state_manager.delegation_requests[sid].get('agent_type', 'unknown')})"
|
|
65
68
|
)
|
|
66
69
|
else:
|
|
67
|
-
|
|
70
|
+
_log(" - No stored sessions in delegation_requests!")
|
|
68
71
|
|
|
69
72
|
# Get agent type and other basic info
|
|
70
73
|
agent_type, agent_id, reason, agent_type_inferred = self._extract_basic_info(
|
|
@@ -73,9 +76,8 @@ class SubagentResponseProcessor:
|
|
|
73
76
|
|
|
74
77
|
# Always log SubagentStop events for debugging
|
|
75
78
|
if DEBUG or agent_type != "unknown":
|
|
76
|
-
|
|
77
|
-
f"Hook handler: Processing SubagentStop - agent: '{agent_type}', session: '{session_id}', reason: '{reason}'"
|
|
78
|
-
file=sys.stderr,
|
|
79
|
+
_log(
|
|
80
|
+
f"Hook handler: Processing SubagentStop - agent: '{agent_type}', session: '{session_id}', reason: '{reason}'"
|
|
79
81
|
)
|
|
80
82
|
|
|
81
83
|
# Get working directory and git branch
|
|
@@ -115,9 +117,8 @@ class SubagentResponseProcessor:
|
|
|
115
117
|
|
|
116
118
|
# Debug log the processed data
|
|
117
119
|
if DEBUG:
|
|
118
|
-
|
|
119
|
-
f"SubagentStop processed data: agent_type='{agent_type}', session_id='{session_id}'"
|
|
120
|
-
file=sys.stderr,
|
|
120
|
+
_log(
|
|
121
|
+
f"SubagentStop processed data: agent_type='{agent_type}', session_id='{session_id}'"
|
|
121
122
|
)
|
|
122
123
|
|
|
123
124
|
# Emit to default namespace (consistent with subagent_start)
|
|
@@ -163,10 +164,7 @@ class SubagentResponseProcessor:
|
|
|
163
164
|
agent_type = "pm"
|
|
164
165
|
agent_type_inferred = True
|
|
165
166
|
if DEBUG:
|
|
166
|
-
|
|
167
|
-
" - Inferred agent_type='pm' (no explicit type found)",
|
|
168
|
-
file=sys.stderr,
|
|
169
|
-
)
|
|
167
|
+
_log(" - Inferred agent_type='pm' (no explicit type found)")
|
|
170
168
|
|
|
171
169
|
return agent_type, agent_id, reason, agent_type_inferred
|
|
172
170
|
|
|
@@ -182,17 +180,15 @@ class SubagentResponseProcessor:
|
|
|
182
180
|
if json_match:
|
|
183
181
|
structured_response = json.loads(json_match.group(1))
|
|
184
182
|
if DEBUG:
|
|
185
|
-
|
|
186
|
-
f"Extracted structured response from {agent_type} agent in SubagentStop"
|
|
187
|
-
file=sys.stderr,
|
|
183
|
+
_log(
|
|
184
|
+
f"Extracted structured response from {agent_type} agent in SubagentStop"
|
|
188
185
|
)
|
|
189
186
|
|
|
190
187
|
# Log if MEMORIES field is present
|
|
191
188
|
if structured_response.get("MEMORIES") and DEBUG:
|
|
192
189
|
memories_count = len(structured_response["MEMORIES"])
|
|
193
|
-
|
|
194
|
-
f"Agent {agent_type} returned MEMORIES field with {memories_count} items"
|
|
195
|
-
file=sys.stderr,
|
|
190
|
+
_log(
|
|
191
|
+
f"Agent {agent_type} returned MEMORIES field with {memories_count} items"
|
|
196
192
|
)
|
|
197
193
|
|
|
198
194
|
return structured_response
|
|
@@ -214,20 +210,15 @@ class SubagentResponseProcessor:
|
|
|
214
210
|
):
|
|
215
211
|
"""Track the agent response if response tracking is enabled."""
|
|
216
212
|
if DEBUG:
|
|
217
|
-
|
|
218
|
-
f" - response_tracking_enabled: {self.response_tracking_manager.response_tracking_enabled}"
|
|
219
|
-
file=sys.stderr,
|
|
220
|
-
)
|
|
221
|
-
print(
|
|
222
|
-
f" - response_tracker exists: {self.response_tracking_manager.response_tracker is not None}",
|
|
223
|
-
file=sys.stderr,
|
|
213
|
+
_log(
|
|
214
|
+
f" - response_tracking_enabled: {self.response_tracking_manager.response_tracking_enabled}"
|
|
224
215
|
)
|
|
225
|
-
|
|
226
|
-
f" -
|
|
227
|
-
file=sys.stderr,
|
|
216
|
+
_log(
|
|
217
|
+
f" - response_tracker exists: {self.response_tracking_manager.response_tracker is not None}"
|
|
228
218
|
)
|
|
229
|
-
|
|
230
|
-
|
|
219
|
+
_log(f" - session_id: {session_id[:16] if session_id else 'None'}...")
|
|
220
|
+
_log(f" - agent_type: {agent_type}")
|
|
221
|
+
_log(f" - reason: {reason}")
|
|
231
222
|
|
|
232
223
|
if (
|
|
233
224
|
self.response_tracking_manager.response_tracking_enabled
|
|
@@ -238,27 +229,16 @@ class SubagentResponseProcessor:
|
|
|
238
229
|
request_info = self.state_manager.find_matching_request(session_id)
|
|
239
230
|
|
|
240
231
|
if DEBUG:
|
|
241
|
-
|
|
242
|
-
f" - request_info present: {bool(request_info)}",
|
|
243
|
-
file=sys.stderr,
|
|
244
|
-
)
|
|
232
|
+
_log(f" - request_info present: {bool(request_info)}")
|
|
245
233
|
if request_info:
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
print(
|
|
251
|
-
f" - stored agent_type: {request_info.get('agent_type')}",
|
|
252
|
-
file=sys.stderr,
|
|
253
|
-
)
|
|
254
|
-
print(
|
|
255
|
-
f" - request keys: {list(request_info.get('request', {}).keys())}",
|
|
256
|
-
file=sys.stderr,
|
|
234
|
+
_log(" - ✅ Found request data for response tracking")
|
|
235
|
+
_log(f" - stored agent_type: {request_info.get('agent_type')}")
|
|
236
|
+
_log(
|
|
237
|
+
f" - request keys: {list(request_info.get('request', {}).keys())}"
|
|
257
238
|
)
|
|
258
239
|
else:
|
|
259
|
-
|
|
260
|
-
f" - ❌ No request data found for session {session_id[:16]}..."
|
|
261
|
-
file=sys.stderr,
|
|
240
|
+
_log(
|
|
241
|
+
f" - ❌ No request data found for session {session_id[:16]}..."
|
|
262
242
|
)
|
|
263
243
|
|
|
264
244
|
if request_info:
|
|
@@ -310,9 +290,8 @@ class SubagentResponseProcessor:
|
|
|
310
290
|
# Check for MEMORIES field and process if present
|
|
311
291
|
if structured_response.get("MEMORIES") and DEBUG:
|
|
312
292
|
memories = structured_response["MEMORIES"]
|
|
313
|
-
|
|
314
|
-
f"Found MEMORIES field in {agent_type} response with {len(memories)} items"
|
|
315
|
-
file=sys.stderr,
|
|
293
|
+
_log(
|
|
294
|
+
f"Found MEMORIES field in {agent_type} response with {len(memories)} items"
|
|
316
295
|
)
|
|
317
296
|
# The memory will be processed by extract_and_update_memory
|
|
318
297
|
# which is called by the memory hook service
|
|
@@ -329,26 +308,21 @@ class SubagentResponseProcessor:
|
|
|
329
308
|
)
|
|
330
309
|
|
|
331
310
|
if file_path and DEBUG:
|
|
332
|
-
|
|
333
|
-
f"✅ Tracked {agent_type} agent response on SubagentStop: {file_path.name}"
|
|
334
|
-
file=sys.stderr,
|
|
311
|
+
_log(
|
|
312
|
+
f"✅ Tracked {agent_type} agent response on SubagentStop: {file_path.name}"
|
|
335
313
|
)
|
|
336
314
|
|
|
337
315
|
# Clean up the request data
|
|
338
316
|
self.state_manager.remove_request(session_id)
|
|
339
317
|
|
|
340
318
|
elif DEBUG:
|
|
341
|
-
|
|
342
|
-
f"No request data for SubagentStop session {session_id[:8]}..., agent: {agent_type}"
|
|
343
|
-
file=sys.stderr,
|
|
319
|
+
_log(
|
|
320
|
+
f"No request data for SubagentStop session {session_id[:8]}..., agent: {agent_type}"
|
|
344
321
|
)
|
|
345
322
|
|
|
346
323
|
except Exception as e:
|
|
347
324
|
if DEBUG:
|
|
348
|
-
|
|
349
|
-
f"❌ Failed to track response on SubagentStop: {e}",
|
|
350
|
-
file=sys.stderr,
|
|
351
|
-
)
|
|
325
|
+
_log(f"❌ Failed to track response on SubagentStop: {e}")
|
|
352
326
|
|
|
353
327
|
def _build_subagent_stop_data(
|
|
354
328
|
self,
|
|
@@ -12,13 +12,21 @@ DESIGN DECISIONS:
|
|
|
12
12
|
"""
|
|
13
13
|
|
|
14
14
|
import json
|
|
15
|
-
import sys
|
|
16
15
|
from pathlib import Path
|
|
17
16
|
from typing import Any, Dict, Optional
|
|
18
17
|
|
|
19
18
|
from claude_mpm.core.logger import get_logger
|
|
20
19
|
from claude_mpm.services.cli.session_resume_helper import SessionResumeHelper
|
|
21
20
|
|
|
21
|
+
# Try to import _log from hook_handler, fall back to no-op
|
|
22
|
+
try:
|
|
23
|
+
from claude_mpm.hooks.claude_hooks.hook_handler import _log
|
|
24
|
+
except ImportError:
|
|
25
|
+
|
|
26
|
+
def _log(msg: str) -> None:
|
|
27
|
+
pass # Silent fallback
|
|
28
|
+
|
|
29
|
+
|
|
22
30
|
logger = get_logger(__name__)
|
|
23
31
|
|
|
24
32
|
|
|
@@ -86,23 +94,19 @@ class SessionResumeStartupHook:
|
|
|
86
94
|
Args:
|
|
87
95
|
pause_info: Pause session metadata from check_for_active_pause()
|
|
88
96
|
"""
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
)
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
)
|
|
101
|
-
|
|
102
|
-
print(" 1. Continue (actions will be appended)", file=sys.stderr)
|
|
103
|
-
print(" 2. Use /mpm-init pause --finalize to create snapshot", file=sys.stderr)
|
|
104
|
-
print(" 3. Use /mpm-init pause --discard to abandon", file=sys.stderr)
|
|
105
|
-
print("=" * 60 + "\n", file=sys.stderr)
|
|
97
|
+
_log("=" * 60)
|
|
98
|
+
_log("⚠️ ACTIVE AUTO-PAUSE SESSION DETECTED")
|
|
99
|
+
_log("=" * 60)
|
|
100
|
+
_log(f"Session ID: {pause_info['session_id']}")
|
|
101
|
+
_log(f"Started at: {pause_info['started_at']}")
|
|
102
|
+
_log(f"Context at pause: {pause_info['context_at_start']:.1%}")
|
|
103
|
+
_log(f"Actions recorded: {pause_info['action_count']}")
|
|
104
|
+
_log("\nThis session was auto-paused due to high context usage.")
|
|
105
|
+
_log("Options:")
|
|
106
|
+
_log(" 1. Continue (actions will be appended)")
|
|
107
|
+
_log(" 2. Use /mpm-init pause --finalize to create snapshot")
|
|
108
|
+
_log(" 3. Use /mpm-init pause --discard to abandon")
|
|
109
|
+
_log("=" * 60 + "\n")
|
|
106
110
|
|
|
107
111
|
def on_pm_startup(self) -> Optional[Dict[str, Any]]:
|
|
108
112
|
"""Execute on PM startup to check for paused sessions.
|
|
@@ -55,6 +55,14 @@ import sys
|
|
|
55
55
|
from pathlib import Path
|
|
56
56
|
from typing import Any, Dict, Optional
|
|
57
57
|
|
|
58
|
+
# Try to import _log from hook_handler, fall back to no-op
|
|
59
|
+
try:
|
|
60
|
+
from claude_mpm.hooks.claude_hooks.hook_handler import _log
|
|
61
|
+
except ImportError:
|
|
62
|
+
|
|
63
|
+
def _log(msg: str) -> None:
|
|
64
|
+
pass # Silent fallback
|
|
65
|
+
|
|
58
66
|
|
|
59
67
|
class PreToolUseHook:
|
|
60
68
|
"""Base class for PreToolUse hooks with input modification support."""
|
|
@@ -64,9 +72,9 @@ class PreToolUseHook:
|
|
|
64
72
|
self.debug = os.environ.get("CLAUDE_MPM_HOOK_DEBUG", "false").lower() == "true"
|
|
65
73
|
|
|
66
74
|
def log_debug(self, message: str) -> None:
|
|
67
|
-
"""Log debug message
|
|
75
|
+
"""Log debug message using _log helper."""
|
|
68
76
|
if self.debug:
|
|
69
|
-
|
|
77
|
+
_log(f"[PreToolUse Hook] {message}")
|
|
70
78
|
|
|
71
79
|
def read_event(self) -> Optional[Dict[str, Any]]:
|
|
72
80
|
"""Read and parse the hook event from stdin."""
|