claude-mpm 5.1.9__py3-none-any.whl → 5.4.22__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Potentially problematic release.
This version of claude-mpm might be problematic. Click here for more details.
- claude_mpm/VERSION +1 -1
- claude_mpm/__init__.py +4 -0
- claude_mpm/agents/CLAUDE_MPM_TEACHER_OUTPUT_STYLE.md +1 -1
- claude_mpm/agents/PM_INSTRUCTIONS.md +290 -34
- claude_mpm/agents/agent_loader.py +13 -44
- claude_mpm/agents/templates/circuit-breakers.md +138 -1
- claude_mpm/cli/__main__.py +4 -0
- claude_mpm/cli/chrome_devtools_installer.py +175 -0
- claude_mpm/cli/commands/agent_state_manager.py +8 -17
- claude_mpm/cli/commands/agents.py +0 -31
- claude_mpm/cli/commands/auto_configure.py +210 -25
- claude_mpm/cli/commands/config.py +88 -2
- claude_mpm/cli/commands/configure.py +1097 -158
- claude_mpm/cli/commands/configure_agent_display.py +15 -6
- claude_mpm/cli/commands/mpm_init/core.py +160 -46
- claude_mpm/cli/commands/mpm_init/knowledge_extractor.py +481 -0
- claude_mpm/cli/commands/mpm_init/prompts.py +280 -0
- claude_mpm/cli/commands/skills.py +214 -189
- claude_mpm/cli/commands/summarize.py +413 -0
- claude_mpm/cli/executor.py +11 -3
- claude_mpm/cli/parsers/agents_parser.py +0 -9
- claude_mpm/cli/parsers/auto_configure_parser.py +0 -138
- claude_mpm/cli/parsers/base_parser.py +5 -0
- claude_mpm/cli/parsers/config_parser.py +153 -83
- claude_mpm/cli/parsers/skills_parser.py +3 -2
- claude_mpm/cli/startup.py +550 -94
- claude_mpm/commands/mpm-config.md +265 -0
- claude_mpm/commands/mpm-help.md +14 -95
- claude_mpm/commands/mpm-organize.md +500 -0
- claude_mpm/config/agent_sources.py +27 -0
- claude_mpm/core/framework/formatters/content_formatter.py +3 -13
- claude_mpm/core/framework/loaders/agent_loader.py +8 -5
- claude_mpm/core/framework_loader.py +4 -2
- claude_mpm/core/logger.py +13 -0
- claude_mpm/core/socketio_pool.py +3 -3
- claude_mpm/core/unified_agent_registry.py +5 -15
- claude_mpm/hooks/claude_hooks/correlation_manager.py +60 -0
- claude_mpm/hooks/claude_hooks/event_handlers.py +211 -78
- claude_mpm/hooks/claude_hooks/hook_handler.py +6 -0
- claude_mpm/hooks/claude_hooks/installer.py +33 -10
- claude_mpm/hooks/claude_hooks/memory_integration.py +26 -9
- claude_mpm/hooks/claude_hooks/response_tracking.py +2 -3
- claude_mpm/hooks/claude_hooks/services/connection_manager.py +4 -0
- claude_mpm/hooks/memory_integration_hook.py +46 -1
- claude_mpm/init.py +0 -19
- claude_mpm/scripts/claude-hook-handler.sh +58 -18
- claude_mpm/scripts/launch_monitor.py +93 -13
- claude_mpm/scripts/start_activity_logging.py +0 -0
- claude_mpm/services/agents/agent_recommendation_service.py +278 -0
- claude_mpm/services/agents/agent_review_service.py +280 -0
- claude_mpm/services/agents/deployment/agent_discovery_service.py +2 -3
- claude_mpm/services/agents/deployment/agent_template_builder.py +4 -2
- claude_mpm/services/agents/deployment/multi_source_deployment_service.py +78 -9
- claude_mpm/services/agents/deployment/remote_agent_discovery_service.py +335 -53
- claude_mpm/services/agents/git_source_manager.py +34 -0
- claude_mpm/services/agents/loading/base_agent_manager.py +1 -13
- claude_mpm/services/agents/sources/git_source_sync_service.py +8 -1
- claude_mpm/services/agents/toolchain_detector.py +10 -6
- claude_mpm/services/analysis/__init__.py +11 -1
- claude_mpm/services/analysis/clone_detector.py +1030 -0
- claude_mpm/services/command_deployment_service.py +81 -10
- claude_mpm/services/event_bus/config.py +3 -1
- claude_mpm/services/git/git_operations_service.py +93 -8
- claude_mpm/services/monitor/daemon.py +9 -2
- claude_mpm/services/monitor/daemon_manager.py +39 -3
- claude_mpm/services/monitor/server.py +225 -19
- claude_mpm/services/self_upgrade_service.py +120 -12
- claude_mpm/services/skills/__init__.py +3 -0
- claude_mpm/services/skills/git_skill_source_manager.py +32 -2
- claude_mpm/services/skills/selective_skill_deployer.py +704 -0
- claude_mpm/services/skills/skill_to_agent_mapper.py +406 -0
- claude_mpm/services/skills_deployer.py +126 -9
- claude_mpm/services/socketio/event_normalizer.py +15 -1
- claude_mpm/services/socketio/server/core.py +160 -21
- claude_mpm/services/version_control/git_operations.py +103 -0
- claude_mpm/utils/agent_filters.py +17 -44
- {claude_mpm-5.1.9.dist-info → claude_mpm-5.4.22.dist-info}/METADATA +47 -84
- {claude_mpm-5.1.9.dist-info → claude_mpm-5.4.22.dist-info}/RECORD +82 -161
- claude_mpm-5.4.22.dist-info/entry_points.txt +5 -0
- claude_mpm-5.4.22.dist-info/licenses/LICENSE +94 -0
- claude_mpm-5.4.22.dist-info/licenses/LICENSE-FAQ.md +153 -0
- claude_mpm/agents/BASE_AGENT_TEMPLATE.md +0 -292
- claude_mpm/agents/BASE_DOCUMENTATION.md +0 -53
- claude_mpm/agents/BASE_ENGINEER.md +0 -658
- claude_mpm/agents/BASE_OPS.md +0 -219
- claude_mpm/agents/BASE_PM.md +0 -480
- claude_mpm/agents/BASE_PROMPT_ENGINEER.md +0 -787
- claude_mpm/agents/BASE_QA.md +0 -167
- claude_mpm/agents/BASE_RESEARCH.md +0 -53
- claude_mpm/agents/base_agent.json +0 -31
- claude_mpm/agents/base_agent_loader.py +0 -601
- claude_mpm/cli/commands/agents_detect.py +0 -380
- claude_mpm/cli/commands/agents_recommend.py +0 -309
- claude_mpm/cli/ticket_cli.py +0 -35
- claude_mpm/commands/mpm-agents-auto-configure.md +0 -278
- claude_mpm/commands/mpm-agents-detect.md +0 -177
- claude_mpm/commands/mpm-agents-list.md +0 -131
- claude_mpm/commands/mpm-agents-recommend.md +0 -223
- claude_mpm/commands/mpm-config-view.md +0 -150
- claude_mpm/commands/mpm-ticket-organize.md +0 -304
- claude_mpm/dashboard/analysis_runner.py +0 -455
- claude_mpm/dashboard/index.html +0 -13
- claude_mpm/dashboard/open_dashboard.py +0 -66
- claude_mpm/dashboard/static/css/activity.css +0 -1958
- claude_mpm/dashboard/static/css/connection-status.css +0 -370
- claude_mpm/dashboard/static/css/dashboard.css +0 -4701
- claude_mpm/dashboard/static/js/components/activity-tree.js +0 -1871
- claude_mpm/dashboard/static/js/components/agent-hierarchy.js +0 -777
- claude_mpm/dashboard/static/js/components/agent-inference.js +0 -956
- claude_mpm/dashboard/static/js/components/build-tracker.js +0 -333
- claude_mpm/dashboard/static/js/components/code-simple.js +0 -857
- claude_mpm/dashboard/static/js/components/connection-debug.js +0 -654
- claude_mpm/dashboard/static/js/components/diff-viewer.js +0 -891
- claude_mpm/dashboard/static/js/components/event-processor.js +0 -542
- claude_mpm/dashboard/static/js/components/event-viewer.js +0 -1155
- claude_mpm/dashboard/static/js/components/export-manager.js +0 -368
- claude_mpm/dashboard/static/js/components/file-change-tracker.js +0 -443
- claude_mpm/dashboard/static/js/components/file-change-viewer.js +0 -690
- claude_mpm/dashboard/static/js/components/file-tool-tracker.js +0 -724
- claude_mpm/dashboard/static/js/components/file-viewer.js +0 -580
- claude_mpm/dashboard/static/js/components/hud-library-loader.js +0 -211
- claude_mpm/dashboard/static/js/components/hud-manager.js +0 -671
- claude_mpm/dashboard/static/js/components/hud-visualizer.js +0 -1718
- claude_mpm/dashboard/static/js/components/module-viewer.js +0 -2764
- claude_mpm/dashboard/static/js/components/session-manager.js +0 -579
- claude_mpm/dashboard/static/js/components/socket-manager.js +0 -368
- claude_mpm/dashboard/static/js/components/ui-state-manager.js +0 -749
- claude_mpm/dashboard/static/js/components/unified-data-viewer.js +0 -1824
- claude_mpm/dashboard/static/js/components/working-directory.js +0 -920
- claude_mpm/dashboard/static/js/connection-manager.js +0 -536
- claude_mpm/dashboard/static/js/dashboard.js +0 -1914
- claude_mpm/dashboard/static/js/extension-error-handler.js +0 -164
- claude_mpm/dashboard/static/js/socket-client.js +0 -1474
- claude_mpm/dashboard/static/js/tab-isolation-fix.js +0 -185
- claude_mpm/dashboard/static/socket.io.min.js +0 -7
- claude_mpm/dashboard/static/socket.io.v4.8.1.backup.js +0 -7
- claude_mpm/dashboard/templates/code_simple.html +0 -153
- claude_mpm/dashboard/templates/index.html +0 -606
- claude_mpm/dashboard/test_dashboard.html +0 -372
- claude_mpm/scripts/mcp_server.py +0 -75
- claude_mpm/scripts/mcp_wrapper.py +0 -39
- claude_mpm/services/mcp_gateway/__init__.py +0 -159
- claude_mpm/services/mcp_gateway/auto_configure.py +0 -369
- claude_mpm/services/mcp_gateway/config/__init__.py +0 -17
- claude_mpm/services/mcp_gateway/config/config_loader.py +0 -296
- claude_mpm/services/mcp_gateway/config/config_schema.py +0 -243
- claude_mpm/services/mcp_gateway/config/configuration.py +0 -429
- claude_mpm/services/mcp_gateway/core/__init__.py +0 -43
- claude_mpm/services/mcp_gateway/core/base.py +0 -312
- claude_mpm/services/mcp_gateway/core/exceptions.py +0 -253
- claude_mpm/services/mcp_gateway/core/interfaces.py +0 -443
- claude_mpm/services/mcp_gateway/core/process_pool.py +0 -977
- claude_mpm/services/mcp_gateway/core/singleton_manager.py +0 -315
- claude_mpm/services/mcp_gateway/core/startup_verification.py +0 -316
- claude_mpm/services/mcp_gateway/main.py +0 -589
- claude_mpm/services/mcp_gateway/registry/__init__.py +0 -12
- claude_mpm/services/mcp_gateway/registry/service_registry.py +0 -412
- claude_mpm/services/mcp_gateway/registry/tool_registry.py +0 -489
- claude_mpm/services/mcp_gateway/server/__init__.py +0 -15
- claude_mpm/services/mcp_gateway/server/mcp_gateway.py +0 -414
- claude_mpm/services/mcp_gateway/server/stdio_handler.py +0 -372
- claude_mpm/services/mcp_gateway/server/stdio_server.py +0 -712
- claude_mpm/services/mcp_gateway/tools/__init__.py +0 -36
- claude_mpm/services/mcp_gateway/tools/base_adapter.py +0 -485
- claude_mpm/services/mcp_gateway/tools/document_summarizer.py +0 -789
- claude_mpm/services/mcp_gateway/tools/external_mcp_services.py +0 -654
- claude_mpm/services/mcp_gateway/tools/health_check_tool.py +0 -456
- claude_mpm/services/mcp_gateway/tools/hello_world.py +0 -551
- claude_mpm/services/mcp_gateway/tools/kuzu_memory_service.py +0 -555
- claude_mpm/services/mcp_gateway/utils/__init__.py +0 -14
- claude_mpm/services/mcp_gateway/utils/package_version_checker.py +0 -160
- claude_mpm/services/mcp_gateway/utils/update_preferences.py +0 -170
- claude_mpm-5.1.9.dist-info/entry_points.txt +0 -10
- claude_mpm-5.1.9.dist-info/licenses/LICENSE +0 -21
- {claude_mpm-5.1.9.dist-info → claude_mpm-5.4.22.dist-info}/WHEEL +0 -0
- {claude_mpm-5.1.9.dist-info → claude_mpm-5.4.22.dist-info}/top_level.txt +0 -0
|
@@ -0,0 +1,704 @@
|
|
|
1
|
+
"""Selective skill deployment based on agent requirements.
|
|
2
|
+
|
|
3
|
+
WHY: Agents now have a skills field in their frontmatter. We should only deploy
|
|
4
|
+
skills that agents actually reference, reducing deployed skills from ~78 to ~20
|
|
5
|
+
for a typical project.
|
|
6
|
+
|
|
7
|
+
DESIGN DECISIONS:
|
|
8
|
+
- Dual-source skill discovery:
|
|
9
|
+
1. Explicit frontmatter declarations (skills: field)
|
|
10
|
+
2. SkillToAgentMapper inference (pattern-based)
|
|
11
|
+
- Support both legacy flat list and new required/optional dict formats
|
|
12
|
+
- Parse YAML frontmatter from agent markdown files
|
|
13
|
+
- Combine explicit + inferred skills for comprehensive coverage
|
|
14
|
+
- Return set of unique skill names for filtering
|
|
15
|
+
- Track deployed skills in .mpm-deployed-skills.json index
|
|
16
|
+
- Remove orphaned skills (deployed by mpm but no longer referenced)
|
|
17
|
+
|
|
18
|
+
FORMATS SUPPORTED:
|
|
19
|
+
1. Legacy: skills: [skill-a, skill-b, ...]
|
|
20
|
+
2. New: skills: {required: [...], optional: [...]}
|
|
21
|
+
|
|
22
|
+
SKILL DISCOVERY FLOW:
|
|
23
|
+
1. Scan deployed agents (.claude/agents/*.md)
|
|
24
|
+
2. Extract frontmatter skills (explicit declarations)
|
|
25
|
+
3. Query SkillToAgentMapper for pattern-based skills
|
|
26
|
+
4. Combine both sources into unified set
|
|
27
|
+
|
|
28
|
+
DEPLOYMENT TRACKING:
|
|
29
|
+
1. Track which skills were deployed by claude-mpm in index file
|
|
30
|
+
2. Update index after each deployment operation
|
|
31
|
+
3. Clean up orphaned skills no longer referenced by agents
|
|
32
|
+
|
|
33
|
+
References:
|
|
34
|
+
- Feature: Progressive skills discovery (#117)
|
|
35
|
+
- Service: SkillToAgentMapper (skill_to_agent_mapper.py)
|
|
36
|
+
"""
|
|
37
|
+
|
|
38
|
+
import json
|
|
39
|
+
import re
|
|
40
|
+
from datetime import datetime, timezone
|
|
41
|
+
from pathlib import Path
|
|
42
|
+
from typing import Any, Dict, List, Set, Tuple
|
|
43
|
+
|
|
44
|
+
import yaml
|
|
45
|
+
|
|
46
|
+
from claude_mpm.core.logging_config import get_logger
|
|
47
|
+
from claude_mpm.services.skills.skill_to_agent_mapper import SkillToAgentMapper
|
|
48
|
+
|
|
49
|
+
logger = get_logger(__name__)
|
|
50
|
+
|
|
51
|
+
# Deployment tracking index file
|
|
52
|
+
DEPLOYED_INDEX_FILE = ".mpm-deployed-skills.json"
|
|
53
|
+
|
|
54
|
+
|
|
55
|
+
def parse_agent_frontmatter(agent_file: Path) -> Dict[str, Any]:
|
|
56
|
+
"""Parse YAML frontmatter from agent markdown file.
|
|
57
|
+
|
|
58
|
+
Args:
|
|
59
|
+
agent_file: Path to agent markdown file
|
|
60
|
+
|
|
61
|
+
Returns:
|
|
62
|
+
Parsed frontmatter as dictionary, or empty dict if parsing fails
|
|
63
|
+
|
|
64
|
+
Example:
|
|
65
|
+
>>> frontmatter = parse_agent_frontmatter(Path("agent.md"))
|
|
66
|
+
>>> skills = frontmatter.get('skills', [])
|
|
67
|
+
"""
|
|
68
|
+
try:
|
|
69
|
+
content = agent_file.read_text(encoding="utf-8")
|
|
70
|
+
except Exception as e:
|
|
71
|
+
logger.warning(f"Failed to read {agent_file}: {e}")
|
|
72
|
+
return {}
|
|
73
|
+
|
|
74
|
+
# Match YAML frontmatter between --- delimiters
|
|
75
|
+
match = re.match(r"^---\n(.*?)\n---", content, re.DOTALL)
|
|
76
|
+
if not match:
|
|
77
|
+
logger.debug(f"No frontmatter found in {agent_file}")
|
|
78
|
+
return {}
|
|
79
|
+
|
|
80
|
+
try:
|
|
81
|
+
frontmatter = yaml.safe_load(match.group(1))
|
|
82
|
+
return frontmatter or {}
|
|
83
|
+
except yaml.YAMLError as e:
|
|
84
|
+
logger.warning(f"Failed to parse frontmatter in {agent_file}: {e}")
|
|
85
|
+
return {}
|
|
86
|
+
|
|
87
|
+
|
|
88
|
+
def get_skills_from_agent(frontmatter: Dict[str, Any]) -> Set[str]:
|
|
89
|
+
"""Extract skill names from agent frontmatter (handles both formats).
|
|
90
|
+
|
|
91
|
+
Supports both legacy and new formats:
|
|
92
|
+
- Legacy: skills: [skill-a, skill-b, ...]
|
|
93
|
+
- New: skills: {required: [...], optional: [...]}
|
|
94
|
+
|
|
95
|
+
Args:
|
|
96
|
+
frontmatter: Parsed agent frontmatter
|
|
97
|
+
|
|
98
|
+
Returns:
|
|
99
|
+
Set of unique skill names
|
|
100
|
+
|
|
101
|
+
Example:
|
|
102
|
+
>>> # Legacy format
|
|
103
|
+
>>> frontmatter = {'skills': ['skill-a', 'skill-b']}
|
|
104
|
+
>>> get_skills_from_agent(frontmatter)
|
|
105
|
+
{'skill-a', 'skill-b'}
|
|
106
|
+
|
|
107
|
+
>>> # New format
|
|
108
|
+
>>> frontmatter = {'skills': {'required': ['skill-a'], 'optional': ['skill-b']}}
|
|
109
|
+
>>> get_skills_from_agent(frontmatter)
|
|
110
|
+
{'skill-a', 'skill-b'}
|
|
111
|
+
"""
|
|
112
|
+
skills_field = frontmatter.get("skills")
|
|
113
|
+
|
|
114
|
+
# Handle None or missing skills field
|
|
115
|
+
if skills_field is None:
|
|
116
|
+
return set()
|
|
117
|
+
|
|
118
|
+
# New format: {required: [...], optional: [...]}
|
|
119
|
+
if isinstance(skills_field, dict):
|
|
120
|
+
required = skills_field.get("required") or []
|
|
121
|
+
optional = skills_field.get("optional") or []
|
|
122
|
+
|
|
123
|
+
# Ensure both are lists
|
|
124
|
+
if not isinstance(required, list):
|
|
125
|
+
required = []
|
|
126
|
+
if not isinstance(optional, list):
|
|
127
|
+
optional = []
|
|
128
|
+
|
|
129
|
+
return set(required + optional)
|
|
130
|
+
|
|
131
|
+
# Legacy format: [skill1, skill2, ...]
|
|
132
|
+
if isinstance(skills_field, list):
|
|
133
|
+
return set(skills_field)
|
|
134
|
+
|
|
135
|
+
# Unsupported format
|
|
136
|
+
logger.warning(f"Unexpected skills field type: {type(skills_field)}")
|
|
137
|
+
return set()
|
|
138
|
+
|
|
139
|
+
|
|
140
|
+
def get_skills_from_mapping(agent_ids: List[str]) -> Set[str]:
|
|
141
|
+
"""Get skills for agents using SkillToAgentMapper inference.
|
|
142
|
+
|
|
143
|
+
Uses SkillToAgentMapper to find all skills associated with given agent IDs.
|
|
144
|
+
This provides pattern-based skill discovery beyond explicit frontmatter declarations.
|
|
145
|
+
|
|
146
|
+
Args:
|
|
147
|
+
agent_ids: List of agent identifiers (e.g., ["python-engineer", "typescript-engineer"])
|
|
148
|
+
|
|
149
|
+
Returns:
|
|
150
|
+
Set of unique skill names inferred from mapping configuration
|
|
151
|
+
|
|
152
|
+
Example:
|
|
153
|
+
>>> agent_ids = ["python-engineer", "typescript-engineer"]
|
|
154
|
+
>>> skills = get_skills_from_mapping(agent_ids)
|
|
155
|
+
>>> print(f"Found {len(skills)} skills from mapping")
|
|
156
|
+
"""
|
|
157
|
+
try:
|
|
158
|
+
mapper = SkillToAgentMapper()
|
|
159
|
+
all_skills = set()
|
|
160
|
+
|
|
161
|
+
for agent_id in agent_ids:
|
|
162
|
+
agent_skills = mapper.get_skills_for_agent(agent_id)
|
|
163
|
+
if agent_skills:
|
|
164
|
+
all_skills.update(agent_skills)
|
|
165
|
+
logger.debug(f"Mapped {len(agent_skills)} skills to {agent_id}")
|
|
166
|
+
|
|
167
|
+
logger.info(
|
|
168
|
+
f"Mapped {len(all_skills)} unique skills for {len(agent_ids)} agents"
|
|
169
|
+
)
|
|
170
|
+
return all_skills
|
|
171
|
+
|
|
172
|
+
except Exception as e:
|
|
173
|
+
logger.warning(f"Failed to load SkillToAgentMapper: {e}")
|
|
174
|
+
logger.info("Falling back to frontmatter-only skill discovery")
|
|
175
|
+
return set()
|
|
176
|
+
|
|
177
|
+
|
|
178
|
+
def get_required_skills_from_agents(agents_dir: Path) -> Set[str]:
|
|
179
|
+
"""Extract all skills referenced by deployed agents.
|
|
180
|
+
|
|
181
|
+
Combines skills from two sources:
|
|
182
|
+
1. Explicit frontmatter declarations (skills: field in agent .md files)
|
|
183
|
+
2. SkillToAgentMapper inference (pattern-based skill discovery)
|
|
184
|
+
|
|
185
|
+
This dual-source approach ensures agents get both explicitly declared skills
|
|
186
|
+
and skills inferred from their domain/toolchain patterns.
|
|
187
|
+
|
|
188
|
+
Args:
|
|
189
|
+
agents_dir: Path to deployed agents directory (e.g., .claude/agents/)
|
|
190
|
+
|
|
191
|
+
Returns:
|
|
192
|
+
Set of unique skill names referenced across all agents
|
|
193
|
+
|
|
194
|
+
Example:
|
|
195
|
+
>>> agents_dir = Path(".claude/agents")
|
|
196
|
+
>>> required_skills = get_required_skills_from_agents(agents_dir)
|
|
197
|
+
>>> print(f"Found {len(required_skills)} unique skills")
|
|
198
|
+
"""
|
|
199
|
+
if not agents_dir.exists():
|
|
200
|
+
logger.warning(f"Agents directory not found: {agents_dir}")
|
|
201
|
+
return set()
|
|
202
|
+
|
|
203
|
+
# Scan all agent markdown files
|
|
204
|
+
agent_files = list(agents_dir.glob("*.md"))
|
|
205
|
+
logger.debug(f"Scanning {len(agent_files)} agent files in {agents_dir}")
|
|
206
|
+
|
|
207
|
+
# Source 1: Extract skills from frontmatter
|
|
208
|
+
frontmatter_skills = set()
|
|
209
|
+
agent_ids = []
|
|
210
|
+
|
|
211
|
+
for agent_file in agent_files:
|
|
212
|
+
agent_id = agent_file.stem
|
|
213
|
+
agent_ids.append(agent_id)
|
|
214
|
+
|
|
215
|
+
frontmatter = parse_agent_frontmatter(agent_file)
|
|
216
|
+
agent_skills = get_skills_from_agent(frontmatter)
|
|
217
|
+
|
|
218
|
+
if agent_skills:
|
|
219
|
+
frontmatter_skills.update(agent_skills)
|
|
220
|
+
logger.debug(
|
|
221
|
+
f"Agent {agent_id}: {len(agent_skills)} skills from frontmatter"
|
|
222
|
+
)
|
|
223
|
+
|
|
224
|
+
logger.info(f"Found {len(frontmatter_skills)} unique skills from frontmatter")
|
|
225
|
+
|
|
226
|
+
# Source 2: Get skills from SkillToAgentMapper
|
|
227
|
+
mapped_skills = get_skills_from_mapping(agent_ids)
|
|
228
|
+
|
|
229
|
+
# Combine both sources
|
|
230
|
+
required_skills = frontmatter_skills | mapped_skills
|
|
231
|
+
|
|
232
|
+
# Normalize skill paths: convert slashes to dashes for compatibility with deployment
|
|
233
|
+
# SkillToAgentMapper returns paths like "toolchains/python/frameworks/django"
|
|
234
|
+
# but deployment expects "toolchains-python-frameworks-django"
|
|
235
|
+
normalized_skills = {skill.replace("/", "-") for skill in required_skills}
|
|
236
|
+
|
|
237
|
+
logger.info(
|
|
238
|
+
f"Combined {len(frontmatter_skills)} frontmatter + {len(mapped_skills)} mapped "
|
|
239
|
+
f"= {len(required_skills)} total unique skills (normalized to {len(normalized_skills)})"
|
|
240
|
+
)
|
|
241
|
+
|
|
242
|
+
return normalized_skills
|
|
243
|
+
|
|
244
|
+
|
|
245
|
+
# === Deployment Tracking Functions ===
|
|
246
|
+
|
|
247
|
+
|
|
248
|
+
def load_deployment_index(claude_skills_dir: Path) -> Dict[str, Any]:
|
|
249
|
+
"""Load deployment tracking index from ~/.claude/skills/.
|
|
250
|
+
|
|
251
|
+
Args:
|
|
252
|
+
claude_skills_dir: Path to Claude skills directory (~/.claude/skills/)
|
|
253
|
+
|
|
254
|
+
Returns:
|
|
255
|
+
Dict containing:
|
|
256
|
+
- deployed_skills: Dict mapping skill name to deployment metadata
|
|
257
|
+
- user_requested_skills: List of skill names manually requested by user
|
|
258
|
+
- last_sync: ISO timestamp of last sync operation
|
|
259
|
+
|
|
260
|
+
Example:
|
|
261
|
+
>>> index = load_deployment_index(Path.home() / ".claude" / "skills")
|
|
262
|
+
>>> print(f"Tracked skills: {len(index['deployed_skills'])}")
|
|
263
|
+
"""
|
|
264
|
+
index_path = claude_skills_dir / DEPLOYED_INDEX_FILE
|
|
265
|
+
|
|
266
|
+
if not index_path.exists():
|
|
267
|
+
logger.debug(f"No deployment index found at {index_path}, creating new")
|
|
268
|
+
return {"deployed_skills": {}, "user_requested_skills": [], "last_sync": None}
|
|
269
|
+
|
|
270
|
+
try:
|
|
271
|
+
with open(index_path, encoding="utf-8") as f:
|
|
272
|
+
index = json.load(f)
|
|
273
|
+
|
|
274
|
+
# Ensure required keys exist
|
|
275
|
+
if "deployed_skills" not in index:
|
|
276
|
+
index["deployed_skills"] = {}
|
|
277
|
+
if "user_requested_skills" not in index:
|
|
278
|
+
index["user_requested_skills"] = []
|
|
279
|
+
if "last_sync" not in index:
|
|
280
|
+
index["last_sync"] = None
|
|
281
|
+
|
|
282
|
+
logger.debug(
|
|
283
|
+
f"Loaded deployment index: {len(index['deployed_skills'])} tracked skills, "
|
|
284
|
+
f"{len(index['user_requested_skills'])} user-requested"
|
|
285
|
+
)
|
|
286
|
+
return index
|
|
287
|
+
|
|
288
|
+
except (json.JSONDecodeError, OSError) as e:
|
|
289
|
+
logger.warning(f"Failed to load deployment index: {e}, creating new")
|
|
290
|
+
return {"deployed_skills": {}, "user_requested_skills": [], "last_sync": None}
|
|
291
|
+
|
|
292
|
+
|
|
293
|
+
def save_deployment_index(claude_skills_dir: Path, index: Dict[str, Any]) -> None:
|
|
294
|
+
"""Save deployment tracking index to ~/.claude/skills/.
|
|
295
|
+
|
|
296
|
+
Args:
|
|
297
|
+
claude_skills_dir: Path to Claude skills directory (~/.claude/skills/)
|
|
298
|
+
index: Index data to save
|
|
299
|
+
|
|
300
|
+
Example:
|
|
301
|
+
>>> index = {"deployed_skills": {...}, "last_sync": "2025-12-22T10:30:00Z"}
|
|
302
|
+
>>> save_deployment_index(Path.home() / ".claude" / "skills", index)
|
|
303
|
+
"""
|
|
304
|
+
index_path = claude_skills_dir / DEPLOYED_INDEX_FILE
|
|
305
|
+
|
|
306
|
+
try:
|
|
307
|
+
# Ensure directory exists
|
|
308
|
+
claude_skills_dir.mkdir(parents=True, exist_ok=True)
|
|
309
|
+
|
|
310
|
+
with open(index_path, "w", encoding="utf-8") as f:
|
|
311
|
+
json.dump(index, f, indent=2, ensure_ascii=False)
|
|
312
|
+
|
|
313
|
+
logger.debug(f"Saved deployment index: {len(index['deployed_skills'])} skills")
|
|
314
|
+
|
|
315
|
+
except OSError as e:
|
|
316
|
+
logger.error(f"Failed to save deployment index: {e}")
|
|
317
|
+
raise
|
|
318
|
+
|
|
319
|
+
|
|
320
|
+
def track_deployed_skill(
|
|
321
|
+
claude_skills_dir: Path, skill_name: str, collection: str
|
|
322
|
+
) -> None:
|
|
323
|
+
"""Track a newly deployed skill in the deployment index.
|
|
324
|
+
|
|
325
|
+
Args:
|
|
326
|
+
claude_skills_dir: Path to Claude skills directory (~/.claude/skills/)
|
|
327
|
+
skill_name: Name of deployed skill
|
|
328
|
+
collection: Collection name skill was deployed from
|
|
329
|
+
|
|
330
|
+
Example:
|
|
331
|
+
>>> track_deployed_skill(
|
|
332
|
+
... Path.home() / ".claude" / "skills",
|
|
333
|
+
... "systematic-debugging",
|
|
334
|
+
... "claude-mpm-skills"
|
|
335
|
+
... )
|
|
336
|
+
"""
|
|
337
|
+
index = load_deployment_index(claude_skills_dir)
|
|
338
|
+
|
|
339
|
+
# Add skill to deployed_skills
|
|
340
|
+
index["deployed_skills"][skill_name] = {
|
|
341
|
+
"collection": collection,
|
|
342
|
+
"deployed_at": datetime.now(timezone.utc).isoformat().replace("+00:00", "Z"),
|
|
343
|
+
}
|
|
344
|
+
|
|
345
|
+
# Update last_sync timestamp
|
|
346
|
+
index["last_sync"] = datetime.now(timezone.utc).isoformat().replace("+00:00", "Z")
|
|
347
|
+
|
|
348
|
+
save_deployment_index(claude_skills_dir, index)
|
|
349
|
+
logger.debug(f"Tracked deployment: {skill_name} from {collection}")
|
|
350
|
+
|
|
351
|
+
|
|
352
|
+
def untrack_skill(claude_skills_dir: Path, skill_name: str) -> None:
|
|
353
|
+
"""Remove skill from deployment tracking index.
|
|
354
|
+
|
|
355
|
+
Args:
|
|
356
|
+
claude_skills_dir: Path to Claude skills directory (~/.claude/skills/)
|
|
357
|
+
skill_name: Name of skill to untrack
|
|
358
|
+
|
|
359
|
+
Example:
|
|
360
|
+
>>> untrack_skill(
|
|
361
|
+
... Path.home() / ".claude" / "skills",
|
|
362
|
+
... "old-skill"
|
|
363
|
+
... )
|
|
364
|
+
"""
|
|
365
|
+
index = load_deployment_index(claude_skills_dir)
|
|
366
|
+
|
|
367
|
+
if skill_name in index["deployed_skills"]:
|
|
368
|
+
del index["deployed_skills"][skill_name]
|
|
369
|
+
index["last_sync"] = (
|
|
370
|
+
datetime.now(timezone.utc).isoformat().replace("+00:00", "Z")
|
|
371
|
+
)
|
|
372
|
+
save_deployment_index(claude_skills_dir, index)
|
|
373
|
+
logger.debug(f"Untracked skill: {skill_name}")
|
|
374
|
+
|
|
375
|
+
|
|
376
|
+
def cleanup_orphan_skills(
|
|
377
|
+
claude_skills_dir: Path, required_skills: Set[str]
|
|
378
|
+
) -> Dict[str, Any]:
|
|
379
|
+
"""Remove skills deployed by claude-mpm but no longer referenced by agents.
|
|
380
|
+
|
|
381
|
+
This function:
|
|
382
|
+
1. Loads deployment tracking index
|
|
383
|
+
2. Identifies orphaned skills (tracked but not in required_skills AND not user-requested)
|
|
384
|
+
3. Removes orphaned skill directories from ~/.claude/skills/
|
|
385
|
+
4. Updates deployment index
|
|
386
|
+
|
|
387
|
+
User-requested skills are NEVER cleaned up as orphans - they are treated as required.
|
|
388
|
+
|
|
389
|
+
Args:
|
|
390
|
+
claude_skills_dir: Path to Claude skills directory (~/.claude/skills/)
|
|
391
|
+
required_skills: Set of skill names currently required by agents
|
|
392
|
+
|
|
393
|
+
Returns:
|
|
394
|
+
Dict containing:
|
|
395
|
+
- removed_count: Number of skills removed
|
|
396
|
+
- removed_skills: List of removed skill names
|
|
397
|
+
- kept_count: Number of skills kept
|
|
398
|
+
- errors: List of error messages
|
|
399
|
+
|
|
400
|
+
Example:
|
|
401
|
+
>>> required = {"skill-a", "skill-b"}
|
|
402
|
+
>>> result = cleanup_orphan_skills(
|
|
403
|
+
... Path.home() / ".claude" / "skills",
|
|
404
|
+
... required
|
|
405
|
+
... )
|
|
406
|
+
>>> print(f"Removed {result['removed_count']} orphaned skills")
|
|
407
|
+
"""
|
|
408
|
+
import shutil
|
|
409
|
+
|
|
410
|
+
index = load_deployment_index(claude_skills_dir)
|
|
411
|
+
tracked_skills = set(index["deployed_skills"].keys())
|
|
412
|
+
user_requested = set(index.get("user_requested_skills", []))
|
|
413
|
+
|
|
414
|
+
# Find orphaned skills: tracked by mpm but not in required_skills AND not user-requested
|
|
415
|
+
# User-requested skills are treated as required and NEVER cleaned up
|
|
416
|
+
all_required = required_skills | user_requested
|
|
417
|
+
orphaned = tracked_skills - all_required
|
|
418
|
+
|
|
419
|
+
if not orphaned:
|
|
420
|
+
logger.info("No orphaned skills to remove")
|
|
421
|
+
return {
|
|
422
|
+
"removed_count": 0,
|
|
423
|
+
"removed_skills": [],
|
|
424
|
+
"kept_count": len(tracked_skills),
|
|
425
|
+
"errors": [],
|
|
426
|
+
}
|
|
427
|
+
|
|
428
|
+
logger.info(
|
|
429
|
+
f"Found {len(orphaned)} orphaned skills (tracked but not required by agents)"
|
|
430
|
+
)
|
|
431
|
+
|
|
432
|
+
removed = []
|
|
433
|
+
errors = []
|
|
434
|
+
|
|
435
|
+
for skill_name in orphaned:
|
|
436
|
+
skill_dir = claude_skills_dir / skill_name
|
|
437
|
+
|
|
438
|
+
# Remove skill directory if it exists
|
|
439
|
+
if skill_dir.exists():
|
|
440
|
+
try:
|
|
441
|
+
# Validate path is within claude_skills_dir (security)
|
|
442
|
+
skill_dir.resolve().relative_to(claude_skills_dir.resolve())
|
|
443
|
+
|
|
444
|
+
# Remove directory
|
|
445
|
+
if skill_dir.is_symlink():
|
|
446
|
+
logger.debug(f"Removing symlink: {skill_dir}")
|
|
447
|
+
skill_dir.unlink()
|
|
448
|
+
else:
|
|
449
|
+
shutil.rmtree(skill_dir)
|
|
450
|
+
|
|
451
|
+
removed.append(skill_name)
|
|
452
|
+
logger.info(f"Removed orphaned skill: {skill_name}")
|
|
453
|
+
|
|
454
|
+
except ValueError:
|
|
455
|
+
error_msg = f"Path traversal attempt detected: {skill_dir}"
|
|
456
|
+
logger.error(error_msg)
|
|
457
|
+
errors.append(error_msg)
|
|
458
|
+
continue
|
|
459
|
+
except Exception as e:
|
|
460
|
+
error_msg = f"Failed to remove {skill_name}: {e}"
|
|
461
|
+
logger.error(error_msg)
|
|
462
|
+
errors.append(error_msg)
|
|
463
|
+
continue
|
|
464
|
+
|
|
465
|
+
# Remove from tracking index
|
|
466
|
+
untrack_skill(claude_skills_dir, skill_name)
|
|
467
|
+
|
|
468
|
+
kept_count = len(tracked_skills) - len(removed)
|
|
469
|
+
|
|
470
|
+
logger.info(
|
|
471
|
+
f"Cleanup complete: removed {len(removed)} skills, kept {kept_count} skills"
|
|
472
|
+
)
|
|
473
|
+
|
|
474
|
+
return {
|
|
475
|
+
"removed_count": len(removed),
|
|
476
|
+
"removed_skills": removed,
|
|
477
|
+
"kept_count": kept_count,
|
|
478
|
+
"errors": errors,
|
|
479
|
+
}
|
|
480
|
+
|
|
481
|
+
|
|
482
|
+
# === Configuration Management Functions ===
|
|
483
|
+
|
|
484
|
+
|
|
485
|
+
def save_agent_skills_to_config(skills: List[str], config_path: Path) -> None:
|
|
486
|
+
"""Save agent-scanned skills to configuration.yaml under skills.agent_referenced.
|
|
487
|
+
|
|
488
|
+
Args:
|
|
489
|
+
skills: List of skill names scanned from deployed agents
|
|
490
|
+
config_path: Path to configuration.yaml file
|
|
491
|
+
|
|
492
|
+
Example:
|
|
493
|
+
>>> skills = ["systematic-debugging", "typescript-core"]
|
|
494
|
+
>>> save_agent_skills_to_config(skills, Path(".claude-mpm/configuration.yaml"))
|
|
495
|
+
"""
|
|
496
|
+
import yaml
|
|
497
|
+
|
|
498
|
+
try:
|
|
499
|
+
# Load existing configuration (or create empty dict)
|
|
500
|
+
if config_path.exists():
|
|
501
|
+
with open(config_path, encoding="utf-8") as f:
|
|
502
|
+
config = yaml.safe_load(f) or {}
|
|
503
|
+
else:
|
|
504
|
+
config = {}
|
|
505
|
+
|
|
506
|
+
# Ensure skills section exists
|
|
507
|
+
if "skills" not in config:
|
|
508
|
+
config["skills"] = {}
|
|
509
|
+
|
|
510
|
+
# Update agent_referenced skills (sorted for consistency)
|
|
511
|
+
config["skills"]["agent_referenced"] = sorted(skills)
|
|
512
|
+
|
|
513
|
+
# Ensure user_defined exists (but don't overwrite if set)
|
|
514
|
+
if "user_defined" not in config["skills"]:
|
|
515
|
+
config["skills"]["user_defined"] = []
|
|
516
|
+
|
|
517
|
+
# Save configuration
|
|
518
|
+
config_path.parent.mkdir(parents=True, exist_ok=True)
|
|
519
|
+
with open(config_path, "w", encoding="utf-8") as f:
|
|
520
|
+
yaml.dump(config, f, default_flow_style=False, sort_keys=False)
|
|
521
|
+
|
|
522
|
+
logger.info(
|
|
523
|
+
f"Saved {len(skills)} agent-referenced skills to configuration.yaml"
|
|
524
|
+
)
|
|
525
|
+
|
|
526
|
+
except Exception as e:
|
|
527
|
+
logger.error(f"Failed to save agent skills to config: {e}")
|
|
528
|
+
raise
|
|
529
|
+
|
|
530
|
+
|
|
531
|
+
def get_skills_to_deploy(config_path: Path) -> Tuple[List[str], str]:
|
|
532
|
+
"""Resolve which skills to deploy based on configuration priority.
|
|
533
|
+
|
|
534
|
+
Returns (skills_list, source) where source is 'user_defined' or 'agent_referenced'.
|
|
535
|
+
|
|
536
|
+
Logic:
|
|
537
|
+
- If config.skills.user_defined is non-empty → return (user_defined, 'user_defined')
|
|
538
|
+
- Otherwise → return (agent_referenced, 'agent_referenced')
|
|
539
|
+
|
|
540
|
+
Args:
|
|
541
|
+
config_path: Path to configuration.yaml file
|
|
542
|
+
|
|
543
|
+
Returns:
|
|
544
|
+
Tuple of (skills list, source string)
|
|
545
|
+
|
|
546
|
+
Example:
|
|
547
|
+
>>> skills, source = get_skills_to_deploy(Path(".claude-mpm/configuration.yaml"))
|
|
548
|
+
>>> print(f"Deploy {len(skills)} skills from {source}")
|
|
549
|
+
"""
|
|
550
|
+
import yaml
|
|
551
|
+
|
|
552
|
+
try:
|
|
553
|
+
# Load configuration
|
|
554
|
+
if not config_path.exists():
|
|
555
|
+
logger.warning(f"Configuration file not found: {config_path}")
|
|
556
|
+
return ([], "agent_referenced")
|
|
557
|
+
|
|
558
|
+
with open(config_path, encoding="utf-8") as f:
|
|
559
|
+
config = yaml.safe_load(f) or {}
|
|
560
|
+
|
|
561
|
+
skills_config = config.get("skills", {})
|
|
562
|
+
user_defined = skills_config.get("user_defined", [])
|
|
563
|
+
agent_referenced = skills_config.get("agent_referenced", [])
|
|
564
|
+
|
|
565
|
+
# Priority: user_defined if non-empty, otherwise agent_referenced
|
|
566
|
+
if user_defined:
|
|
567
|
+
logger.info(
|
|
568
|
+
f"Using {len(user_defined)} user-defined skills from configuration"
|
|
569
|
+
)
|
|
570
|
+
return (user_defined, "user_defined")
|
|
571
|
+
logger.info(
|
|
572
|
+
f"Using {len(agent_referenced)} agent-referenced skills from configuration"
|
|
573
|
+
)
|
|
574
|
+
return (agent_referenced, "agent_referenced")
|
|
575
|
+
|
|
576
|
+
except Exception as e:
|
|
577
|
+
logger.error(f"Failed to load skills from config: {e}")
|
|
578
|
+
return ([], "agent_referenced")
|
|
579
|
+
|
|
580
|
+
|
|
581
|
+
# === User-Requested Skills Management ===
|
|
582
|
+
|
|
583
|
+
|
|
584
|
+
def get_user_requested_skills(claude_skills_dir: Path) -> List[str]:
|
|
585
|
+
"""Get list of user-requested skills.
|
|
586
|
+
|
|
587
|
+
Args:
|
|
588
|
+
claude_skills_dir: Path to Claude skills directory (~/.claude/skills/)
|
|
589
|
+
|
|
590
|
+
Returns:
|
|
591
|
+
List of skill names manually requested by user
|
|
592
|
+
|
|
593
|
+
Example:
|
|
594
|
+
>>> skills = get_user_requested_skills(Path.home() / ".claude" / "skills")
|
|
595
|
+
>>> print(f"User requested {len(skills)} skills")
|
|
596
|
+
"""
|
|
597
|
+
index = load_deployment_index(claude_skills_dir)
|
|
598
|
+
return index.get("user_requested_skills", [])
|
|
599
|
+
|
|
600
|
+
|
|
601
|
+
def add_user_requested_skill(skill_name: str, claude_skills_dir: Path) -> bool:
|
|
602
|
+
"""Add a skill to user_requested_skills list.
|
|
603
|
+
|
|
604
|
+
This function:
|
|
605
|
+
1. Loads deployment index
|
|
606
|
+
2. Adds skill name to user_requested_skills (if not already present)
|
|
607
|
+
3. Saves updated index
|
|
608
|
+
4. Returns success status
|
|
609
|
+
|
|
610
|
+
Note: This function does NOT deploy the skill, it only marks it as user-requested.
|
|
611
|
+
Use this in conjunction with skill deployment functions.
|
|
612
|
+
|
|
613
|
+
Args:
|
|
614
|
+
skill_name: Name of skill to mark as user-requested
|
|
615
|
+
claude_skills_dir: Path to Claude skills directory (~/.claude/skills/)
|
|
616
|
+
|
|
617
|
+
Returns:
|
|
618
|
+
True if skill was added, False if already present
|
|
619
|
+
|
|
620
|
+
Example:
|
|
621
|
+
>>> added = add_user_requested_skill(
|
|
622
|
+
... "django-framework",
|
|
623
|
+
... Path.home() / ".claude" / "skills"
|
|
624
|
+
... )
|
|
625
|
+
>>> print(f"Skill added: {added}")
|
|
626
|
+
"""
|
|
627
|
+
index = load_deployment_index(claude_skills_dir)
|
|
628
|
+
user_requested = index.get("user_requested_skills", [])
|
|
629
|
+
|
|
630
|
+
if skill_name in user_requested:
|
|
631
|
+
logger.debug(f"Skill {skill_name} already in user_requested_skills")
|
|
632
|
+
return False
|
|
633
|
+
|
|
634
|
+
user_requested.append(skill_name)
|
|
635
|
+
index["user_requested_skills"] = user_requested
|
|
636
|
+
index["last_sync"] = datetime.now(timezone.utc).isoformat().replace("+00:00", "Z")
|
|
637
|
+
|
|
638
|
+
save_deployment_index(claude_skills_dir, index)
|
|
639
|
+
logger.info(f"Added {skill_name} to user_requested_skills")
|
|
640
|
+
return True
|
|
641
|
+
|
|
642
|
+
|
|
643
|
+
def remove_user_requested_skill(skill_name: str, claude_skills_dir: Path) -> bool:
|
|
644
|
+
"""Remove a skill from user_requested_skills list.
|
|
645
|
+
|
|
646
|
+
This function:
|
|
647
|
+
1. Loads deployment index
|
|
648
|
+
2. Removes skill name from user_requested_skills
|
|
649
|
+
3. Saves updated index
|
|
650
|
+
4. Returns success status
|
|
651
|
+
|
|
652
|
+
Note: This function does NOT remove the deployed skill directory.
|
|
653
|
+
It only removes the skill from user_requested_skills, making it eligible
|
|
654
|
+
for cleanup during orphan removal.
|
|
655
|
+
|
|
656
|
+
Args:
|
|
657
|
+
skill_name: Name of skill to remove from user_requested_skills
|
|
658
|
+
claude_skills_dir: Path to Claude skills directory (~/.claude/skills/)
|
|
659
|
+
|
|
660
|
+
Returns:
|
|
661
|
+
True if skill was removed, False if not present
|
|
662
|
+
|
|
663
|
+
Example:
|
|
664
|
+
>>> removed = remove_user_requested_skill(
|
|
665
|
+
... "django-framework",
|
|
666
|
+
... Path.home() / ".claude" / "skills"
|
|
667
|
+
... )
|
|
668
|
+
>>> print(f"Skill removed: {removed}")
|
|
669
|
+
"""
|
|
670
|
+
index = load_deployment_index(claude_skills_dir)
|
|
671
|
+
user_requested = index.get("user_requested_skills", [])
|
|
672
|
+
|
|
673
|
+
if skill_name not in user_requested:
|
|
674
|
+
logger.debug(f"Skill {skill_name} not in user_requested_skills")
|
|
675
|
+
return False
|
|
676
|
+
|
|
677
|
+
user_requested.remove(skill_name)
|
|
678
|
+
index["user_requested_skills"] = user_requested
|
|
679
|
+
index["last_sync"] = datetime.now(timezone.utc).isoformat().replace("+00:00", "Z")
|
|
680
|
+
|
|
681
|
+
save_deployment_index(claude_skills_dir, index)
|
|
682
|
+
logger.info(f"Removed {skill_name} from user_requested_skills")
|
|
683
|
+
return True
|
|
684
|
+
|
|
685
|
+
|
|
686
|
+
def is_user_requested_skill(skill_name: str, claude_skills_dir: Path) -> bool:
|
|
687
|
+
"""Check if a skill is in the user_requested_skills list.
|
|
688
|
+
|
|
689
|
+
Args:
|
|
690
|
+
skill_name: Name of skill to check
|
|
691
|
+
claude_skills_dir: Path to Claude skills directory (~/.claude/skills/)
|
|
692
|
+
|
|
693
|
+
Returns:
|
|
694
|
+
True if skill is user-requested, False otherwise
|
|
695
|
+
|
|
696
|
+
Example:
|
|
697
|
+
>>> is_requested = is_user_requested_skill(
|
|
698
|
+
... "django-framework",
|
|
699
|
+
... Path.home() / ".claude" / "skills"
|
|
700
|
+
... )
|
|
701
|
+
>>> print(f"User requested: {is_requested}")
|
|
702
|
+
"""
|
|
703
|
+
user_requested = get_user_requested_skills(claude_skills_dir)
|
|
704
|
+
return skill_name in user_requested
|