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,310 @@
1
+ """Skills manager - integrates skills with agents."""
2
+
3
+ import json
4
+ from pathlib import Path
5
+ from typing import Dict, List, Optional
6
+
7
+ from claude_mpm.core.logging_utils import get_logger
8
+
9
+ from .registry import Skill, get_registry
10
+
11
+ logger = get_logger(__name__)
12
+
13
+
14
+ class SkillManager:
15
+ """Manages skills and their integration with agents."""
16
+
17
+ def __init__(self):
18
+ """Initialize the skill manager."""
19
+ self.registry = get_registry()
20
+ self.agent_skill_mapping: Dict[str, List[str]] = {}
21
+ self._load_agent_mappings()
22
+
23
+ def _load_agent_mappings(self):
24
+ """Load skill mappings from agent templates."""
25
+ # Load mappings from agent JSON templates that have 'skills' field
26
+ agent_templates_dir = Path(__file__).parent.parent / "agents" / "templates"
27
+
28
+ if not agent_templates_dir.exists():
29
+ logger.warning(
30
+ f"Agent templates directory not found: {agent_templates_dir}"
31
+ )
32
+ return
33
+
34
+ mapping_count = 0
35
+ for template_file in agent_templates_dir.glob("*.json"):
36
+ try:
37
+ with open(template_file, encoding="utf-8") as f:
38
+ agent_data = json.load(f)
39
+
40
+ agent_id = agent_data.get("agent_id") or agent_data.get("agent_type")
41
+ if not agent_id:
42
+ continue
43
+
44
+ # Extract skills list if present
45
+ skills = agent_data.get("skills", [])
46
+ if skills:
47
+ self.agent_skill_mapping[agent_id] = skills
48
+ mapping_count += 1
49
+ logger.debug(
50
+ f"Agent '{agent_id}' mapped to {len(skills)} skills: {', '.join(skills)}"
51
+ )
52
+
53
+ except Exception as e:
54
+ logger.error(f"Error loading agent mapping from {template_file}: {e}")
55
+
56
+ if mapping_count > 0:
57
+ logger.info(f"Loaded skill mappings for {mapping_count} agents")
58
+
59
+ def get_agent_skills(self, agent_type: str) -> List[Skill]:
60
+ """
61
+ Get all skills for an agent (bundled + discovered).
62
+
63
+ Args:
64
+ agent_type: Agent type/ID (e.g., 'engineer', 'python_engineer')
65
+
66
+ Returns:
67
+ List of Skill objects for this agent
68
+ """
69
+ skill_names = self.agent_skill_mapping.get(agent_type, [])
70
+
71
+ # Get skills from registry
72
+ skills = []
73
+ for name in skill_names:
74
+ skill = self.registry.get_skill(name)
75
+ if skill:
76
+ skills.append(skill)
77
+ else:
78
+ logger.warning(
79
+ f"Skill '{name}' referenced by agent '{agent_type}' not found"
80
+ )
81
+
82
+ # Also include skills that have no agent restriction
83
+ # or explicitly list this agent type
84
+ additional_skills = self.registry.get_skills_for_agent(agent_type)
85
+ for skill in additional_skills:
86
+ if skill not in skills:
87
+ skills.append(skill)
88
+
89
+ return skills
90
+
91
+ def enhance_agent_prompt(
92
+ self, agent_type: str, base_prompt: str, include_all: bool = False
93
+ ) -> str:
94
+ """
95
+ Enhance agent prompt with available skills.
96
+
97
+ Args:
98
+ agent_type: Agent type/ID
99
+ base_prompt: Original agent prompt
100
+ include_all: If True, include all available skills regardless of mapping
101
+
102
+ Returns:
103
+ Enhanced prompt with skills section appended
104
+ """
105
+ if include_all:
106
+ skills = self.registry.list_skills()
107
+ else:
108
+ skills = self.get_agent_skills(agent_type)
109
+
110
+ if not skills:
111
+ return base_prompt
112
+
113
+ # Build skills section
114
+ skills_section = "\n\n" + "=" * 80 + "\n"
115
+ skills_section += "## 🎯 Available Skills\n\n"
116
+ skills_section += f"You have access to {len(skills)} specialized skills:\n\n"
117
+
118
+ for skill in skills:
119
+ skills_section += f"### 📚 {skill.name.replace('-', ' ').title()}\n\n"
120
+ skills_section += f"**Source:** {skill.source}\n"
121
+ if skill.description:
122
+ skills_section += f"**Description:** {skill.description}\n"
123
+ skills_section += "\n```\n"
124
+ skills_section += skill.content
125
+ skills_section += "\n```\n\n"
126
+
127
+ skills_section += "=" * 80 + "\n"
128
+
129
+ return base_prompt + skills_section
130
+
131
+ def list_agent_skill_mappings(self) -> Dict[str, List[str]]:
132
+ """
133
+ Get all agent-to-skill mappings.
134
+
135
+ Returns:
136
+ Dictionary mapping agent IDs to lists of skill names
137
+ """
138
+ return self.agent_skill_mapping.copy()
139
+
140
+ def add_skill_to_agent(self, agent_type: str, skill_name: str) -> bool:
141
+ """
142
+ Add a skill to an agent's mapping.
143
+
144
+ Args:
145
+ agent_type: Agent type/ID
146
+ skill_name: Name of the skill to add
147
+
148
+ Returns:
149
+ True if successful, False if skill not found
150
+ """
151
+ skill = self.registry.get_skill(skill_name)
152
+ if not skill:
153
+ logger.error(f"Cannot add skill '{skill_name}': skill not found")
154
+ return False
155
+
156
+ if agent_type not in self.agent_skill_mapping:
157
+ self.agent_skill_mapping[agent_type] = []
158
+
159
+ if skill_name not in self.agent_skill_mapping[agent_type]:
160
+ self.agent_skill_mapping[agent_type].append(skill_name)
161
+ logger.info(f"Added skill '{skill_name}' to agent '{agent_type}'")
162
+
163
+ return True
164
+
165
+ def remove_skill_from_agent(self, agent_type: str, skill_name: str) -> bool:
166
+ """
167
+ Remove a skill from an agent's mapping.
168
+
169
+ Args:
170
+ agent_type: Agent type/ID
171
+ skill_name: Name of the skill to remove
172
+
173
+ Returns:
174
+ True if successful, False if not found
175
+ """
176
+ if agent_type not in self.agent_skill_mapping:
177
+ return False
178
+
179
+ if skill_name in self.agent_skill_mapping[agent_type]:
180
+ self.agent_skill_mapping[agent_type].remove(skill_name)
181
+ logger.info(f"Removed skill '{skill_name}' from agent '{agent_type}'")
182
+ return True
183
+
184
+ return False
185
+
186
+ def reload(self):
187
+ """Reload skills and agent mappings."""
188
+ logger.info("Reloading skill manager...")
189
+ self.registry.reload()
190
+ self.agent_skill_mapping.clear()
191
+ self._load_agent_mappings()
192
+ logger.info("Skill manager reloaded")
193
+
194
+ def infer_agents_for_skill(self, skill) -> List[str]:
195
+ """Infer which agents should have this skill based on tags/name.
196
+
197
+ Args:
198
+ skill: Skill object to analyze
199
+
200
+ Returns:
201
+ List of agent IDs that should have this skill
202
+ """
203
+ agents = []
204
+ content_lower = skill.content.lower()
205
+ name_lower = skill.name.lower()
206
+
207
+ # Python-related
208
+ if any(
209
+ tag in content_lower or tag in name_lower
210
+ for tag in ["python", "django", "flask", "fastapi"]
211
+ ):
212
+ agents.append("python-engineer")
213
+
214
+ # TypeScript/JavaScript-related
215
+ if any(
216
+ tag in content_lower or tag in name_lower
217
+ for tag in ["typescript", "javascript", "react", "next", "vue", "node"]
218
+ ):
219
+ agents.extend(["typescript-engineer", "react-engineer", "nextjs-engineer"])
220
+
221
+ # Go-related
222
+ if any(tag in content_lower or tag in name_lower for tag in ["golang", "go "]):
223
+ agents.append("golang-engineer")
224
+
225
+ # Ops-related
226
+ if any(
227
+ tag in content_lower or tag in name_lower
228
+ for tag in ["docker", "kubernetes", "deploy", "devops", "ops"]
229
+ ):
230
+ agents.extend(["ops", "devops", "local-ops"])
231
+
232
+ # Testing/QA-related
233
+ if any(
234
+ tag in content_lower or tag in name_lower
235
+ for tag in ["test", "qa", "quality", "assert"]
236
+ ):
237
+ agents.extend(["qa", "web-qa", "api-qa"])
238
+
239
+ # Documentation-related
240
+ if any(
241
+ tag in content_lower or tag in name_lower
242
+ for tag in ["documentation", "docs", "api doc", "openapi"]
243
+ ):
244
+ agents.extend(["docs", "documentation", "technical-writer"])
245
+
246
+ # Remove duplicates
247
+ return list(set(agents))
248
+
249
+ def save_mappings_to_config(self, config_path: Optional[Path] = None):
250
+ """Save current agent-skill mappings to configuration file.
251
+
252
+ Args:
253
+ config_path: Path to configuration file. If None, uses default.
254
+ """
255
+ import json
256
+
257
+ if config_path is None:
258
+ config_path = Path.cwd() / ".claude-mpm" / "skills_config.json"
259
+
260
+ # Ensure directory exists
261
+ config_path.parent.mkdir(parents=True, exist_ok=True)
262
+
263
+ # Save mappings
264
+ with open(config_path, "w", encoding="utf-8") as f:
265
+ json.dump(self.agent_skill_mapping, f, indent=2)
266
+
267
+ logger.info(f"Saved skill mappings to {config_path}")
268
+
269
+ def load_mappings_from_config(self, config_path: Optional[Path] = None):
270
+ """Load agent-skill mappings from configuration file.
271
+
272
+ Args:
273
+ config_path: Path to configuration file. If None, uses default.
274
+ """
275
+ import json
276
+
277
+ if config_path is None:
278
+ config_path = Path.cwd() / ".claude-mpm" / "skills_config.json"
279
+
280
+ if not config_path.exists():
281
+ logger.debug(f"No skill mappings config found at {config_path}")
282
+ return
283
+
284
+ try:
285
+ with open(config_path, encoding="utf-8") as f:
286
+ loaded_mappings = json.load(f)
287
+
288
+ # Merge with existing mappings
289
+ for agent_id, skills in loaded_mappings.items():
290
+ if agent_id not in self.agent_skill_mapping:
291
+ self.agent_skill_mapping[agent_id] = []
292
+ for skill in skills:
293
+ if skill not in self.agent_skill_mapping[agent_id]:
294
+ self.agent_skill_mapping[agent_id].append(skill)
295
+
296
+ logger.info(f"Loaded skill mappings from {config_path}")
297
+ except Exception as e:
298
+ logger.error(f"Error loading skill mappings from {config_path}: {e}")
299
+
300
+
301
+ # Global manager instance (singleton pattern)
302
+ _manager: Optional[SkillManager] = None
303
+
304
+
305
+ def get_manager() -> SkillManager:
306
+ """Get the global skill manager (singleton)."""
307
+ global _manager
308
+ if _manager is None:
309
+ _manager = SkillManager()
310
+ return _manager
@@ -86,7 +86,7 @@ class StateStorage:
86
86
  return True
