claude-mpm 4.7.4__py3-none-any.whl → 4.18.2__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.
Files changed (308) hide show
  1. claude_mpm/VERSION +1 -1
  2. claude_mpm/agents/BASE_AGENT_TEMPLATE.md +118 -0
  3. claude_mpm/agents/BASE_ENGINEER.md +286 -0
  4. claude_mpm/agents/BASE_PM.md +106 -1
  5. claude_mpm/agents/OUTPUT_STYLE.md +329 -11
  6. claude_mpm/agents/PM_INSTRUCTIONS.md +397 -459
  7. claude_mpm/agents/agent_loader.py +17 -5
  8. claude_mpm/agents/frontmatter_validator.py +284 -253
  9. claude_mpm/agents/templates/README.md +465 -0
  10. claude_mpm/agents/templates/agent-manager.json +4 -1
  11. claude_mpm/agents/templates/agentic-coder-optimizer.json +13 -3
  12. claude_mpm/agents/templates/api_qa.json +11 -2
  13. claude_mpm/agents/templates/circuit_breakers.md +638 -0
  14. claude_mpm/agents/templates/clerk-ops.json +12 -2
  15. claude_mpm/agents/templates/code_analyzer.json +8 -2
  16. claude_mpm/agents/templates/content-agent.json +358 -0
  17. claude_mpm/agents/templates/dart_engineer.json +15 -2
  18. claude_mpm/agents/templates/data_engineer.json +15 -2
  19. claude_mpm/agents/templates/documentation.json +10 -2
  20. claude_mpm/agents/templates/engineer.json +21 -1
  21. claude_mpm/agents/templates/gcp_ops_agent.json +12 -2
  22. claude_mpm/agents/templates/git_file_tracking.md +584 -0
  23. claude_mpm/agents/templates/golang_engineer.json +270 -0
  24. claude_mpm/agents/templates/imagemagick.json +4 -1
  25. claude_mpm/agents/templates/java_engineer.json +346 -0
  26. claude_mpm/agents/templates/local_ops_agent.json +1227 -6
  27. claude_mpm/agents/templates/memory_manager.json +4 -1
  28. claude_mpm/agents/templates/nextjs_engineer.json +141 -133
  29. claude_mpm/agents/templates/ops.json +12 -2
  30. claude_mpm/agents/templates/php-engineer.json +270 -174
  31. claude_mpm/agents/templates/pm_examples.md +474 -0
  32. claude_mpm/agents/templates/pm_red_flags.md +240 -0
  33. claude_mpm/agents/templates/product_owner.json +338 -0
  34. claude_mpm/agents/templates/project_organizer.json +14 -4
  35. claude_mpm/agents/templates/prompt-engineer.json +13 -2
  36. claude_mpm/agents/templates/python_engineer.json +174 -81
  37. claude_mpm/agents/templates/qa.json +11 -2
  38. claude_mpm/agents/templates/react_engineer.json +16 -3
  39. claude_mpm/agents/templates/refactoring_engineer.json +12 -2
  40. claude_mpm/agents/templates/research.json +34 -21
  41. claude_mpm/agents/templates/response_format.md +583 -0
  42. claude_mpm/agents/templates/ruby-engineer.json +129 -192
  43. claude_mpm/agents/templates/rust_engineer.json +270 -0
  44. claude_mpm/agents/templates/security.json +10 -2
  45. claude_mpm/agents/templates/svelte-engineer.json +225 -0
  46. claude_mpm/agents/templates/ticketing.json +10 -2
  47. claude_mpm/agents/templates/typescript_engineer.json +116 -125
  48. claude_mpm/agents/templates/validation_templates.md +312 -0
  49. claude_mpm/agents/templates/vercel_ops_agent.json +12 -2
  50. claude_mpm/agents/templates/version_control.json +12 -2
  51. claude_mpm/agents/templates/web_qa.json +11 -2
  52. claude_mpm/agents/templates/web_ui.json +15 -2
  53. claude_mpm/cli/__init__.py +34 -614
  54. claude_mpm/cli/commands/agent_manager.py +25 -12
  55. claude_mpm/cli/commands/agent_state_manager.py +186 -0
  56. claude_mpm/cli/commands/agents.py +235 -148
  57. claude_mpm/cli/commands/agents_detect.py +380 -0
  58. claude_mpm/cli/commands/agents_recommend.py +309 -0
  59. claude_mpm/cli/commands/aggregate.py +7 -3
  60. claude_mpm/cli/commands/analyze.py +9 -4
  61. claude_mpm/cli/commands/analyze_code.py +7 -2
  62. claude_mpm/cli/commands/auto_configure.py +570 -0
  63. claude_mpm/cli/commands/config.py +47 -13
  64. claude_mpm/cli/commands/configure.py +419 -1571
  65. claude_mpm/cli/commands/configure_agent_display.py +261 -0
  66. claude_mpm/cli/commands/configure_behavior_manager.py +204 -0
  67. claude_mpm/cli/commands/configure_hook_manager.py +225 -0
  68. claude_mpm/cli/commands/configure_models.py +18 -0
  69. claude_mpm/cli/commands/configure_navigation.py +167 -0
  70. claude_mpm/cli/commands/configure_paths.py +104 -0
  71. claude_mpm/cli/commands/configure_persistence.py +254 -0
  72. claude_mpm/cli/commands/configure_startup_manager.py +646 -0
  73. claude_mpm/cli/commands/configure_template_editor.py +497 -0
  74. claude_mpm/cli/commands/configure_validators.py +73 -0
  75. claude_mpm/cli/commands/local_deploy.py +537 -0
  76. claude_mpm/cli/commands/memory.py +54 -20
  77. claude_mpm/cli/commands/mpm_init.py +585 -196
  78. claude_mpm/cli/commands/mpm_init_handler.py +37 -3
  79. claude_mpm/cli/commands/search.py +170 -4
  80. claude_mpm/cli/commands/upgrade.py +152 -0
  81. claude_mpm/cli/executor.py +202 -0
  82. claude_mpm/cli/helpers.py +105 -0
  83. claude_mpm/cli/interactive/__init__.py +3 -0
  84. claude_mpm/cli/interactive/skills_wizard.py +491 -0
  85. claude_mpm/cli/parsers/__init__.py +7 -1
  86. claude_mpm/cli/parsers/agents_parser.py +9 -0
  87. claude_mpm/cli/parsers/auto_configure_parser.py +245 -0
  88. claude_mpm/cli/parsers/base_parser.py +110 -3
  89. claude_mpm/cli/parsers/local_deploy_parser.py +227 -0
  90. claude_mpm/cli/parsers/mpm_init_parser.py +65 -5
  91. claude_mpm/cli/shared/output_formatters.py +28 -19
  92. claude_mpm/cli/startup.py +481 -0
  93. claude_mpm/cli/utils.py +52 -1
  94. claude_mpm/commands/mpm-agents-detect.md +168 -0
  95. claude_mpm/commands/mpm-agents-recommend.md +214 -0
  96. claude_mpm/commands/mpm-agents.md +75 -1
  97. claude_mpm/commands/mpm-auto-configure.md +217 -0
  98. claude_mpm/commands/mpm-help.md +163 -0
  99. claude_mpm/commands/mpm-init.md +148 -3
  100. claude_mpm/commands/mpm-version.md +113 -0
  101. claude_mpm/commands/mpm.md +1 -0
  102. claude_mpm/config/agent_config.py +2 -2
  103. claude_mpm/config/model_config.py +428 -0
  104. claude_mpm/constants.py +1 -0
  105. claude_mpm/core/base_service.py +13 -12
  106. claude_mpm/core/enums.py +452 -0
  107. claude_mpm/core/factories.py +1 -1
  108. claude_mpm/core/instruction_reinforcement_hook.py +2 -1
  109. claude_mpm/core/interactive_session.py +9 -3
  110. claude_mpm/core/log_manager.py +2 -0
  111. claude_mpm/core/logging_config.py +6 -2
  112. claude_mpm/core/oneshot_session.py +8 -4
  113. claude_mpm/core/optimized_agent_loader.py +3 -3
  114. claude_mpm/core/output_style_manager.py +12 -192
  115. claude_mpm/core/service_registry.py +5 -1
  116. claude_mpm/core/types.py +2 -9
  117. claude_mpm/core/typing_utils.py +7 -6
  118. claude_mpm/dashboard/static/js/dashboard.js +0 -14
  119. claude_mpm/dashboard/templates/index.html +3 -41
  120. claude_mpm/hooks/__init__.py +20 -0
  121. claude_mpm/hooks/claude_hooks/event_handlers.py +4 -2
  122. claude_mpm/hooks/claude_hooks/response_tracking.py +35 -1
  123. claude_mpm/hooks/claude_hooks/services/connection_manager_http.py +23 -2
  124. claude_mpm/hooks/failure_learning/__init__.py +60 -0
  125. claude_mpm/hooks/failure_learning/failure_detection_hook.py +235 -0
  126. claude_mpm/hooks/failure_learning/fix_detection_hook.py +217 -0
  127. claude_mpm/hooks/failure_learning/learning_extraction_hook.py +286 -0
  128. claude_mpm/hooks/instruction_reinforcement.py +7 -2
  129. claude_mpm/hooks/kuzu_enrichment_hook.py +263 -0
  130. claude_mpm/hooks/kuzu_memory_hook.py +37 -12
  131. claude_mpm/hooks/kuzu_response_hook.py +183 -0
  132. claude_mpm/models/resume_log.py +340 -0
  133. claude_mpm/services/agents/__init__.py +18 -5
  134. claude_mpm/services/agents/auto_config_manager.py +796 -0
  135. claude_mpm/services/agents/deployment/agent_configuration_manager.py +1 -1
  136. claude_mpm/services/agents/deployment/agent_record_service.py +1 -1
  137. claude_mpm/services/agents/deployment/agent_validator.py +17 -1
  138. claude_mpm/services/agents/deployment/async_agent_deployment.py +1 -1
  139. claude_mpm/services/agents/deployment/interface_adapter.py +3 -2
  140. claude_mpm/services/agents/deployment/local_template_deployment.py +1 -1
  141. claude_mpm/services/agents/deployment/pipeline/steps/agent_processing_step.py +7 -6
  142. claude_mpm/services/agents/deployment/pipeline/steps/base_step.py +7 -16
  143. claude_mpm/services/agents/deployment/pipeline/steps/configuration_step.py +4 -3
  144. claude_mpm/services/agents/deployment/pipeline/steps/target_directory_step.py +5 -3
  145. claude_mpm/services/agents/deployment/pipeline/steps/validation_step.py +6 -5
  146. claude_mpm/services/agents/deployment/refactored_agent_deployment_service.py +9 -6
  147. claude_mpm/services/agents/deployment/validation/__init__.py +3 -1
  148. claude_mpm/services/agents/deployment/validation/validation_result.py +1 -9
  149. claude_mpm/services/agents/local_template_manager.py +1 -1
  150. claude_mpm/services/agents/memory/agent_memory_manager.py +5 -2
  151. claude_mpm/services/agents/observers.py +547 -0
  152. claude_mpm/services/agents/recommender.py +568 -0
  153. claude_mpm/services/agents/registry/modification_tracker.py +5 -2
  154. claude_mpm/services/command_handler_service.py +11 -5
  155. claude_mpm/services/core/__init__.py +33 -1
  156. claude_mpm/services/core/interfaces/__init__.py +90 -3
  157. claude_mpm/services/core/interfaces/agent.py +184 -0
  158. claude_mpm/services/core/interfaces/health.py +172 -0
  159. claude_mpm/services/core/interfaces/model.py +281 -0
  160. claude_mpm/services/core/interfaces/process.py +372 -0
  161. claude_mpm/services/core/interfaces/project.py +121 -0
  162. claude_mpm/services/core/interfaces/restart.py +307 -0
  163. claude_mpm/services/core/interfaces/stability.py +260 -0
  164. claude_mpm/services/core/memory_manager.py +11 -24
  165. claude_mpm/services/core/models/__init__.py +79 -0
  166. claude_mpm/services/core/models/agent_config.py +381 -0
  167. claude_mpm/services/core/models/health.py +162 -0
  168. claude_mpm/services/core/models/process.py +235 -0
  169. claude_mpm/services/core/models/restart.py +302 -0
  170. claude_mpm/services/core/models/stability.py +264 -0
  171. claude_mpm/services/core/models/toolchain.py +306 -0
  172. claude_mpm/services/core/path_resolver.py +23 -7
  173. claude_mpm/services/diagnostics/__init__.py +2 -2
  174. claude_mpm/services/diagnostics/checks/agent_check.py +25 -24
  175. claude_mpm/services/diagnostics/checks/claude_code_check.py +24 -23
  176. claude_mpm/services/diagnostics/checks/common_issues_check.py +25 -24
  177. claude_mpm/services/diagnostics/checks/configuration_check.py +24 -23
  178. claude_mpm/services/diagnostics/checks/filesystem_check.py +18 -17
  179. claude_mpm/services/diagnostics/checks/installation_check.py +30 -29
  180. claude_mpm/services/diagnostics/checks/instructions_check.py +20 -19
  181. claude_mpm/services/diagnostics/checks/mcp_check.py +50 -36
  182. claude_mpm/services/diagnostics/checks/mcp_services_check.py +38 -33
  183. claude_mpm/services/diagnostics/checks/monitor_check.py +23 -22
  184. claude_mpm/services/diagnostics/checks/startup_log_check.py +9 -8
  185. claude_mpm/services/diagnostics/diagnostic_runner.py +6 -5
  186. claude_mpm/services/diagnostics/doctor_reporter.py +28 -25
  187. claude_mpm/services/diagnostics/models.py +19 -24
  188. claude_mpm/services/infrastructure/monitoring/__init__.py +1 -1
  189. claude_mpm/services/infrastructure/monitoring/aggregator.py +12 -12
  190. claude_mpm/services/infrastructure/monitoring/base.py +5 -13
  191. claude_mpm/services/infrastructure/monitoring/network.py +7 -6
  192. claude_mpm/services/infrastructure/monitoring/process.py +13 -12
  193. claude_mpm/services/infrastructure/monitoring/resources.py +7 -6
  194. claude_mpm/services/infrastructure/monitoring/service.py +16 -15
  195. claude_mpm/services/infrastructure/resume_log_generator.py +439 -0
  196. claude_mpm/services/local_ops/__init__.py +163 -0
  197. claude_mpm/services/local_ops/crash_detector.py +257 -0
  198. claude_mpm/services/local_ops/health_checks/__init__.py +28 -0
  199. claude_mpm/services/local_ops/health_checks/http_check.py +224 -0
  200. claude_mpm/services/local_ops/health_checks/process_check.py +236 -0
  201. claude_mpm/services/local_ops/health_checks/resource_check.py +255 -0
  202. claude_mpm/services/local_ops/health_manager.py +430 -0
  203. claude_mpm/services/local_ops/log_monitor.py +396 -0
  204. claude_mpm/services/local_ops/memory_leak_detector.py +294 -0
  205. claude_mpm/services/local_ops/process_manager.py +595 -0
  206. claude_mpm/services/local_ops/resource_monitor.py +331 -0
  207. claude_mpm/services/local_ops/restart_manager.py +401 -0
  208. claude_mpm/services/local_ops/restart_policy.py +387 -0
  209. claude_mpm/services/local_ops/state_manager.py +372 -0
  210. claude_mpm/services/local_ops/unified_manager.py +600 -0
  211. claude_mpm/services/mcp_config_manager.py +9 -4
  212. claude_mpm/services/mcp_gateway/core/__init__.py +1 -2
  213. claude_mpm/services/mcp_gateway/core/base.py +18 -31
  214. claude_mpm/services/mcp_gateway/main.py +30 -0
  215. claude_mpm/services/mcp_gateway/tools/external_mcp_services.py +206 -32
  216. claude_mpm/services/mcp_gateway/tools/health_check_tool.py +30 -28
  217. claude_mpm/services/mcp_gateway/tools/kuzu_memory_service.py +25 -5
  218. claude_mpm/services/mcp_service_verifier.py +1 -1
  219. claude_mpm/services/memory/failure_tracker.py +563 -0
  220. claude_mpm/services/memory_hook_service.py +165 -4
  221. claude_mpm/services/model/__init__.py +147 -0
  222. claude_mpm/services/model/base_provider.py +365 -0
  223. claude_mpm/services/model/claude_provider.py +412 -0
  224. claude_mpm/services/model/model_router.py +453 -0
  225. claude_mpm/services/model/ollama_provider.py +415 -0
  226. claude_mpm/services/monitor/daemon_manager.py +3 -2
  227. claude_mpm/services/monitor/handlers/dashboard.py +2 -1
  228. claude_mpm/services/monitor/handlers/hooks.py +2 -1
  229. claude_mpm/services/monitor/management/lifecycle.py +3 -2
  230. claude_mpm/services/monitor/server.py +2 -1
  231. claude_mpm/services/project/__init__.py +23 -0
  232. claude_mpm/services/project/detection_strategies.py +719 -0
  233. claude_mpm/services/project/toolchain_analyzer.py +581 -0
  234. claude_mpm/services/self_upgrade_service.py +342 -0
  235. claude_mpm/services/session_management_service.py +3 -2
  236. claude_mpm/services/session_manager.py +205 -1
  237. claude_mpm/services/shared/async_service_base.py +16 -27
  238. claude_mpm/services/shared/lifecycle_service_base.py +1 -14
  239. claude_mpm/services/socketio/handlers/__init__.py +5 -2
  240. claude_mpm/services/socketio/handlers/hook.py +13 -2
  241. claude_mpm/services/socketio/handlers/registry.py +4 -2
  242. claude_mpm/services/socketio/server/main.py +10 -8
  243. claude_mpm/services/subprocess_launcher_service.py +14 -5
  244. claude_mpm/services/unified/analyzer_strategies/code_analyzer.py +8 -7
  245. claude_mpm/services/unified/analyzer_strategies/dependency_analyzer.py +6 -5
  246. claude_mpm/services/unified/analyzer_strategies/performance_analyzer.py +8 -7
  247. claude_mpm/services/unified/analyzer_strategies/security_analyzer.py +7 -6
  248. claude_mpm/services/unified/analyzer_strategies/structure_analyzer.py +5 -4
  249. claude_mpm/services/unified/config_strategies/validation_strategy.py +13 -9
  250. claude_mpm/services/unified/deployment_strategies/cloud_strategies.py +10 -3
  251. claude_mpm/services/unified/deployment_strategies/local.py +6 -5
  252. claude_mpm/services/unified/deployment_strategies/utils.py +6 -5
  253. claude_mpm/services/unified/deployment_strategies/vercel.py +7 -6
  254. claude_mpm/services/unified/interfaces.py +3 -1
  255. claude_mpm/services/unified/unified_analyzer.py +14 -10
  256. claude_mpm/services/unified/unified_config.py +2 -1
  257. claude_mpm/services/unified/unified_deployment.py +9 -4
  258. claude_mpm/services/version_service.py +104 -1
  259. claude_mpm/skills/__init__.py +21 -0
  260. claude_mpm/skills/bundled/__init__.py +6 -0
  261. claude_mpm/skills/bundled/api-documentation.md +393 -0
  262. claude_mpm/skills/bundled/async-testing.md +571 -0
  263. claude_mpm/skills/bundled/code-review.md +143 -0
  264. claude_mpm/skills/bundled/database-migration.md +199 -0
  265. claude_mpm/skills/bundled/docker-containerization.md +194 -0
  266. claude_mpm/skills/bundled/express-local-dev.md +1429 -0
  267. claude_mpm/skills/bundled/fastapi-local-dev.md +1199 -0
  268. claude_mpm/skills/bundled/git-workflow.md +414 -0
  269. claude_mpm/skills/bundled/imagemagick.md +204 -0
  270. claude_mpm/skills/bundled/json-data-handling.md +223 -0
  271. claude_mpm/skills/bundled/nextjs-local-dev.md +807 -0
  272. claude_mpm/skills/bundled/pdf.md +141 -0
  273. claude_mpm/skills/bundled/performance-profiling.md +567 -0
  274. claude_mpm/skills/bundled/refactoring-patterns.md +180 -0
  275. claude_mpm/skills/bundled/security-scanning.md +327 -0
  276. claude_mpm/skills/bundled/systematic-debugging.md +473 -0
  277. claude_mpm/skills/bundled/test-driven-development.md +378 -0
  278. claude_mpm/skills/bundled/vite-local-dev.md +1061 -0
  279. claude_mpm/skills/bundled/web-performance-optimization.md +2305 -0
  280. claude_mpm/skills/bundled/xlsx.md +157 -0
  281. claude_mpm/skills/registry.py +286 -0
  282. claude_mpm/skills/skill_manager.py +310 -0
  283. claude_mpm/storage/state_storage.py +15 -15
  284. claude_mpm/tools/code_tree_analyzer.py +177 -141
  285. claude_mpm/tools/code_tree_events.py +4 -2
  286. claude_mpm/utils/agent_dependency_loader.py +40 -20
  287. claude_mpm/utils/display_helper.py +260 -0
  288. claude_mpm/utils/git_analyzer.py +407 -0
  289. claude_mpm/utils/robust_installer.py +73 -19
  290. {claude_mpm-4.7.4.dist-info → claude_mpm-4.18.2.dist-info}/METADATA +129 -12
  291. {claude_mpm-4.7.4.dist-info → claude_mpm-4.18.2.dist-info}/RECORD +295 -193
  292. claude_mpm/dashboard/static/css/code-tree.css +0 -1639
  293. claude_mpm/dashboard/static/index-hub-backup.html +0 -713
  294. claude_mpm/dashboard/static/js/components/code-tree/tree-breadcrumb.js +0 -353
  295. claude_mpm/dashboard/static/js/components/code-tree/tree-constants.js +0 -235
  296. claude_mpm/dashboard/static/js/components/code-tree/tree-search.js +0 -409
  297. claude_mpm/dashboard/static/js/components/code-tree/tree-utils.js +0 -435
  298. claude_mpm/dashboard/static/js/components/code-tree.js +0 -5869
  299. claude_mpm/dashboard/static/js/components/code-viewer.js +0 -1386
  300. claude_mpm/hooks/claude_hooks/hook_handler_eventbus.py +0 -425
  301. claude_mpm/hooks/claude_hooks/hook_handler_original.py +0 -1041
  302. claude_mpm/hooks/claude_hooks/hook_handler_refactored.py +0 -347
  303. claude_mpm/services/agents/deployment/agent_lifecycle_manager_refactored.py +0 -575
  304. claude_mpm/services/project/analyzer_refactored.py +0 -450
  305. {claude_mpm-4.7.4.dist-info → claude_mpm-4.18.2.dist-info}/WHEEL +0 -0
  306. {claude_mpm-4.7.4.dist-info → claude_mpm-4.18.2.dist-info}/entry_points.txt +0 -0
  307. {claude_mpm-4.7.4.dist-info → claude_mpm-4.18.2.dist-info}/licenses/LICENSE +0 -0
  308. {claude_mpm-4.7.4.dist-info → claude_mpm-4.18.2.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,183 @@
1
+ """
2
+ Kuzu-Memory Response Learning Hook
3
+ ===================================
4
+
5
+ Captures assistant responses and extracts learnings to store in kuzu-memory.
6
+ This completes the bidirectional enrichment cycle:
7
+ - KuzuMemoryHook enriches prompts with memories (READ)
8
+ - KuzuResponseHook stores learnings from responses (WRITE)
9
+
10
+ WHY: To automatically capture and persist important information from agent
11
+ responses, enabling continuous learning across conversations.
12
+
13
+ DESIGN DECISIONS:
14
+ - Priority 80 to run late after main processing
15
+ - Reuses KuzuMemoryHook's storage methods for consistency
16
+ - Graceful degradation if kuzu-memory is not available
17
+ - Extracts structured learnings using patterns and AI
18
+ """
19
+
20
+ import logging
21
+ from typing import Any, Optional
22
+
23
+ from claude_mpm.hooks.base_hook import (
24
+ HookContext,
25
+ HookResult,
26
+ PostDelegationHook,
27
+ )
28
+ from claude_mpm.hooks.kuzu_memory_hook import get_kuzu_memory_hook
29
+
30
+ logger = logging.getLogger(__name__)
31
+
32
+
33
+ class KuzuResponseHook(PostDelegationHook):
34
+ """
35
+ Hook that captures agent responses and stores learnings in kuzu-memory.
36
+
37
+ This hook:
38
+ 1. Processes agent responses after delegation completes
39
+ 2. Extracts important learnings and information
40
+ 3. Stores memories in kuzu-memory for future retrieval
41
+ 4. Tags memories for better categorization
42
+ """
43
+
44
+ def __init__(self):
45
+ """Initialize the kuzu-memory response learning hook."""
46
+ super().__init__(name="kuzu_response_learner", priority=80)
47
+
48
+ # Reuse the kuzu-memory hook instance for storage
49
+ self.kuzu_hook = get_kuzu_memory_hook()
50
+ self.enabled = self.kuzu_hook.enabled
51
+
52
+ if not self.enabled:
53
+ logger.info(
54
+ "Kuzu-memory response hook disabled (kuzu-memory not available)"
55
+ )
56
+ else:
57
+ logger.info("Kuzu-memory response learning hook enabled")
58
+
59
+ def validate(self, context: HookContext) -> bool:
60
+ """
61
+ Validate if hook should process this context.
62
+
63
+ Args:
64
+ context: Hook context to validate
65
+
66
+ Returns:
67
+ True if hook should execute
68
+ """
69
+ if not self.enabled:
70
+ return False
71
+
72
+ # Check base validation (enabled, correct hook type, has result)
73
+ if not super().validate(context):
74
+ return False
75
+
76
+ # Must have result data to extract learnings from
77
+ result_data = context.data.get("result")
78
+ if not result_data:
79
+ return False
80
+
81
+ return True
82
+
83
+ def execute(self, context: HookContext) -> HookResult:
84
+ """
85
+ Extract and store learnings from agent responses.
86
+
87
+ Args:
88
+ context: Hook context containing response data
89
+
90
+ Returns:
91
+ HookResult with success status and metadata
92
+ """
93
+ if not self.enabled:
94
+ return HookResult(success=True, data=context.data, modified=False)
95
+
96
+ try:
97
+ # Extract response content from various possible formats
98
+ result_data = context.data.get("result", {})
99
+ response_content = self._extract_response_content(result_data)
100
+
101
+ if not response_content:
102
+ logger.debug("No response content found for learning extraction")
103
+ return HookResult(success=True, data=context.data, modified=False)
104
+
105
+ # Extract and store learnings
106
+ count = self.kuzu_hook.extract_and_store_learnings(response_content)
107
+
108
+ if count > 0:
109
+ logger.info(f"Stored {count} learnings from agent response")
110
+ return HookResult(
111
+ success=True,
112
+ data=context.data,
113
+ modified=False,
114
+ metadata={"learnings_stored": count, "memory_backend": "kuzu"},
115
+ )
116
+
117
+ return HookResult(success=True, data=context.data, modified=False)
118
+
119
+ except Exception as e:
120
+ logger.error(f"Error in kuzu response hook: {e}")
121
+ # Don't fail the operation if learning extraction fails
122
+ return HookResult(
123
+ success=True,
124
+ data=context.data,
125
+ modified=False,
126
+ error=f"Learning extraction failed: {e}",
127
+ )
128
+
129
+ def _extract_response_content(self, result_data: Any) -> Optional[str]:
130
+ """
131
+ Extract response content from various result formats.
132
+
133
+ Args:
134
+ result_data: Result data in various possible formats
135
+
136
+ Returns:
137
+ Extracted response content as string, or None
138
+ """
139
+ if not result_data:
140
+ return None
141
+
142
+ # Handle dict format
143
+ if isinstance(result_data, dict):
144
+ # Try common response fields
145
+ for field in ["content", "text", "response", "output", "message"]:
146
+ if field in result_data:
147
+ content = result_data[field]
148
+ if isinstance(content, str):
149
+ return content
150
+ if isinstance(content, dict):
151
+ # Recursively extract from nested dict
152
+ return self._extract_response_content(content)
153
+
154
+ # If dict has no recognizable fields, try converting to string
155
+ return str(result_data)
156
+
157
+ # Handle string format
158
+ if isinstance(result_data, str):
159
+ return result_data
160
+
161
+ # Handle list format (concatenate items)
162
+ if isinstance(result_data, list):
163
+ items = []
164
+ for item in result_data:
165
+ extracted = self._extract_response_content(item)
166
+ if extracted:
167
+ items.append(extracted)
168
+ return "\n\n".join(items) if items else None
169
+
170
+ # Fallback to string conversion
171
+ return str(result_data) if result_data else None
172
+
173
+
174
+ # Create a singleton instance
175
+ _kuzu_response_hook = None
176
+
177
+
178
+ def get_kuzu_response_hook() -> KuzuResponseHook:
179
+ """Get the singleton kuzu-memory response hook instance."""
180
+ global _kuzu_response_hook
181
+ if _kuzu_response_hook is None:
182
+ _kuzu_response_hook = KuzuResponseHook()
183
+ return _kuzu_response_hook
@@ -0,0 +1,340 @@
1
+ """Resume Log Data Model.
2
+
3
+ This module defines the data structure for session resume logs that enable
4
+ seamless context restoration when Claude hits token limits.
5
+
6
+ Design Philosophy:
7
+ - Target 10k tokens maximum per resume log
8
+ - Human-readable markdown format
9
+ - Structured sections with token budgets
10
+ - Optimized for Claude consumption on session resume
11
+ """
12
+
13
+ from dataclasses import dataclass, field
14
+ from datetime import datetime, timezone
15
+ from pathlib import Path
16
+ from typing import Any, Dict, List, Optional
17
+
18
+ from claude_mpm.core.logging_utils import get_logger
19
+
20
+ logger = get_logger(__name__)
21
+
22
+
23
+ @dataclass
24
+ class ContextMetrics:
25
+ """Context window usage metrics."""
26
+
27
+ total_budget: int = 200000
28
+ used_tokens: int = 0
29
+ remaining_tokens: int = 0
30
+ percentage_used: float = 0.0
31
+ stop_reason: Optional[str] = None
32
+ model: str = "claude-sonnet-4.5"
33
+ session_id: str = ""
34
+ timestamp: str = field(
35
+ default_factory=lambda: datetime.now(timezone.utc).isoformat()
36
+ )
37
+
38
+ def to_dict(self) -> Dict[str, Any]:
39
+ """Convert to dictionary."""
40
+ return {
41
+ "total_budget": self.total_budget,
42
+ "used_tokens": self.used_tokens,
43
+ "remaining_tokens": self.remaining_tokens,
44
+ "percentage_used": self.percentage_used,
45
+ "stop_reason": self.stop_reason,
46
+ "model": self.model,
47
+ "session_id": self.session_id,
48
+ "timestamp": self.timestamp,
49
+ }
50
+
51
+ @classmethod
52
+ def from_dict(cls, data: Dict[str, Any]) -> "ContextMetrics":
53
+ """Create from dictionary."""
54
+ return cls(
55
+ total_budget=data.get("total_budget", 200000),
56
+ used_tokens=data.get("used_tokens", 0),
57
+ remaining_tokens=data.get("remaining_tokens", 0),
58
+ percentage_used=data.get("percentage_used", 0.0),
59
+ stop_reason=data.get("stop_reason"),
60
+ model=data.get("model", "claude-sonnet-4.5"),
61
+ session_id=data.get("session_id", ""),
62
+ timestamp=data.get("timestamp", datetime.now(timezone.utc).isoformat()),
63
+ )
64
+
65
+
66
+ @dataclass
67
+ class ResumeLog:
68
+ """Resume log containing all information needed to restore session context.
69
+
70
+ Token Budget Distribution (10k tokens total):
71
+ - Context Metrics: 500 tokens
72
+ - Mission Summary: 1,000 tokens
73
+ - Accomplishments: 2,000 tokens
74
+ - Key Findings: 2,500 tokens
75
+ - Decisions & Rationale: 1,500 tokens
76
+ - Next Steps: 1,500 tokens
77
+ - Critical Context: 1,000 tokens
78
+ """
79
+
80
+ # Session identification
81
+ session_id: str
82
+ previous_session_id: Optional[str] = None
83
+ created_at: str = field(
84
+ default_factory=lambda: datetime.now(timezone.utc).isoformat()
85
+ )
86
+
87
+ # Context metrics
88
+ context_metrics: ContextMetrics = field(default_factory=ContextMetrics)
89
+
90
+ # Core content sections (with token budgets)
91
+ mission_summary: str = "" # 1,000 tokens - What was the overall goal?
92
+ accomplishments: List[str] = field(
93
+ default_factory=list
94
+ ) # 2,000 tokens - What was completed?
95
+ key_findings: List[str] = field(
96
+ default_factory=list
97
+ ) # 2,500 tokens - What was discovered?
98
+ decisions_made: List[Dict[str, str]] = field(
99
+ default_factory=list
100
+ ) # 1,500 tokens - What choices were made and why?
101
+ next_steps: List[str] = field(
102
+ default_factory=list
103
+ ) # 1,500 tokens - What needs to happen next?
104
+ critical_context: Dict[str, Any] = field(
105
+ default_factory=dict
106
+ ) # 1,000 tokens - Essential state/data
107
+
108
+ # Metadata
109
+ files_modified: List[str] = field(default_factory=list)
110
+ agents_used: Dict[str, int] = field(default_factory=dict)
111
+ errors_encountered: List[str] = field(default_factory=list)
112
+ warnings: List[str] = field(default_factory=list)
113
+
114
+ def to_markdown(self) -> str:
115
+ """Generate markdown format for Claude consumption.
116
+
117
+ Returns:
118
+ Markdown-formatted resume log
119
+ """
120
+ sections = []
121
+
122
+ # Header
123
+ sections.append(f"# Session Resume Log: {self.session_id}\n")
124
+ sections.append(f"**Created**: {self.created_at}")
125
+ if self.previous_session_id:
126
+ sections.append(f"**Previous Session**: {self.previous_session_id}")
127
+ sections.append("")
128
+
129
+ # Context Metrics (500 tokens)
130
+ sections.append("## Context Metrics\n")
131
+ sections.append(f"- **Model**: {self.context_metrics.model}")
132
+ sections.append(
133
+ f"- **Tokens Used**: {self.context_metrics.used_tokens:,} / {self.context_metrics.total_budget:,}"
134
+ )
135
+ sections.append(
136
+ f"- **Percentage**: {self.context_metrics.percentage_used:.1f}%"
137
+ )
138
+ sections.append(
139
+ f"- **Remaining**: {self.context_metrics.remaining_tokens:,} tokens"
140
+ )
141
+ if self.context_metrics.stop_reason:
142
+ sections.append(f"- **Stop Reason**: {self.context_metrics.stop_reason}")
143
+ sections.append("")
144
+
145
+ # Mission Summary (1,000 tokens)
146
+ sections.append("## Mission Summary\n")
147
+ sections.append(
148
+ self.mission_summary
149
+ if self.mission_summary
150
+ else "_No mission summary provided_"
151
+ )
152
+ sections.append("")
153
+
154
+ # Accomplishments (2,000 tokens)
155
+ sections.append("## Accomplishments\n")
156
+ if self.accomplishments:
157
+ for i, item in enumerate(self.accomplishments, 1):
158
+ sections.append(f"{i}. {item}")
159
+ else:
160
+ sections.append("_No accomplishments recorded_")
161
+ sections.append("")
162
+
163
+ # Key Findings (2,500 tokens)
164
+ sections.append("## Key Findings\n")
165
+ if self.key_findings:
166
+ for i, finding in enumerate(self.key_findings, 1):
167
+ sections.append(f"{i}. {finding}")
168
+ else:
169
+ sections.append("_No key findings recorded_")
170
+ sections.append("")
171
+
172
+ # Decisions & Rationale (1,500 tokens)
173
+ sections.append("## Decisions & Rationale\n")
174
+ if self.decisions_made:
175
+ for i, decision in enumerate(self.decisions_made, 1):
176
+ decision_text = decision.get("decision", "")
177
+ rationale = decision.get("rationale", "")
178
+ sections.append(f"{i}. **Decision**: {decision_text}")
179
+ if rationale:
180
+ sections.append(f" **Rationale**: {rationale}")
181
+ else:
182
+ sections.append("_No decisions recorded_")
183
+ sections.append("")
184
+
185
+ # Next Steps (1,500 tokens)
186
+ sections.append("## Next Steps\n")
187
+ if self.next_steps:
188
+ for i, step in enumerate(self.next_steps, 1):
189
+ sections.append(f"{i}. {step}")
190
+ else:
191
+ sections.append("_No next steps defined_")
192
+ sections.append("")
193
+
194
+ # Critical Context (1,000 tokens)
195
+ sections.append("## Critical Context\n")
196
+ if self.critical_context:
197
+ for key, value in self.critical_context.items():
198
+ sections.append(f"- **{key}**: {value}")
199
+ else:
200
+ sections.append("_No critical context preserved_")
201
+ sections.append("")
202
+
203
+ # Metadata
204
+ sections.append("## Session Metadata\n")
205
+ if self.files_modified:
206
+ sections.append(f"**Files Modified** ({len(self.files_modified)}):")
207
+ for file in self.files_modified[:20]: # Limit to first 20
208
+ sections.append(f"- {file}")
209
+ if len(self.files_modified) > 20:
210
+ sections.append(f"- ... and {len(self.files_modified) - 20} more")
211
+ sections.append("")
212
+
213
+ if self.agents_used:
214
+ sections.append("**Agents Used**:")
215
+ for agent, count in self.agents_used.items():
216
+ sections.append(f"- {agent}: {count} delegations")
217
+ sections.append("")
218
+
219
+ if self.errors_encountered:
220
+ sections.append(f"**Errors** ({len(self.errors_encountered)}):")
221
+ for error in self.errors_encountered[:5]: # Limit to first 5
222
+ sections.append(f"- {error}")
223
+ sections.append("")
224
+
225
+ if self.warnings:
226
+ sections.append(f"**Warnings** ({len(self.warnings)}):")
227
+ for warning in self.warnings[:5]: # Limit to first 5
228
+ sections.append(f"- {warning}")
229
+ sections.append("")
230
+
231
+ return "\n".join(sections)
232
+
233
+ def to_dict(self) -> Dict[str, Any]:
234
+ """Convert to dictionary for JSON serialization."""
235
+ return {
236
+ "session_id": self.session_id,
237
+ "previous_session_id": self.previous_session_id,
238
+ "created_at": self.created_at,
239
+ "context_metrics": self.context_metrics.to_dict(),
240
+ "mission_summary": self.mission_summary,
241
+ "accomplishments": self.accomplishments,
242
+ "key_findings": self.key_findings,
243
+ "decisions_made": self.decisions_made,
244
+ "next_steps": self.next_steps,
245
+ "critical_context": self.critical_context,
246
+ "files_modified": self.files_modified,
247
+ "agents_used": self.agents_used,
248
+ "errors_encountered": self.errors_encountered,
249
+ "warnings": self.warnings,
250
+ }
251
+
252
+ @classmethod
253
+ def from_dict(cls, data: Dict[str, Any]) -> "ResumeLog":
254
+ """Create from dictionary."""
255
+ context_metrics_data = data.get("context_metrics", {})
256
+ context_metrics = ContextMetrics.from_dict(context_metrics_data)
257
+
258
+ return cls(
259
+ session_id=data.get("session_id", ""),
260
+ previous_session_id=data.get("previous_session_id"),
261
+ created_at=data.get("created_at", datetime.now(timezone.utc).isoformat()),
262
+ context_metrics=context_metrics,
263
+ mission_summary=data.get("mission_summary", ""),
264
+ accomplishments=data.get("accomplishments", []),
265
+ key_findings=data.get("key_findings", []),
266
+ decisions_made=data.get("decisions_made", []),
267
+ next_steps=data.get("next_steps", []),
268
+ critical_context=data.get("critical_context", {}),
269
+ files_modified=data.get("files_modified", []),
270
+ agents_used=data.get("agents_used", {}),
271
+ errors_encountered=data.get("errors_encountered", []),
272
+ warnings=data.get("warnings", []),
273
+ )
274
+
275
+ def save(self, storage_dir: Optional[Path] = None) -> Path:
276
+ """Save resume log to markdown file.
277
+
278
+ Args:
279
+ storage_dir: Directory to save the log (default: .claude-mpm/resume-logs)
280
+
281
+ Returns:
282
+ Path to saved file
283
+ """
284
+ if storage_dir is None:
285
+ storage_dir = Path.home() / ".claude-mpm" / "resume-logs"
286
+
287
+ storage_dir.mkdir(parents=True, exist_ok=True)
288
+
289
+ # Generate filename
290
+ file_path = storage_dir / f"session-{self.session_id}.md"
291
+
292
+ try:
293
+ # Write markdown file
294
+ markdown_content = self.to_markdown()
295
+ file_path.write_text(markdown_content, encoding="utf-8")
296
+
297
+ logger.info(f"Resume log saved: {file_path}")
298
+ return file_path
299
+
300
+ except Exception as e:
301
+ logger.error(f"Failed to save resume log: {e}")
302
+ raise
303
+
304
+ @classmethod
305
+ def load(
306
+ cls, session_id: str, storage_dir: Optional[Path] = None
307
+ ) -> Optional["ResumeLog"]:
308
+ """Load resume log from file.
309
+
310
+ Args:
311
+ session_id: Session ID to load
312
+ storage_dir: Directory to load from (default: .claude-mpm/resume-logs)
313
+
314
+ Returns:
315
+ ResumeLog instance or None if not found
316
+ """
317
+ if storage_dir is None:
318
+ storage_dir = Path.home() / ".claude-mpm" / "resume-logs"
319
+
320
+ file_path = storage_dir / f"session-{session_id}.md"
321
+
322
+ if not file_path.exists():
323
+ logger.debug(f"Resume log not found: {file_path}")
324
+ return None
325
+
326
+ try:
327
+ # For now, we just return the markdown content
328
+ # In the future, could parse markdown back to structured data
329
+ _ = file_path.read_text(encoding="utf-8")
330
+ logger.info(f"Resume log loaded: {file_path}")
331
+
332
+ # Return a basic ResumeLog with the markdown content embedded
333
+ return cls(
334
+ session_id=session_id,
335
+ mission_summary=f"Loaded from previous session. See full context in {file_path}",
336
+ )
337
+
338
+ except Exception as e:
339
+ logger.error(f"Failed to load resume log: {e}")
340
+ return None
@@ -1,5 +1,8 @@
1
1
  """Agent services module - hierarchical organization of agent-related services."""
2
2
 
3
+ # Auto-configuration exports
4
+ from .auto_config_manager import AutoConfigManagerService
5
+
3
6
  # Registry exports
4
7
  # Deployment exports
5
8
  from .deployment.agent_deployment import AgentDeploymentService
@@ -29,6 +32,15 @@ from .memory.agent_persistence_service import (
29
32
  PersistenceRecord,
30
33
  PersistenceStrategy,
31
34
  )
35
+ from .observers import (
36
+ CompositeObserver,
37
+ ConsoleProgressObserver,
38
+ IDeploymentObserver,
39
+ NullObserver,
40
+ )
41
+
42
+ # Recommender exports
43
+ from .recommender import AgentRecommenderService
32
44
  from .registry import AgentMetadata, AgentRegistry, AgentTier, AgentType
33
45
  from .registry.deployed_agent_discovery import DeployedAgentDiscovery
34
46
  from .registry.modification_tracker import (
@@ -41,34 +53,35 @@ from .registry.modification_tracker import (
41
53
 
42
54
  __all__ = [
43
55
  "AgentCapabilitiesGenerator",
44
- # Deployment
45
56
  "AgentDeploymentService",
46
57
  "AgentLifecycleManager",
47
58
  "AgentLifecycleRecord",
48
- # Management
49
59
  "AgentManager",
50
- # Memory
51
60
  "AgentMemoryManager",
52
61
  "AgentMetadata",
53
62
  "AgentModification",
54
63
  "AgentModificationTracker",
55
64
  "AgentPersistenceService",
56
65
  "AgentProfileLoader",
57
- # Registry
66
+ "AgentRecommenderService",
58
67
  "AgentRegistry",
59
68
  "AgentTier",
60
69
  "AgentType",
61
70
  "AgentVersionManager",
71
+ "AutoConfigManagerService",
62
72
  "BaseAgentManager",
73
+ "CompositeObserver",
74
+ "ConsoleProgressObserver",
63
75
  "DeployedAgentDiscovery",
64
- # Loading
65
76
  "FrameworkAgentLoader",
77
+ "IDeploymentObserver",
66
78
  "LifecycleOperation",
67
79
  "LifecycleOperationResult",
68
80
  "LifecycleState",
69
81
  "ModificationHistory",
70
82
  "ModificationTier",
71
83
  "ModificationType",
84
+ "NullObserver",
72
85
  "PersistenceOperation",
73
86
  "PersistenceRecord",
74
87
  "PersistenceStrategy",