87
87
 
88
88
  except Exception as e:
89
- self.logger.error(f"Failed to write JSON to {file_path}: {e}")
89
+ logger.error(f"Failed to write JSON to {file_path}: {e}")
90
90
  self.error_count += 1
91
91
  return False
92
92
 
@@ -106,7 +106,7 @@ class StateStorage:
106
106
  file_path = Path(file_path)
107
107
 
108
108
  if not file_path.exists():
109
- self.logger.debug(f"File not found: {file_path}")
109
+ logger.debug(f"File not found: {file_path}")
110
110
  return None
111
111
 
112
112
  # Auto-detect compression
@@ -125,7 +125,7 @@ class StateStorage:
125
125
  return data
126
126
 
127
127
  except Exception as e:
128
- self.logger.error(f"Failed to read JSON from {file_path}: {e}")
128
+ logger.error(f"Failed to read JSON from {file_path}: {e}")
129
129
  self.error_count += 1
130
130
  return None
131
131
 
@@ -166,7 +166,7 @@ class StateStorage:
166
166
  return True
167
167
 
168
168
  except Exception as e:
169
- self.logger.error(f"Failed to write pickle to {file_path}: {e}")
169
+ logger.error(f"Failed to write pickle to {file_path}: {e}")
170
170
  self.error_count += 1
171
171
  return False
172
172
 
@@ -186,7 +186,7 @@ class StateStorage:
186
186
  file_path = Path(file_path)
187
187
 
188
188
  if not file_path.exists():
189
- self.logger.debug(f"File not found: {file_path}")
189
+ logger.debug(f"File not found: {file_path}")
190
190
  return None
191
191
 
192
192
  # Auto-detect compression
@@ -205,7 +205,7 @@ class StateStorage:
205
205
  return data
206
206
 
207
207
  except Exception as e:
208
- self.logger.error(f"Failed to read pickle from {file_path}: {e}")
208
+ logger.error(f"Failed to read pickle from {file_path}: {e}")
209
209
  self.error_count += 1
210
210
  return None
211
211
 
@@ -262,7 +262,7 @@ class StateStorage:
262
262
  Path(temp_path).replace(file_path)
263
263
 
264
264
  self.write_count += 1
265
- self.logger.debug(f"Atomic write successful: {file_path}")
265
+ logger.debug(f"Atomic write successful: {file_path}")
266
266
  return True
267
267
 
268
268
  finally:
@@ -271,7 +271,7 @@ class StateStorage:
271
271
  Path(temp_path).unlink()
272
272
 
273
273
  except Exception as e:
274
- self.logger.error(f"Atomic write failed for {file_path}: {e}")
274
+ logger.error(f"Atomic write failed for {file_path}: {e}")
275
275
  self.error_count += 1
276
276
  return False
277
277
 
@@ -339,7 +339,7 @@ class StateStorage:
339
339
  f.write(checksum)
340
340
 
341
341
  except Exception as e:
342
- self.logger.warning(f"Could not add checksum: {e}")
342
+ logger.warning(f"Could not add checksum: {e}")
343
343
 
344
344
  def verify_checksum(self, file_path: Union[str, Path]) -> bool:
345
345
  """Verify file checksum for integrity.
@@ -370,13 +370,13 @@ class StateStorage:
370
370
  actual = hasher.hexdigest()
371
371
 
372
372
  if actual != expected:
373
- self.logger.error(f"Checksum mismatch for {file_path}")
373
+ logger.error(f"Checksum mismatch for {file_path}")
374
374
  return False
375
375
 
376
376
  return True
377
377
 
378
378
  except Exception as e:
379
- self.logger.warning(f"Could not verify checksum: {e}")
379
+ logger.warning(f"Could not verify checksum: {e}")
380
380
  return True # Assume valid if can't verify
381
381
 
382
382
  def cleanup_temp_files(self) -> int:
@@ -396,7 +396,7 @@ class StateStorage:
396
396
  if age > 3600:
397
397
  temp_file.unlink()
398
398
  cleaned += 1
399
- self.logger.debug(f"Cleaned up temp file: {temp_file}")
399
+ logger.debug(f"Cleaned up temp file: {temp_file}")
400
400
  except Exception:
401
401
  pass
402
402
 
@@ -408,12 +408,12 @@ class StateStorage:
408
408
  cleaned += 1
409
409
 
410
410
  if cleaned > 0:
411
- self.logger.info(f"Cleaned up {cleaned} temporary files")
411
+ logger.info(f"Cleaned up {cleaned} temporary files")
412
412
 
413
413
  return cleaned
414
414
 
415
415
  except Exception as e:
416
- self.logger.error(f"Error cleaning up temp files: {e}")
416
+ logger.error(f"Error cleaning up temp files: {e}")
417
417
  return 0
418
418
 
419
419
  def get_storage_info(self) -> Dict[str, Any]:
@@ -447,7 +447,7 @@ class StateStorage:
447
447
  }
448
448
 
449
449
  except Exception as e:
450
- self.logger.error(f"Error getting storage info: {e}")
450
+ logger.error(f"Error getting storage info: {e}")
451
451
  return {"storage_directory": str(self.storage_dir), "error": str(e)}
452
452
 
453
453