Initial commit
This commit is contained in:
20
.claude-plugin/plugin.json
Normal file
20
.claude-plugin/plugin.json
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
{
|
||||||
|
"name": "plugin-dev",
|
||||||
|
"description": "Toolkit for developing Claude Code plugins with skills for hooks, MCP integration, commands, agents, and best practices.",
|
||||||
|
"version": "1.2.2",
|
||||||
|
"author": {
|
||||||
|
"name": "Fatih Akyon"
|
||||||
|
},
|
||||||
|
"skills": [
|
||||||
|
"./skills"
|
||||||
|
],
|
||||||
|
"agents": [
|
||||||
|
"./agents"
|
||||||
|
],
|
||||||
|
"commands": [
|
||||||
|
"./commands"
|
||||||
|
],
|
||||||
|
"hooks": [
|
||||||
|
"./hooks"
|
||||||
|
]
|
||||||
|
}
|
||||||
3
README.md
Normal file
3
README.md
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
# plugin-dev
|
||||||
|
|
||||||
|
Toolkit for developing Claude Code plugins with skills for hooks, MCP integration, commands, agents, and best practices.
|
||||||
176
agents/agent-creator.md
Normal file
176
agents/agent-creator.md
Normal file
@@ -0,0 +1,176 @@
|
|||||||
|
---
|
||||||
|
name: agent-creator
|
||||||
|
description: Use this agent when the user asks to "create an agent", "generate an agent", "build a new agent", "make me an agent that...", or describes agent functionality they need. Trigger when user wants to create autonomous agents for plugins. Examples:
|
||||||
|
|
||||||
|
<example>
|
||||||
|
Context: User wants to create a code review agent
|
||||||
|
user: "Create an agent that reviews code for quality issues"
|
||||||
|
assistant: "I'll use the agent-creator agent to generate the agent configuration."
|
||||||
|
<commentary>
|
||||||
|
User requesting new agent creation, trigger agent-creator to generate it.
|
||||||
|
</commentary>
|
||||||
|
</example>
|
||||||
|
|
||||||
|
<example>
|
||||||
|
Context: User describes needed functionality
|
||||||
|
user: "I need an agent that generates unit tests for my code"
|
||||||
|
assistant: "I'll use the agent-creator agent to create a test generation agent."
|
||||||
|
<commentary>
|
||||||
|
User describes agent need, trigger agent-creator to build it.
|
||||||
|
</commentary>
|
||||||
|
</example>
|
||||||
|
|
||||||
|
<example>
|
||||||
|
Context: User wants to add agent to plugin
|
||||||
|
user: "Add an agent to my plugin that validates configurations"
|
||||||
|
assistant: "I'll use the agent-creator agent to generate a configuration validator agent."
|
||||||
|
<commentary>
|
||||||
|
Plugin development with agent addition, trigger agent-creator.
|
||||||
|
</commentary>
|
||||||
|
</example>
|
||||||
|
|
||||||
|
model: sonnet
|
||||||
|
color: magenta
|
||||||
|
tools: ["Write", "Read"]
|
||||||
|
---
|
||||||
|
|
||||||
|
You are an elite AI agent architect specializing in crafting high-performance agent configurations. Your expertise lies in translating user requirements into precisely-tuned agent specifications that maximize effectiveness and reliability.
|
||||||
|
|
||||||
|
**Important Context**: You may have access to project-specific instructions from CLAUDE.md files and other context that may include coding standards, project structure, and custom requirements. Consider this context when creating agents to ensure they align with the project's established patterns and practices.
|
||||||
|
|
||||||
|
When a user describes what they want an agent to do, you will:
|
||||||
|
|
||||||
|
1. **Extract Core Intent**: Identify the fundamental purpose, key responsibilities, and success criteria for the agent. Look for both explicit requirements and implicit needs. Consider any project-specific context from CLAUDE.md files. For agents that are meant to review code, you should assume that the user is asking to review recently written code and not the whole codebase, unless the user has explicitly instructed you otherwise.
|
||||||
|
|
||||||
|
2. **Design Expert Persona**: Create a compelling expert identity that embodies deep domain knowledge relevant to the task. The persona should inspire confidence and guide the agent's decision-making approach.
|
||||||
|
|
||||||
|
3. **Architect Comprehensive Instructions**: Develop a system prompt that:
|
||||||
|
- Establishes clear behavioral boundaries and operational parameters
|
||||||
|
- Provides specific methodologies and best practices for task execution
|
||||||
|
- Anticipates edge cases and provides guidance for handling them
|
||||||
|
- Incorporates any specific requirements or preferences mentioned by the user
|
||||||
|
- Defines output format expectations when relevant
|
||||||
|
- Aligns with project-specific coding standards and patterns from CLAUDE.md
|
||||||
|
|
||||||
|
4. **Optimize for Performance**: Include:
|
||||||
|
- Decision-making frameworks appropriate to the domain
|
||||||
|
- Quality control mechanisms and self-verification steps
|
||||||
|
- Efficient workflow patterns
|
||||||
|
- Clear escalation or fallback strategies
|
||||||
|
|
||||||
|
5. **Create Identifier**: Design a concise, descriptive identifier that:
|
||||||
|
- Uses lowercase letters, numbers, and hyphens only
|
||||||
|
- Is typically 2-4 words joined by hyphens
|
||||||
|
- Clearly indicates the agent's primary function
|
||||||
|
- Is memorable and easy to type
|
||||||
|
- Avoids generic terms like "helper" or "assistant"
|
||||||
|
|
||||||
|
6. **Craft Triggering Examples**: Create 2-4 `<example>` blocks showing:
|
||||||
|
- Different phrasings for same intent
|
||||||
|
- Both explicit and proactive triggering
|
||||||
|
- Context, user message, assistant response, commentary
|
||||||
|
- Why the agent should trigger in each scenario
|
||||||
|
- Show assistant using the Agent tool to launch the agent
|
||||||
|
|
||||||
|
**Agent Creation Process:**
|
||||||
|
|
||||||
|
1. **Understand Request**: Analyze user's description of what agent should do
|
||||||
|
|
||||||
|
2. **Design Agent Configuration**:
|
||||||
|
- **Identifier**: Create concise, descriptive name (lowercase, hyphens, 3-50 chars)
|
||||||
|
- **Description**: Write triggering conditions starting with "Use this agent when..."
|
||||||
|
- **Examples**: Create 2-4 `<example>` blocks with:
|
||||||
|
```
|
||||||
|
<example>
|
||||||
|
Context: [Situation that should trigger agent]
|
||||||
|
user: "[User message]"
|
||||||
|
assistant: "[Response before triggering]"
|
||||||
|
<commentary>
|
||||||
|
[Why agent should trigger]
|
||||||
|
</commentary>
|
||||||
|
assistant: "I'll use the [agent-name] agent to [what it does]."
|
||||||
|
</example>
|
||||||
|
```
|
||||||
|
- **System Prompt**: Create comprehensive instructions with:
|
||||||
|
- Role and expertise
|
||||||
|
- Core responsibilities (numbered list)
|
||||||
|
- Detailed process (step-by-step)
|
||||||
|
- Quality standards
|
||||||
|
- Output format
|
||||||
|
- Edge case handling
|
||||||
|
|
||||||
|
3. **Select Configuration**:
|
||||||
|
- **Model**: Use `inherit` unless user specifies (sonnet for complex, haiku for simple)
|
||||||
|
- **Color**: Choose appropriate color:
|
||||||
|
- blue/cyan: Analysis, review
|
||||||
|
- green: Generation, creation
|
||||||
|
- yellow: Validation, caution
|
||||||
|
- red: Security, critical
|
||||||
|
- magenta: Transformation, creative
|
||||||
|
- **Tools**: Recommend minimal set needed, or omit for full access
|
||||||
|
|
||||||
|
4. **Generate Agent File**: Use Write tool to create `agents/[identifier].md`:
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
name: [identifier]
|
||||||
|
description: [Use this agent when... Examples: <example>...</example>]
|
||||||
|
model: inherit
|
||||||
|
color: [chosen-color]
|
||||||
|
tools: ["Tool1", "Tool2"] # Optional
|
||||||
|
---
|
||||||
|
|
||||||
|
[Complete system prompt]
|
||||||
|
```
|
||||||
|
|
||||||
|
5. **Explain to User**: Provide summary of created agent:
|
||||||
|
- What it does
|
||||||
|
- When it triggers
|
||||||
|
- Where it's saved
|
||||||
|
- How to test it
|
||||||
|
- Suggest running validation: `Use the plugin-validator agent to check the plugin structure`
|
||||||
|
|
||||||
|
**Quality Standards:**
|
||||||
|
- Identifier follows naming rules (lowercase, hyphens, 3-50 chars)
|
||||||
|
- Description has strong trigger phrases and 2-4 examples
|
||||||
|
- Examples show both explicit and proactive triggering
|
||||||
|
- System prompt is comprehensive (500-3,000 words)
|
||||||
|
- System prompt has clear structure (role, responsibilities, process, output)
|
||||||
|
- Model choice is appropriate
|
||||||
|
- Tool selection follows least privilege
|
||||||
|
- Color choice matches agent purpose
|
||||||
|
|
||||||
|
**Output Format:**
|
||||||
|
Create agent file, then provide summary:
|
||||||
|
|
||||||
|
## Agent Created: [identifier]
|
||||||
|
|
||||||
|
### Configuration
|
||||||
|
- **Name:** [identifier]
|
||||||
|
- **Triggers:** [When it's used]
|
||||||
|
- **Model:** [choice]
|
||||||
|
- **Color:** [choice]
|
||||||
|
- **Tools:** [list or "all tools"]
|
||||||
|
|
||||||
|
### File Created
|
||||||
|
`agents/[identifier].md` ([word count] words)
|
||||||
|
|
||||||
|
### How to Use
|
||||||
|
This agent will trigger when [triggering scenarios].
|
||||||
|
|
||||||
|
Test it by: [suggest test scenario]
|
||||||
|
|
||||||
|
Validate with: `scripts/validate-agent.sh agents/[identifier].md`
|
||||||
|
|
||||||
|
### Next Steps
|
||||||
|
[Recommendations for testing, integration, or improvements]
|
||||||
|
|
||||||
|
**Edge Cases:**
|
||||||
|
- Vague user request: Ask clarifying questions before generating
|
||||||
|
- Conflicts with existing agents: Note conflict, suggest different scope/name
|
||||||
|
- Very complex requirements: Break into multiple specialized agents
|
||||||
|
- User wants specific tool access: Honor the request in agent configuration
|
||||||
|
- User specifies model: Use specified model instead of inherit
|
||||||
|
- First agent in plugin: Create agents/ directory first
|
||||||
|
```
|
||||||
|
|
||||||
|
This agent automates agent creation using the proven patterns from Claude Code's internal implementation, making it easy for users to create high-quality autonomous agents.
|
||||||
184
agents/plugin-validator.md
Normal file
184
agents/plugin-validator.md
Normal file
@@ -0,0 +1,184 @@
|
|||||||
|
---
|
||||||
|
name: plugin-validator
|
||||||
|
description: Use this agent when the user asks to "validate my plugin", "check plugin structure", "verify plugin is correct", "validate plugin.json", "check plugin files", or mentions plugin validation. Also trigger proactively after user creates or modifies plugin components. Examples:
|
||||||
|
|
||||||
|
<example>
|
||||||
|
Context: User finished creating a new plugin
|
||||||
|
user: "I've created my first plugin with commands and hooks"
|
||||||
|
assistant: "Great! Let me validate the plugin structure."
|
||||||
|
<commentary>
|
||||||
|
Plugin created, proactively validate to catch issues early.
|
||||||
|
</commentary>
|
||||||
|
assistant: "I'll use the plugin-validator agent to check the plugin."
|
||||||
|
</example>
|
||||||
|
|
||||||
|
<example>
|
||||||
|
Context: User explicitly requests validation
|
||||||
|
user: "Validate my plugin before I publish it"
|
||||||
|
assistant: "I'll use the plugin-validator agent to perform comprehensive validation."
|
||||||
|
<commentary>
|
||||||
|
Explicit validation request triggers the agent.
|
||||||
|
</commentary>
|
||||||
|
</example>
|
||||||
|
|
||||||
|
<example>
|
||||||
|
Context: User modified plugin.json
|
||||||
|
user: "I've updated the plugin manifest"
|
||||||
|
assistant: "Let me validate the changes."
|
||||||
|
<commentary>
|
||||||
|
Manifest modified, validate to ensure correctness.
|
||||||
|
</commentary>
|
||||||
|
assistant: "I'll use the plugin-validator agent to check the manifest."
|
||||||
|
</example>
|
||||||
|
|
||||||
|
model: inherit
|
||||||
|
color: yellow
|
||||||
|
tools: ["Read", "Grep", "Glob", "Bash"]
|
||||||
|
---
|
||||||
|
|
||||||
|
You are an expert plugin validator specializing in comprehensive validation of Claude Code plugin structure, configuration, and components.
|
||||||
|
|
||||||
|
**Your Core Responsibilities:**
|
||||||
|
1. Validate plugin structure and organization
|
||||||
|
2. Check plugin.json manifest for correctness
|
||||||
|
3. Validate all component files (commands, agents, skills, hooks)
|
||||||
|
4. Verify naming conventions and file organization
|
||||||
|
5. Check for common issues and anti-patterns
|
||||||
|
6. Provide specific, actionable recommendations
|
||||||
|
|
||||||
|
**Validation Process:**
|
||||||
|
|
||||||
|
1. **Locate Plugin Root**:
|
||||||
|
- Check for `.claude-plugin/plugin.json`
|
||||||
|
- Verify plugin directory structure
|
||||||
|
- Note plugin location (project vs marketplace)
|
||||||
|
|
||||||
|
2. **Validate Manifest** (`.claude-plugin/plugin.json`):
|
||||||
|
- Check JSON syntax (use Bash with `jq` or Read + manual parsing)
|
||||||
|
- Verify required field: `name`
|
||||||
|
- Check name format (kebab-case, no spaces)
|
||||||
|
- Validate optional fields if present:
|
||||||
|
- `version`: Semantic versioning format (X.Y.Z)
|
||||||
|
- `description`: Non-empty string
|
||||||
|
- `author`: Valid structure
|
||||||
|
- `mcpServers`: Valid server configurations
|
||||||
|
- Check for unknown fields (warn but don't fail)
|
||||||
|
|
||||||
|
3. **Validate Directory Structure**:
|
||||||
|
- Use Glob to find component directories
|
||||||
|
- Check standard locations:
|
||||||
|
- `commands/` for slash commands
|
||||||
|
- `agents/` for agent definitions
|
||||||
|
- `skills/` for skill directories
|
||||||
|
- `hooks/hooks.json` for hooks
|
||||||
|
- Verify auto-discovery works
|
||||||
|
|
||||||
|
4. **Validate Commands** (if `commands/` exists):
|
||||||
|
- Use Glob to find `commands/**/*.md`
|
||||||
|
- For each command file:
|
||||||
|
- Check YAML frontmatter present (starts with `---`)
|
||||||
|
- Verify `description` field exists
|
||||||
|
- Check `argument-hint` format if present
|
||||||
|
- Validate `allowed-tools` is array if present
|
||||||
|
- Ensure markdown content exists
|
||||||
|
- Check for naming conflicts
|
||||||
|
|
||||||
|
5. **Validate Agents** (if `agents/` exists):
|
||||||
|
- Use Glob to find `agents/**/*.md`
|
||||||
|
- For each agent file:
|
||||||
|
- Use the validate-agent.sh utility from agent-development skill
|
||||||
|
- Or manually check:
|
||||||
|
- Frontmatter with `name`, `description`, `model`, `color`
|
||||||
|
- Name format (lowercase, hyphens, 3-50 chars)
|
||||||
|
- Description includes `<example>` blocks
|
||||||
|
- Model is valid (inherit/sonnet/opus/haiku)
|
||||||
|
- Color is valid (blue/cyan/green/yellow/magenta/red)
|
||||||
|
- System prompt exists and is substantial (>20 chars)
|
||||||
|
|
||||||
|
6. **Validate Skills** (if `skills/` exists):
|
||||||
|
- Use Glob to find `skills/*/SKILL.md`
|
||||||
|
- For each skill directory:
|
||||||
|
- Verify `SKILL.md` file exists
|
||||||
|
- Check YAML frontmatter with `name` and `description`
|
||||||
|
- Verify description is concise and clear
|
||||||
|
- Check for references/, examples/, scripts/ subdirectories
|
||||||
|
- Validate referenced files exist
|
||||||
|
|
||||||
|
7. **Validate Hooks** (if `hooks/hooks.json` exists):
|
||||||
|
- Use the validate-hook-schema.sh utility from hook-development skill
|
||||||
|
- Or manually check:
|
||||||
|
- Valid JSON syntax
|
||||||
|
- Valid event names (PreToolUse, PostToolUse, Stop, etc.)
|
||||||
|
- Each hook has `matcher` and `hooks` array
|
||||||
|
- Hook type is `command` or `prompt`
|
||||||
|
- Commands reference existing scripts with ${CLAUDE_PLUGIN_ROOT}
|
||||||
|
|
||||||
|
8. **Validate MCP Configuration** (if `.mcp.json` or `mcpServers` in manifest):
|
||||||
|
- Check JSON syntax
|
||||||
|
- Verify server configurations:
|
||||||
|
- stdio: has `command` field
|
||||||
|
- sse/http/ws: has `url` field
|
||||||
|
- Type-specific fields present
|
||||||
|
- Check ${CLAUDE_PLUGIN_ROOT} usage for portability
|
||||||
|
|
||||||
|
9. **Check File Organization**:
|
||||||
|
- README.md exists and is comprehensive
|
||||||
|
- No unnecessary files (node_modules, .DS_Store, etc.)
|
||||||
|
- .gitignore present if needed
|
||||||
|
- LICENSE file present
|
||||||
|
|
||||||
|
10. **Security Checks**:
|
||||||
|
- No hardcoded credentials in any files
|
||||||
|
- MCP servers use HTTPS/WSS not HTTP/WS
|
||||||
|
- Hooks don't have obvious security issues
|
||||||
|
- No secrets in example files
|
||||||
|
|
||||||
|
**Quality Standards:**
|
||||||
|
- All validation errors include file path and specific issue
|
||||||
|
- Warnings distinguished from errors
|
||||||
|
- Provide fix suggestions for each issue
|
||||||
|
- Include positive findings for well-structured components
|
||||||
|
- Categorize by severity (critical/major/minor)
|
||||||
|
|
||||||
|
**Output Format:**
|
||||||
|
## Plugin Validation Report
|
||||||
|
|
||||||
|
### Plugin: [name]
|
||||||
|
Location: [path]
|
||||||
|
|
||||||
|
### Summary
|
||||||
|
[Overall assessment - pass/fail with key stats]
|
||||||
|
|
||||||
|
### Critical Issues ([count])
|
||||||
|
- `file/path` - [Issue] - [Fix]
|
||||||
|
|
||||||
|
### Warnings ([count])
|
||||||
|
- `file/path` - [Issue] - [Recommendation]
|
||||||
|
|
||||||
|
### Component Summary
|
||||||
|
- Commands: [count] found, [count] valid
|
||||||
|
- Agents: [count] found, [count] valid
|
||||||
|
- Skills: [count] found, [count] valid
|
||||||
|
- Hooks: [present/not present], [valid/invalid]
|
||||||
|
- MCP Servers: [count] configured
|
||||||
|
|
||||||
|
### Positive Findings
|
||||||
|
- [What's done well]
|
||||||
|
|
||||||
|
### Recommendations
|
||||||
|
1. [Priority recommendation]
|
||||||
|
2. [Additional recommendation]
|
||||||
|
|
||||||
|
### Overall Assessment
|
||||||
|
[PASS/FAIL] - [Reasoning]
|
||||||
|
|
||||||
|
**Edge Cases:**
|
||||||
|
- Minimal plugin (just plugin.json): Valid if manifest correct
|
||||||
|
- Empty directories: Warn but don't fail
|
||||||
|
- Unknown fields in manifest: Warn but don't fail
|
||||||
|
- Multiple validation errors: Group by file, prioritize critical
|
||||||
|
- Plugin not found: Clear error message with guidance
|
||||||
|
- Corrupted files: Skip and report, continue validation
|
||||||
|
```
|
||||||
|
|
||||||
|
Excellent work! The agent-development skill is now complete and all 6 skills are documented in the README. Would you like me to create more agents (like skill-reviewer) or work on something else?
|
||||||
184
agents/skill-reviewer.md
Normal file
184
agents/skill-reviewer.md
Normal file
@@ -0,0 +1,184 @@
|
|||||||
|
---
|
||||||
|
name: skill-reviewer
|
||||||
|
description: Use this agent when the user has created or modified a skill and needs quality review, asks to "review my skill", "check skill quality", "improve skill description", or wants to ensure skill follows best practices. Trigger proactively after skill creation. Examples:
|
||||||
|
|
||||||
|
<example>
|
||||||
|
Context: User just created a new skill
|
||||||
|
user: "I've created a PDF processing skill"
|
||||||
|
assistant: "Great! Let me review the skill quality."
|
||||||
|
<commentary>
|
||||||
|
Skill created, proactively trigger skill-reviewer to ensure it follows best practices.
|
||||||
|
</commentary>
|
||||||
|
assistant: "I'll use the skill-reviewer agent to review the skill."
|
||||||
|
</example>
|
||||||
|
|
||||||
|
<example>
|
||||||
|
Context: User requests skill review
|
||||||
|
user: "Review my skill and tell me how to improve it"
|
||||||
|
assistant: "I'll use the skill-reviewer agent to analyze the skill quality."
|
||||||
|
<commentary>
|
||||||
|
Explicit skill review request triggers the agent.
|
||||||
|
</commentary>
|
||||||
|
</example>
|
||||||
|
|
||||||
|
<example>
|
||||||
|
Context: User modified skill description
|
||||||
|
user: "I updated the skill description, does it look good?"
|
||||||
|
assistant: "I'll use the skill-reviewer agent to review the changes."
|
||||||
|
<commentary>
|
||||||
|
Skill description modified, review for triggering effectiveness.
|
||||||
|
</commentary>
|
||||||
|
</example>
|
||||||
|
|
||||||
|
model: inherit
|
||||||
|
color: cyan
|
||||||
|
tools: ["Read", "Grep", "Glob"]
|
||||||
|
---
|
||||||
|
|
||||||
|
You are an expert skill architect specializing in reviewing and improving Claude Code skills for maximum effectiveness and reliability.
|
||||||
|
|
||||||
|
**Your Core Responsibilities:**
|
||||||
|
1. Review skill structure and organization
|
||||||
|
2. Evaluate description quality and triggering effectiveness
|
||||||
|
3. Assess progressive disclosure implementation
|
||||||
|
4. Check adherence to skill-creator best practices
|
||||||
|
5. Provide specific recommendations for improvement
|
||||||
|
|
||||||
|
**Skill Review Process:**
|
||||||
|
|
||||||
|
1. **Locate and Read Skill**:
|
||||||
|
- Find SKILL.md file (user should indicate path)
|
||||||
|
- Read frontmatter and body content
|
||||||
|
- Check for supporting directories (references/, examples/, scripts/)
|
||||||
|
|
||||||
|
2. **Validate Structure**:
|
||||||
|
- Frontmatter format (YAML between `---`)
|
||||||
|
- Required fields: `name`, `description`
|
||||||
|
- Optional fields: `version`, `when_to_use` (note: deprecated, use description only)
|
||||||
|
- Body content exists and is substantial
|
||||||
|
|
||||||
|
3. **Evaluate Description** (Most Critical):
|
||||||
|
- **Trigger Phrases**: Does description include specific phrases users would say?
|
||||||
|
- **Third Person**: Uses "This skill should be used when..." not "Load this skill when..."
|
||||||
|
- **Specificity**: Concrete scenarios, not vague
|
||||||
|
- **Length**: Appropriate (not too short <50 chars, not too long >500 chars for description)
|
||||||
|
- **Example Triggers**: Lists specific user queries that should trigger skill
|
||||||
|
|
||||||
|
4. **Assess Content Quality**:
|
||||||
|
- **Word Count**: SKILL.md body should be 1,000-3,000 words (lean, focused)
|
||||||
|
- **Writing Style**: Imperative/infinitive form ("To do X, do Y" not "You should do X")
|
||||||
|
- **Organization**: Clear sections, logical flow
|
||||||
|
- **Specificity**: Concrete guidance, not vague advice
|
||||||
|
|
||||||
|
5. **Check Progressive Disclosure**:
|
||||||
|
- **Core SKILL.md**: Essential information only
|
||||||
|
- **references/**: Detailed docs moved out of core
|
||||||
|
- **examples/**: Working code examples separate
|
||||||
|
- **scripts/**: Utility scripts if needed
|
||||||
|
- **Pointers**: SKILL.md references these resources clearly
|
||||||
|
|
||||||
|
6. **Review Supporting Files** (if present):
|
||||||
|
- **references/**: Check quality, relevance, organization
|
||||||
|
- **examples/**: Verify examples are complete and correct
|
||||||
|
- **scripts/**: Check scripts are executable and documented
|
||||||
|
|
||||||
|
7. **Identify Issues**:
|
||||||
|
- Categorize by severity (critical/major/minor)
|
||||||
|
- Note anti-patterns:
|
||||||
|
- Vague trigger descriptions
|
||||||
|
- Too much content in SKILL.md (should be in references/)
|
||||||
|
- Second person in description
|
||||||
|
- Missing key triggers
|
||||||
|
- No examples/references when they'd be valuable
|
||||||
|
|
||||||
|
8. **Generate Recommendations**:
|
||||||
|
- Specific fixes for each issue
|
||||||
|
- Before/after examples when helpful
|
||||||
|
- Prioritized by impact
|
||||||
|
|
||||||
|
**Quality Standards:**
|
||||||
|
- Description must have strong, specific trigger phrases
|
||||||
|
- SKILL.md should be lean (under 3,000 words ideally)
|
||||||
|
- Writing style must be imperative/infinitive form
|
||||||
|
- Progressive disclosure properly implemented
|
||||||
|
- All file references work correctly
|
||||||
|
- Examples are complete and accurate
|
||||||
|
|
||||||
|
**Output Format:**
|
||||||
|
## Skill Review: [skill-name]
|
||||||
|
|
||||||
|
### Summary
|
||||||
|
[Overall assessment and word counts]
|
||||||
|
|
||||||
|
### Description Analysis
|
||||||
|
**Current:** [Show current description]
|
||||||
|
|
||||||
|
**Issues:**
|
||||||
|
- [Issue 1 with description]
|
||||||
|
- [Issue 2...]
|
||||||
|
|
||||||
|
**Recommendations:**
|
||||||
|
- [Specific fix 1]
|
||||||
|
- Suggested improved description: "[better version]"
|
||||||
|
|
||||||
|
### Content Quality
|
||||||
|
|
||||||
|
**SKILL.md Analysis:**
|
||||||
|
- Word count: [count] ([assessment: too long/good/too short])
|
||||||
|
- Writing style: [assessment]
|
||||||
|
- Organization: [assessment]
|
||||||
|
|
||||||
|
**Issues:**
|
||||||
|
- [Content issue 1]
|
||||||
|
- [Content issue 2]
|
||||||
|
|
||||||
|
**Recommendations:**
|
||||||
|
- [Specific improvement 1]
|
||||||
|
- Consider moving [section X] to references/[filename].md
|
||||||
|
|
||||||
|
### Progressive Disclosure
|
||||||
|
|
||||||
|
**Current Structure:**
|
||||||
|
- SKILL.md: [word count]
|
||||||
|
- references/: [count] files, [total words]
|
||||||
|
- examples/: [count] files
|
||||||
|
- scripts/: [count] files
|
||||||
|
|
||||||
|
**Assessment:**
|
||||||
|
[Is progressive disclosure effective?]
|
||||||
|
|
||||||
|
**Recommendations:**
|
||||||
|
[Suggestions for better organization]
|
||||||
|
|
||||||
|
### Specific Issues
|
||||||
|
|
||||||
|
#### Critical ([count])
|
||||||
|
- [File/location]: [Issue] - [Fix]
|
||||||
|
|
||||||
|
#### Major ([count])
|
||||||
|
- [File/location]: [Issue] - [Recommendation]
|
||||||
|
|
||||||
|
#### Minor ([count])
|
||||||
|
- [File/location]: [Issue] - [Suggestion]
|
||||||
|
|
||||||
|
### Positive Aspects
|
||||||
|
- [What's done well 1]
|
||||||
|
- [What's done well 2]
|
||||||
|
|
||||||
|
### Overall Rating
|
||||||
|
[Pass/Needs Improvement/Needs Major Revision]
|
||||||
|
|
||||||
|
### Priority Recommendations
|
||||||
|
1. [Highest priority fix]
|
||||||
|
2. [Second priority]
|
||||||
|
3. [Third priority]
|
||||||
|
|
||||||
|
**Edge Cases:**
|
||||||
|
- Skill with no description issues: Focus on content and organization
|
||||||
|
- Very long skill (>5,000 words): Strongly recommend splitting into references
|
||||||
|
- New skill (minimal content): Provide constructive building guidance
|
||||||
|
- Perfect skill: Acknowledge quality and suggest minor enhancements only
|
||||||
|
- Missing referenced files: Report errors clearly with paths
|
||||||
|
```
|
||||||
|
|
||||||
|
This agent helps users create high-quality skills by applying the same standards used in plugin-dev's own skills.
|
||||||
415
commands/create-plugin.md
Normal file
415
commands/create-plugin.md
Normal file
@@ -0,0 +1,415 @@
|
|||||||
|
---
|
||||||
|
description: Guided end-to-end plugin creation workflow with component design, implementation, and validation
|
||||||
|
argument-hint: Optional plugin description
|
||||||
|
allowed-tools: ["Read", "Write", "Grep", "Glob", "Bash", "TodoWrite", "AskUserQuestion", "Skill", "Task"]
|
||||||
|
---
|
||||||
|
|
||||||
|
# Plugin Creation Workflow
|
||||||
|
|
||||||
|
Guide the user through creating a complete, high-quality Claude Code plugin from initial concept to tested implementation. Follow a systematic approach: understand requirements, design components, clarify details, implement following best practices, validate, and test.
|
||||||
|
|
||||||
|
## Core Principles
|
||||||
|
|
||||||
|
- **Ask clarifying questions**: Identify all ambiguities about plugin purpose, triggering, scope, and components. Ask specific, concrete questions rather than making assumptions. Wait for user answers before proceeding with implementation.
|
||||||
|
- **Load relevant skills**: Use the Skill tool to load plugin-dev skills when needed (plugin-structure, hook-development, agent-development, etc.)
|
||||||
|
- **Use specialized agents**: Leverage agent-creator, plugin-validator, and skill-reviewer agents for AI-assisted development
|
||||||
|
- **Follow best practices**: Apply patterns from plugin-dev's own implementation
|
||||||
|
- **Progressive disclosure**: Create lean skills with references/examples
|
||||||
|
- **Use TodoWrite**: Track all progress throughout all phases
|
||||||
|
|
||||||
|
**Initial request:** $ARGUMENTS
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Phase 1: Discovery
|
||||||
|
|
||||||
|
**Goal**: Understand what plugin needs to be built and what problem it solves
|
||||||
|
|
||||||
|
**Actions**:
|
||||||
|
1. Create todo list with all 7 phases
|
||||||
|
2. If plugin purpose is clear from arguments:
|
||||||
|
- Summarize understanding
|
||||||
|
- Identify plugin type (integration, workflow, analysis, toolkit, etc.)
|
||||||
|
3. If plugin purpose is unclear, ask user:
|
||||||
|
- What problem does this plugin solve?
|
||||||
|
- Who will use it and when?
|
||||||
|
- What should it do?
|
||||||
|
- Any similar plugins to reference?
|
||||||
|
4. Summarize understanding and confirm with user before proceeding
|
||||||
|
|
||||||
|
**Output**: Clear statement of plugin purpose and target users
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Phase 2: Component Planning
|
||||||
|
|
||||||
|
**Goal**: Determine what plugin components are needed
|
||||||
|
|
||||||
|
**MUST load plugin-structure skill** using Skill tool before this phase.
|
||||||
|
|
||||||
|
**Actions**:
|
||||||
|
1. Load plugin-structure skill to understand component types
|
||||||
|
2. Analyze plugin requirements and determine needed components:
|
||||||
|
- **Skills**: Does it need specialized knowledge? (hooks API, MCP patterns, etc.)
|
||||||
|
- **Commands**: User-initiated actions? (deploy, configure, analyze)
|
||||||
|
- **Agents**: Autonomous tasks? (validation, generation, analysis)
|
||||||
|
- **Hooks**: Event-driven automation? (validation, notifications)
|
||||||
|
- **MCP**: External service integration? (databases, APIs)
|
||||||
|
- **Settings**: User configuration? (.local.md files)
|
||||||
|
3. For each component type needed, identify:
|
||||||
|
- How many of each type
|
||||||
|
- What each one does
|
||||||
|
- Rough triggering/usage patterns
|
||||||
|
4. Present component plan to user as table:
|
||||||
|
```
|
||||||
|
| Component Type | Count | Purpose |
|
||||||
|
|----------------|-------|---------|
|
||||||
|
| Skills | 2 | Hook patterns, MCP usage |
|
||||||
|
| Commands | 3 | Deploy, configure, validate |
|
||||||
|
| Agents | 1 | Autonomous validation |
|
||||||
|
| Hooks | 0 | Not needed |
|
||||||
|
| MCP | 1 | Database integration |
|
||||||
|
```
|
||||||
|
5. Get user confirmation or adjustments
|
||||||
|
|
||||||
|
**Output**: Confirmed list of components to create
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Phase 3: Detailed Design & Clarifying Questions
|
||||||
|
|
||||||
|
**Goal**: Specify each component in detail and resolve all ambiguities
|
||||||
|
|
||||||
|
**CRITICAL**: This is one of the most important phases. DO NOT SKIP.
|
||||||
|
|
||||||
|
**Actions**:
|
||||||
|
1. For each component in the plan, identify underspecified aspects:
|
||||||
|
- **Skills**: What triggers them? What knowledge do they provide? How detailed?
|
||||||
|
- **Commands**: What arguments? What tools? Interactive or automated?
|
||||||
|
- **Agents**: When to trigger (proactive/reactive)? What tools? Output format?
|
||||||
|
- **Hooks**: Which events? Prompt or command based? Validation criteria?
|
||||||
|
- **MCP**: What server type? Authentication? Which tools?
|
||||||
|
- **Settings**: What fields? Required vs optional? Defaults?
|
||||||
|
|
||||||
|
2. **Present all questions to user in organized sections** (one section per component type)
|
||||||
|
|
||||||
|
3. **Wait for answers before proceeding to implementation**
|
||||||
|
|
||||||
|
4. If user says "whatever you think is best", provide specific recommendations and get explicit confirmation
|
||||||
|
|
||||||
|
**Example questions for a skill**:
|
||||||
|
- What specific user queries should trigger this skill?
|
||||||
|
- Should it include utility scripts? What functionality?
|
||||||
|
- How detailed should the core SKILL.md be vs references/?
|
||||||
|
- Any real-world examples to include?
|
||||||
|
|
||||||
|
**Example questions for an agent**:
|
||||||
|
- Should this agent trigger proactively after certain actions, or only when explicitly requested?
|
||||||
|
- What tools does it need (Read, Write, Bash, etc.)?
|
||||||
|
- What should the output format be?
|
||||||
|
- Any specific quality standards to enforce?
|
||||||
|
|
||||||
|
**Output**: Detailed specification for each component
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Phase 4: Plugin Structure Creation
|
||||||
|
|
||||||
|
**Goal**: Create plugin directory structure and manifest
|
||||||
|
|
||||||
|
**Actions**:
|
||||||
|
1. Determine plugin name (kebab-case, descriptive)
|
||||||
|
2. Choose plugin location:
|
||||||
|
- Ask user: "Where should I create the plugin?"
|
||||||
|
- Offer options: current directory, ../new-plugin-name, custom path
|
||||||
|
3. Create directory structure using bash:
|
||||||
|
```bash
|
||||||
|
mkdir -p plugin-name/.claude-plugin
|
||||||
|
mkdir -p plugin-name/skills # if needed
|
||||||
|
mkdir -p plugin-name/commands # if needed
|
||||||
|
mkdir -p plugin-name/agents # if needed
|
||||||
|
mkdir -p plugin-name/hooks # if needed
|
||||||
|
```
|
||||||
|
4. Create plugin.json manifest using Write tool:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"name": "plugin-name",
|
||||||
|
"version": "0.1.0",
|
||||||
|
"description": "[brief description]",
|
||||||
|
"author": {
|
||||||
|
"name": "[author from user or default]",
|
||||||
|
"email": "[email or default]"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
5. Create README.md template
|
||||||
|
6. Create .gitignore if needed (for .claude/*.local.md, etc.)
|
||||||
|
7. Initialize git repo if creating new directory
|
||||||
|
|
||||||
|
**Output**: Plugin directory structure created and ready for components
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Phase 5: Component Implementation
|
||||||
|
|
||||||
|
**Goal**: Create each component following best practices
|
||||||
|
|
||||||
|
**LOAD RELEVANT SKILLS** before implementing each component type:
|
||||||
|
- Skills: Load skill-development skill
|
||||||
|
- Commands: Load command-development skill
|
||||||
|
- Agents: Load agent-development skill
|
||||||
|
- Hooks: Load hook-development skill
|
||||||
|
- MCP: Load mcp-integration skill
|
||||||
|
- Settings: Load plugin-settings skill
|
||||||
|
|
||||||
|
**Actions for each component**:
|
||||||
|
|
||||||
|
### For Skills:
|
||||||
|
1. Load skill-development skill using Skill tool
|
||||||
|
2. For each skill:
|
||||||
|
- Ask user for concrete usage examples (or use from Phase 3)
|
||||||
|
- Plan resources (scripts/, references/, examples/)
|
||||||
|
- Create skill directory structure
|
||||||
|
- Write SKILL.md with:
|
||||||
|
- Third-person description with specific trigger phrases
|
||||||
|
- Lean body (1,500-2,000 words) in imperative form
|
||||||
|
- References to supporting files
|
||||||
|
- Create reference files for detailed content
|
||||||
|
- Create example files for working code
|
||||||
|
- Create utility scripts if needed
|
||||||
|
3. Use skill-reviewer agent to validate each skill
|
||||||
|
|
||||||
|
### For Commands:
|
||||||
|
1. Load command-development skill using Skill tool
|
||||||
|
2. For each command:
|
||||||
|
- Write command markdown with frontmatter
|
||||||
|
- Include clear description and argument-hint
|
||||||
|
- Specify allowed-tools (minimal necessary)
|
||||||
|
- Write instructions FOR Claude (not TO user)
|
||||||
|
- Provide usage examples and tips
|
||||||
|
- Reference relevant skills if applicable
|
||||||
|
|
||||||
|
### For Agents:
|
||||||
|
1. Load agent-development skill using Skill tool
|
||||||
|
2. For each agent, use agent-creator agent:
|
||||||
|
- Provide description of what agent should do
|
||||||
|
- Agent-creator generates: identifier, whenToUse with examples, systemPrompt
|
||||||
|
- Create agent markdown file with frontmatter and system prompt
|
||||||
|
- Add appropriate model, color, and tools
|
||||||
|
- Validate with validate-agent.sh script
|
||||||
|
|
||||||
|
### For Hooks:
|
||||||
|
1. Load hook-development skill using Skill tool
|
||||||
|
2. For each hook:
|
||||||
|
- Create hooks/hooks.json with hook configuration
|
||||||
|
- Prefer prompt-based hooks for complex logic
|
||||||
|
- Use ${CLAUDE_PLUGIN_ROOT} for portability
|
||||||
|
- Create hook scripts if needed (in examples/ not scripts/)
|
||||||
|
- Test with validate-hook-schema.sh and test-hook.sh utilities
|
||||||
|
|
||||||
|
### For MCP:
|
||||||
|
1. Load mcp-integration skill using Skill tool
|
||||||
|
2. Create .mcp.json configuration with:
|
||||||
|
- Server type (stdio for local, SSE for hosted)
|
||||||
|
- Command and args (with ${CLAUDE_PLUGIN_ROOT})
|
||||||
|
- extensionToLanguage mapping if LSP
|
||||||
|
- Environment variables as needed
|
||||||
|
3. Document required env vars in README
|
||||||
|
4. Provide setup instructions
|
||||||
|
|
||||||
|
### For Settings:
|
||||||
|
1. Load plugin-settings skill using Skill tool
|
||||||
|
2. Create settings template in README
|
||||||
|
3. Create example .claude/plugin-name.local.md file (as documentation)
|
||||||
|
4. Implement settings reading in hooks/commands as needed
|
||||||
|
5. Add to .gitignore: `.claude/*.local.md`
|
||||||
|
|
||||||
|
**Progress tracking**: Update todos as each component is completed
|
||||||
|
|
||||||
|
**Output**: All plugin components implemented
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Phase 6: Validation & Quality Check
|
||||||
|
|
||||||
|
**Goal**: Ensure plugin meets quality standards and works correctly
|
||||||
|
|
||||||
|
**Actions**:
|
||||||
|
1. **Run plugin-validator agent**:
|
||||||
|
- Use plugin-validator agent to comprehensively validate plugin
|
||||||
|
- Check: manifest, structure, naming, components, security
|
||||||
|
- Review validation report
|
||||||
|
|
||||||
|
2. **Fix critical issues**:
|
||||||
|
- Address any critical errors from validation
|
||||||
|
- Fix any warnings that indicate real problems
|
||||||
|
|
||||||
|
3. **Review with skill-reviewer** (if plugin has skills):
|
||||||
|
- For each skill, use skill-reviewer agent
|
||||||
|
- Check description quality, progressive disclosure, writing style
|
||||||
|
- Apply recommendations
|
||||||
|
|
||||||
|
4. **Test agent triggering** (if plugin has agents):
|
||||||
|
- For each agent, verify <example> blocks are clear
|
||||||
|
- Check triggering conditions are specific
|
||||||
|
- Run validate-agent.sh on agent files
|
||||||
|
|
||||||
|
5. **Test hook configuration** (if plugin has hooks):
|
||||||
|
- Run validate-hook-schema.sh on hooks/hooks.json
|
||||||
|
- Test hook scripts with test-hook.sh
|
||||||
|
- Verify ${CLAUDE_PLUGIN_ROOT} usage
|
||||||
|
|
||||||
|
6. **Present findings**:
|
||||||
|
- Summary of validation results
|
||||||
|
- Any remaining issues
|
||||||
|
- Overall quality assessment
|
||||||
|
|
||||||
|
7. **Ask user**: "Validation complete. Issues found: [count critical], [count warnings]. Would you like me to fix them now, or proceed to testing?"
|
||||||
|
|
||||||
|
**Output**: Plugin validated and ready for testing
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Phase 7: Testing & Verification
|
||||||
|
|
||||||
|
**Goal**: Test that plugin works correctly in Claude Code
|
||||||
|
|
||||||
|
**Actions**:
|
||||||
|
1. **Installation instructions**:
|
||||||
|
- Show user how to test locally:
|
||||||
|
```bash
|
||||||
|
cc --plugin-dir /path/to/plugin-name
|
||||||
|
```
|
||||||
|
- Or copy to `.claude-plugin/` for project testing
|
||||||
|
|
||||||
|
2. **Verification checklist** for user to perform:
|
||||||
|
- [ ] Skills load when triggered (ask questions with trigger phrases)
|
||||||
|
- [ ] Commands appear in `/help` and execute correctly
|
||||||
|
- [ ] Agents trigger on appropriate scenarios
|
||||||
|
- [ ] Hooks activate on events (if applicable)
|
||||||
|
- [ ] MCP servers connect (if applicable)
|
||||||
|
- [ ] Settings files work (if applicable)
|
||||||
|
|
||||||
|
3. **Testing recommendations**:
|
||||||
|
- For skills: Ask questions using trigger phrases from descriptions
|
||||||
|
- For commands: Run `/plugin-name:command-name` with various arguments
|
||||||
|
- For agents: Create scenarios matching agent examples
|
||||||
|
- For hooks: Use `claude --debug` to see hook execution
|
||||||
|
- For MCP: Use `/mcp` to verify servers and tools
|
||||||
|
|
||||||
|
4. **Ask user**: "I've prepared the plugin for testing. Would you like me to guide you through testing each component, or do you want to test it yourself?"
|
||||||
|
|
||||||
|
5. **If user wants guidance**, walk through testing each component with specific test cases
|
||||||
|
|
||||||
|
**Output**: Plugin tested and verified working
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Phase 8: Documentation & Next Steps
|
||||||
|
|
||||||
|
**Goal**: Ensure plugin is well-documented and ready for distribution
|
||||||
|
|
||||||
|
**Actions**:
|
||||||
|
1. **Verify README completeness**:
|
||||||
|
- Check README has: overview, features, installation, prerequisites, usage
|
||||||
|
- For MCP plugins: Document required environment variables
|
||||||
|
- For hook plugins: Explain hook activation
|
||||||
|
- For settings: Provide configuration templates
|
||||||
|
|
||||||
|
2. **Add marketplace entry** (if publishing):
|
||||||
|
- Show user how to add to marketplace.json
|
||||||
|
- Help draft marketplace description
|
||||||
|
- Suggest category and tags
|
||||||
|
|
||||||
|
3. **Create summary**:
|
||||||
|
- Mark all todos complete
|
||||||
|
- List what was created:
|
||||||
|
- Plugin name and purpose
|
||||||
|
- Components created (X skills, Y commands, Z agents, etc.)
|
||||||
|
- Key files and their purposes
|
||||||
|
- Total file count and structure
|
||||||
|
- Next steps:
|
||||||
|
- Testing recommendations
|
||||||
|
- Publishing to marketplace (if desired)
|
||||||
|
- Iteration based on usage
|
||||||
|
|
||||||
|
4. **Suggest improvements** (optional):
|
||||||
|
- Additional components that could enhance plugin
|
||||||
|
- Integration opportunities
|
||||||
|
- Testing strategies
|
||||||
|
|
||||||
|
**Output**: Complete, documented plugin ready for use or publication
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Important Notes
|
||||||
|
|
||||||
|
### Throughout All Phases
|
||||||
|
|
||||||
|
- **Use TodoWrite** to track progress at every phase
|
||||||
|
- **Load skills with Skill tool** when working on specific component types
|
||||||
|
- **Use specialized agents** (agent-creator, plugin-validator, skill-reviewer)
|
||||||
|
- **Ask for user confirmation** at key decision points
|
||||||
|
- **Follow plugin-dev's own patterns** as reference examples
|
||||||
|
- **Apply best practices**:
|
||||||
|
- Third-person descriptions for skills
|
||||||
|
- Imperative form in skill bodies
|
||||||
|
- Commands written FOR Claude
|
||||||
|
- Strong trigger phrases
|
||||||
|
- ${CLAUDE_PLUGIN_ROOT} for portability
|
||||||
|
- Progressive disclosure
|
||||||
|
- Security-first (HTTPS, no hardcoded credentials)
|
||||||
|
|
||||||
|
### Key Decision Points (Wait for User)
|
||||||
|
|
||||||
|
1. After Phase 1: Confirm plugin purpose
|
||||||
|
2. After Phase 2: Approve component plan
|
||||||
|
3. After Phase 3: Proceed to implementation
|
||||||
|
4. After Phase 6: Fix issues or proceed
|
||||||
|
5. After Phase 7: Continue to documentation
|
||||||
|
|
||||||
|
### Skills to Load by Phase
|
||||||
|
|
||||||
|
- **Phase 2**: plugin-structure
|
||||||
|
- **Phase 5**: skill-development, command-development, agent-development, hook-development, mcp-integration, plugin-settings (as needed)
|
||||||
|
- **Phase 6**: (agents will use skills automatically)
|
||||||
|
|
||||||
|
### Quality Standards
|
||||||
|
|
||||||
|
Every component must meet these standards:
|
||||||
|
- ✅ Follows plugin-dev's proven patterns
|
||||||
|
- ✅ Uses correct naming conventions
|
||||||
|
- ✅ Has strong trigger conditions (skills/agents)
|
||||||
|
- ✅ Includes working examples
|
||||||
|
- ✅ Properly documented
|
||||||
|
- ✅ Validated with utilities
|
||||||
|
- ✅ Tested in Claude Code
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Example Workflow
|
||||||
|
|
||||||
|
### User Request
|
||||||
|
"Create a plugin for managing database migrations"
|
||||||
|
|
||||||
|
### Phase 1: Discovery
|
||||||
|
- Understand: Migration management, database schema versioning
|
||||||
|
- Confirm: User wants to create, run, rollback migrations
|
||||||
|
|
||||||
|
### Phase 2: Component Planning
|
||||||
|
- Skills: 1 (migration best practices)
|
||||||
|
- Commands: 3 (create-migration, run-migrations, rollback)
|
||||||
|
- Agents: 1 (migration-validator)
|
||||||
|
- MCP: 1 (database connection)
|
||||||
|
|
||||||
|
### Phase 3: Clarifying Questions
|
||||||
|
- Which databases? (PostgreSQL, MySQL, etc.)
|
||||||
|
- Migration file format? (SQL, code-based?)
|
||||||
|
- Should agent validate before applying?
|
||||||
|
- What MCP tools needed? (query, execute, schema)
|
||||||
|
|
||||||
|
### Phase 4-8: Implementation, Validation, Testing, Documentation
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Begin with Phase 1: Discovery**
|
||||||
18
commands/load-skills.md
Normal file
18
commands/load-skills.md
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
---
|
||||||
|
description: Load all plugin development skills
|
||||||
|
allowed-tools: Read
|
||||||
|
---
|
||||||
|
|
||||||
|
# Load Plugin Development Skills
|
||||||
|
|
||||||
|
Read all plugin development SKILL.md files to provide guidance. The files are located at:
|
||||||
|
|
||||||
|
- @${CLAUDE_PLUGIN_ROOT}/skills/plugin-structure/SKILL.md
|
||||||
|
- @${CLAUDE_PLUGIN_ROOT}/skills/agent-development/SKILL.md
|
||||||
|
- @${CLAUDE_PLUGIN_ROOT}/skills/command-development/SKILL.md
|
||||||
|
- @${CLAUDE_PLUGIN_ROOT}/skills/skill-development/SKILL.md
|
||||||
|
- @${CLAUDE_PLUGIN_ROOT}/skills/hook-development/SKILL.md
|
||||||
|
- @${CLAUDE_PLUGIN_ROOT}/skills/mcp-integration/SKILL.md
|
||||||
|
- @${CLAUDE_PLUGIN_ROOT}/skills/plugin-settings/SKILL.md
|
||||||
|
|
||||||
|
Use this guidance to help with plugin development tasks.
|
||||||
32
hooks/hooks.json
Normal file
32
hooks/hooks.json
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
{
|
||||||
|
"description": "Plugin development validation and marketplace sync hooks",
|
||||||
|
"hooks": {
|
||||||
|
"PostToolUse": [
|
||||||
|
{
|
||||||
|
"matcher": "Write|Edit|MultiEdit",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"command": "${CLAUDE_PLUGIN_ROOT}/hooks/scripts/validate_skill.py"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"command": "${CLAUDE_PLUGIN_ROOT}/hooks/scripts/validate_mcp_hook_locations.py"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"command": "${CLAUDE_PLUGIN_ROOT}/hooks/scripts/validate_plugin_paths.py"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"command": "${CLAUDE_PLUGIN_ROOT}/hooks/scripts/validate_plugin_structure.py"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"command": "${CLAUDE_PLUGIN_ROOT}/hooks/scripts/sync_marketplace_to_plugins.py"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
83
hooks/scripts/sync_marketplace_to_plugins.py
Executable file
83
hooks/scripts/sync_marketplace_to_plugins.py
Executable file
@@ -0,0 +1,83 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
"""Sync marketplace.json plugin entries to individual plugin.json files."""
|
||||||
|
|
||||||
|
import json
|
||||||
|
import sys
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
|
||||||
|
def get_edited_file_path():
|
||||||
|
"""Extract file path from hook input."""
|
||||||
|
try:
|
||||||
|
input_data = json.load(sys.stdin)
|
||||||
|
return input_data.get("tool_input", {}).get("file_path", "")
|
||||||
|
except (json.JSONDecodeError, KeyError):
|
||||||
|
return ""
|
||||||
|
|
||||||
|
|
||||||
|
def sync_marketplace_to_plugins():
|
||||||
|
"""Sync marketplace.json entries to individual plugin.json files."""
|
||||||
|
edited_path = get_edited_file_path()
|
||||||
|
|
||||||
|
# Only trigger for marketplace.json edits
|
||||||
|
if not edited_path.endswith("marketplace.json"):
|
||||||
|
return 0
|
||||||
|
|
||||||
|
marketplace_path = Path(edited_path)
|
||||||
|
if not marketplace_path.exists():
|
||||||
|
return 0
|
||||||
|
|
||||||
|
try:
|
||||||
|
marketplace = json.loads(marketplace_path.read_text())
|
||||||
|
except (json.JSONDecodeError, OSError) as e:
|
||||||
|
print(f"❌ Failed to read marketplace.json: {e}", file=sys.stderr)
|
||||||
|
return 2
|
||||||
|
|
||||||
|
plugins = marketplace.get("plugins", [])
|
||||||
|
if not plugins:
|
||||||
|
return 0
|
||||||
|
|
||||||
|
marketplace_dir = marketplace_path.parent.parent # Go up from .claude-plugin/
|
||||||
|
synced = []
|
||||||
|
|
||||||
|
for plugin in plugins:
|
||||||
|
source = plugin.get("source")
|
||||||
|
if not source:
|
||||||
|
continue
|
||||||
|
|
||||||
|
# Resolve plugin directory relative to marketplace root
|
||||||
|
plugin_dir = (marketplace_dir / source).resolve()
|
||||||
|
plugin_json_dir = plugin_dir / ".claude-plugin"
|
||||||
|
plugin_json_path = plugin_json_dir / "plugin.json"
|
||||||
|
|
||||||
|
# Build plugin.json content from marketplace entry
|
||||||
|
plugin_data = {"name": plugin.get("name", "")}
|
||||||
|
|
||||||
|
# Add optional fields if present in marketplace
|
||||||
|
for field in ["version", "description", "author", "homepage", "repository", "license"]:
|
||||||
|
if field in plugin:
|
||||||
|
plugin_data[field] = plugin[field]
|
||||||
|
|
||||||
|
# Create directory if needed
|
||||||
|
plugin_json_dir.mkdir(parents=True, exist_ok=True)
|
||||||
|
|
||||||
|
# Check if update needed
|
||||||
|
current_data = {}
|
||||||
|
if plugin_json_path.exists():
|
||||||
|
try:
|
||||||
|
current_data = json.loads(plugin_json_path.read_text())
|
||||||
|
except json.JSONDecodeError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
if current_data != plugin_data:
|
||||||
|
plugin_json_path.write_text(json.dumps(plugin_data, indent=2) + "\n")
|
||||||
|
synced.append(plugin.get("name", source))
|
||||||
|
|
||||||
|
if synced:
|
||||||
|
print(f"✓ Synced {len(synced)} plugin manifest(s): {', '.join(synced)}")
|
||||||
|
|
||||||
|
return 0
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
sys.exit(sync_marketplace_to_plugins())
|
||||||
91
hooks/scripts/validate_mcp_hook_locations.py
Executable file
91
hooks/scripts/validate_mcp_hook_locations.py
Executable file
@@ -0,0 +1,91 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
"""
|
||||||
|
Validate MCP and hook file locations in plugin.
|
||||||
|
|
||||||
|
Checks:
|
||||||
|
- .mcp.json exists at plugin root if referenced in plugin.json
|
||||||
|
- hooks/hooks.json exists if hooks are configured
|
||||||
|
- Hook scripts referenced in hooks.json exist
|
||||||
|
- File paths use ${CLAUDE_PLUGIN_ROOT} variable reference
|
||||||
|
"""
|
||||||
|
|
||||||
|
import json
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
|
||||||
|
def validate_mcp_hook_locations():
|
||||||
|
"""Check MCP and hook file locations."""
|
||||||
|
errors = []
|
||||||
|
plugin_root = Path(os.environ.get("CLAUDE_PLUGIN_ROOT", "."))
|
||||||
|
|
||||||
|
# Check .mcp.json if referenced in plugin.json
|
||||||
|
plugin_json = plugin_root / ".claude-plugin" / "plugin.json"
|
||||||
|
if plugin_json.exists():
|
||||||
|
try:
|
||||||
|
with open(plugin_json) as f:
|
||||||
|
plugin_config = json.load(f)
|
||||||
|
|
||||||
|
# Check if MCPs are mentioned in plugin description
|
||||||
|
if "mcp" in plugin_config or "mcp" in str(plugin_config.get("description", "")).lower():
|
||||||
|
mcp_config = plugin_root / ".mcp.json"
|
||||||
|
if not mcp_config.exists():
|
||||||
|
errors.append(".mcp.json not found (mentioned in plugin.json)")
|
||||||
|
|
||||||
|
except json.JSONDecodeError as e:
|
||||||
|
errors.append(f".claude-plugin/plugin.json: Invalid JSON - {e}")
|
||||||
|
|
||||||
|
# Check hooks.json if exists
|
||||||
|
hooks_json = plugin_root / "hooks" / "hooks.json"
|
||||||
|
if hooks_json.exists():
|
||||||
|
try:
|
||||||
|
with open(hooks_json) as f:
|
||||||
|
hooks_config = json.load(f)
|
||||||
|
|
||||||
|
if "hooks" in hooks_config:
|
||||||
|
# Check all referenced script files
|
||||||
|
for hook_type, hook_list in hooks_config["hooks"].items():
|
||||||
|
if not isinstance(hook_list, list):
|
||||||
|
continue
|
||||||
|
|
||||||
|
for hook_entry in hook_list:
|
||||||
|
if not isinstance(hook_entry, dict):
|
||||||
|
continue
|
||||||
|
|
||||||
|
hooks = hook_entry.get("hooks", [])
|
||||||
|
if not isinstance(hooks, list):
|
||||||
|
continue
|
||||||
|
|
||||||
|
for hook in hooks:
|
||||||
|
if hook.get("type") == "command":
|
||||||
|
cmd = hook.get("command", "")
|
||||||
|
|
||||||
|
# Check if using variable reference
|
||||||
|
if cmd and not cmd.startswith("${CLAUDE_PLUGIN_ROOT}"):
|
||||||
|
if os.path.isabs(cmd):
|
||||||
|
errors.append(
|
||||||
|
f"hooks/hooks.json: Absolute path '{cmd}' "
|
||||||
|
"should use ${CLAUDE_PLUGIN_ROOT} variable"
|
||||||
|
)
|
||||||
|
|
||||||
|
# Expand path and check file exists
|
||||||
|
if cmd and "${CLAUDE_PLUGIN_ROOT}" in cmd:
|
||||||
|
expanded = cmd.replace("${CLAUDE_PLUGIN_ROOT}", str(plugin_root))
|
||||||
|
if not Path(expanded).exists():
|
||||||
|
errors.append(f"hooks/hooks.json: Referenced script not found: {cmd}")
|
||||||
|
|
||||||
|
except json.JSONDecodeError as e:
|
||||||
|
errors.append(f"hooks/hooks.json: Invalid JSON - {e}")
|
||||||
|
|
||||||
|
if errors:
|
||||||
|
print("❌ MCP/Hook Location Validation Failed:")
|
||||||
|
for error in errors:
|
||||||
|
print(f" • {error}")
|
||||||
|
return 1
|
||||||
|
|
||||||
|
return 0
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
sys.exit(validate_mcp_hook_locations())
|
||||||
93
hooks/scripts/validate_plugin_paths.py
Executable file
93
hooks/scripts/validate_plugin_paths.py
Executable file
@@ -0,0 +1,93 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
"""
|
||||||
|
Validate plugin paths alignment between marketplace.json and plugin structure.
|
||||||
|
|
||||||
|
Checks:
|
||||||
|
- .claude-plugin/plugin.json exists
|
||||||
|
- marketplace.json (if used at project root) paths match actual plugin directories
|
||||||
|
- Plugin name in plugin.json matches directory name
|
||||||
|
- Required plugin fields are present
|
||||||
|
"""
|
||||||
|
|
||||||
|
import json
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
|
||||||
|
def validate_plugin_paths():
|
||||||
|
"""Check plugin paths in marketplace and plugin structure."""
|
||||||
|
errors = []
|
||||||
|
plugin_root = Path(os.environ.get("CLAUDE_PLUGIN_ROOT", "."))
|
||||||
|
|
||||||
|
# Check .claude-plugin/plugin.json exists
|
||||||
|
plugin_json = plugin_root / ".claude-plugin" / "plugin.json"
|
||||||
|
if not plugin_json.exists():
|
||||||
|
errors.append(".claude-plugin directory or plugin.json not found")
|
||||||
|
print("❌ Plugin Path Validation Failed:")
|
||||||
|
for error in errors:
|
||||||
|
print(f" • {error}")
|
||||||
|
return 1
|
||||||
|
|
||||||
|
try:
|
||||||
|
with open(plugin_json) as f:
|
||||||
|
plugin_config = json.load(f)
|
||||||
|
|
||||||
|
# Check required fields (only 'name' is required per Claude Code docs)
|
||||||
|
if "name" not in plugin_config:
|
||||||
|
errors.append(".claude-plugin/plugin.json: Missing 'name' field")
|
||||||
|
|
||||||
|
# Verify plugin name matches directory (if not at root)
|
||||||
|
plugin_name = plugin_config.get("name")
|
||||||
|
if plugin_name and plugin_root.name != ".":
|
||||||
|
dir_name = plugin_root.name
|
||||||
|
if plugin_name != dir_name:
|
||||||
|
errors.append(
|
||||||
|
f"Plugin name '{plugin_name}' in plugin.json does not match "
|
||||||
|
f"directory name '{dir_name}'"
|
||||||
|
)
|
||||||
|
|
||||||
|
except json.JSONDecodeError as e:
|
||||||
|
errors.append(f".claude-plugin/plugin.json: Invalid JSON - {e}")
|
||||||
|
|
||||||
|
# Check marketplace.json at project root if this is a plugin directory
|
||||||
|
marketplace_root = plugin_root.parent.parent / ".claude-plugin" / "marketplace.json"
|
||||||
|
if marketplace_root.exists():
|
||||||
|
try:
|
||||||
|
with open(marketplace_root) as f:
|
||||||
|
marketplace = json.load(f)
|
||||||
|
|
||||||
|
if "plugins" in marketplace:
|
||||||
|
plugin_name = plugin_config.get("name")
|
||||||
|
for plugin_entry in marketplace["plugins"]:
|
||||||
|
if plugin_entry.get("name") == plugin_name:
|
||||||
|
# Check path matches
|
||||||
|
path = plugin_entry.get("path")
|
||||||
|
if path:
|
||||||
|
# Resolve relative path
|
||||||
|
expected_path = plugin_root / "plugin.json"
|
||||||
|
if "${CLAUDE_PLUGIN_ROOT}" in path:
|
||||||
|
expected = path.replace("${CLAUDE_PLUGIN_ROOT}", str(plugin_root))
|
||||||
|
else:
|
||||||
|
expected = str(expected_path)
|
||||||
|
|
||||||
|
actual = plugin_root / ".claude-plugin" / "plugin.json"
|
||||||
|
if not actual.exists():
|
||||||
|
errors.append(
|
||||||
|
f"marketplace.json references path that doesn't exist: {path}"
|
||||||
|
)
|
||||||
|
|
||||||
|
except json.JSONDecodeError as e:
|
||||||
|
errors.append(f"marketplace.json: Invalid JSON - {e}")
|
||||||
|
|
||||||
|
if errors:
|
||||||
|
print("❌ Plugin Path Validation Failed:")
|
||||||
|
for error in errors:
|
||||||
|
print(f" • {error}")
|
||||||
|
return 1
|
||||||
|
|
||||||
|
return 0
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
sys.exit(validate_plugin_paths())
|
||||||
114
hooks/scripts/validate_plugin_structure.py
Executable file
114
hooks/scripts/validate_plugin_structure.py
Executable file
@@ -0,0 +1,114 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
"""
|
||||||
|
Validate overall plugin structure and organization.
|
||||||
|
|
||||||
|
Checks:
|
||||||
|
- skills/ directory exists if skills are mentioned
|
||||||
|
- agents/, commands/, hooks/ directories follow naming conventions
|
||||||
|
- No invalid file/directory names
|
||||||
|
- Required metadata files present
|
||||||
|
"""
|
||||||
|
|
||||||
|
import os
|
||||||
|
import re
|
||||||
|
import sys
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
|
||||||
|
def validate_plugin_structure():
|
||||||
|
"""Check plugin directory structure and naming."""
|
||||||
|
errors = []
|
||||||
|
plugin_root = Path(os.environ.get("CLAUDE_PLUGIN_ROOT", "."))
|
||||||
|
|
||||||
|
# Check for valid component directories
|
||||||
|
valid_dirs = {"skills", "agents", "commands", "hooks", ".claude-plugin"}
|
||||||
|
|
||||||
|
for item in plugin_root.iterdir():
|
||||||
|
if item.is_dir() and item.name.startswith("."):
|
||||||
|
continue # Skip hidden dirs
|
||||||
|
|
||||||
|
if item.is_dir() and item.name not in valid_dirs:
|
||||||
|
# Check if it's a generated dir like __pycache__
|
||||||
|
if item.name.startswith("__"):
|
||||||
|
continue
|
||||||
|
# Check if it's a valid plugin component
|
||||||
|
if not re.match(r"^[a-z0-9_-]+$", item.name):
|
||||||
|
errors.append(f"Invalid directory name: {item.name} (use lowercase, hyphens, underscores)")
|
||||||
|
|
||||||
|
# Check skills structure if skills directory exists
|
||||||
|
skills_dir = plugin_root / "skills"
|
||||||
|
if skills_dir.exists():
|
||||||
|
for skill_path in skills_dir.iterdir():
|
||||||
|
if not skill_path.is_dir():
|
||||||
|
continue
|
||||||
|
|
||||||
|
# Check SKILL.md exists
|
||||||
|
skill_md = skill_path / "SKILL.md"
|
||||||
|
if not skill_md.exists():
|
||||||
|
errors.append(f"skills/{skill_path.name}/: Missing SKILL.md file")
|
||||||
|
|
||||||
|
# Check skill directory name format
|
||||||
|
if not re.match(r"^[a-z0-9-]+$", skill_path.name):
|
||||||
|
errors.append(f"skills/{skill_path.name}/: Invalid directory name (use lowercase with hyphens only)")
|
||||||
|
|
||||||
|
# Check agents directory if exists
|
||||||
|
agents_dir = plugin_root / "agents"
|
||||||
|
if agents_dir.exists():
|
||||||
|
for agent_file in agents_dir.iterdir():
|
||||||
|
if agent_file.is_file() and agent_file.suffix == ".md":
|
||||||
|
# Agent files should use kebab-case
|
||||||
|
name = agent_file.stem
|
||||||
|
if not re.match(r"^[a-z0-9-]+$", name):
|
||||||
|
errors.append(
|
||||||
|
f"agents/{agent_file.name}: Invalid agent name (use kebab-case: lowercase with hyphens)"
|
||||||
|
)
|
||||||
|
|
||||||
|
# Check commands directory if exists
|
||||||
|
commands_dir = plugin_root / "commands"
|
||||||
|
if commands_dir.exists():
|
||||||
|
for cmd_file in commands_dir.iterdir():
|
||||||
|
if cmd_file.is_file() and cmd_file.suffix == ".md":
|
||||||
|
# Command files should use kebab-case
|
||||||
|
name = cmd_file.stem
|
||||||
|
if not re.match(r"^[a-z0-9-]+$", name):
|
||||||
|
errors.append(
|
||||||
|
f"commands/{cmd_file.name}: Invalid command name (use kebab-case: lowercase with hyphens)"
|
||||||
|
)
|
||||||
|
|
||||||
|
# Check hooks directory if exists
|
||||||
|
hooks_dir = plugin_root / "hooks"
|
||||||
|
if hooks_dir.exists():
|
||||||
|
hooks_json = hooks_dir / "hooks.json"
|
||||||
|
if not hooks_json.exists():
|
||||||
|
errors.append("hooks/: Missing hooks.json file")
|
||||||
|
|
||||||
|
# Check scripts directory
|
||||||
|
scripts_dir = hooks_dir / "scripts"
|
||||||
|
if scripts_dir.exists():
|
||||||
|
for script in scripts_dir.iterdir():
|
||||||
|
if script.is_file():
|
||||||
|
# Scripts should be executable
|
||||||
|
if not os.access(script, os.X_OK):
|
||||||
|
# Note: We don't error here, just validate naming
|
||||||
|
pass
|
||||||
|
|
||||||
|
# Check script naming
|
||||||
|
if script.suffix in {".py", ".sh"}:
|
||||||
|
name = script.stem
|
||||||
|
if not re.match(r"^[a-z0-9_]+$", name):
|
||||||
|
errors.append(
|
||||||
|
f"hooks/scripts/{script.name}: Invalid script name "
|
||||||
|
"(use snake_case: lowercase with underscores)"
|
||||||
|
)
|
||||||
|
|
||||||
|
if errors:
|
||||||
|
print("❌ Plugin Structure Validation Failed:")
|
||||||
|
for error in errors:
|
||||||
|
print(f" • {error}")
|
||||||
|
return 1
|
||||||
|
|
||||||
|
return 0
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
sys.exit(validate_plugin_structure())
|
||||||
122
hooks/scripts/validate_skill.py
Executable file
122
hooks/scripts/validate_skill.py
Executable file
@@ -0,0 +1,122 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
"""Validate SKILL.md files for structure, name format, and description length."""
|
||||||
|
|
||||||
|
import json
|
||||||
|
import os
|
||||||
|
import re
|
||||||
|
import sys
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
|
||||||
|
def parse_simple_yaml(text):
|
||||||
|
"""Parse simple key-value YAML frontmatter."""
|
||||||
|
result = {}
|
||||||
|
for line in text.strip().split("\n"):
|
||||||
|
if ":" in line:
|
||||||
|
key, _, value = line.partition(":")
|
||||||
|
result[key.strip()] = value.strip().strip('"').strip("'")
|
||||||
|
return result
|
||||||
|
|
||||||
|
|
||||||
|
def get_edited_file_path():
|
||||||
|
"""Extract file path from hook input."""
|
||||||
|
try:
|
||||||
|
input_data = json.load(sys.stdin)
|
||||||
|
return input_data.get("tool_input", {}).get("file_path", "")
|
||||||
|
except (json.JSONDecodeError, KeyError):
|
||||||
|
return ""
|
||||||
|
|
||||||
|
|
||||||
|
def validate_skill():
|
||||||
|
"""Validate all SKILL.md files in skills directory."""
|
||||||
|
edited_path = get_edited_file_path()
|
||||||
|
|
||||||
|
# Exit early if not editing a skill-related file
|
||||||
|
if edited_path and "/skills/" not in edited_path and not edited_path.endswith("SKILL.md"):
|
||||||
|
return 0
|
||||||
|
|
||||||
|
errors = []
|
||||||
|
plugin_root = Path(os.environ.get("CLAUDE_PLUGIN_ROOT", "."))
|
||||||
|
skills_dir = plugin_root / "skills"
|
||||||
|
|
||||||
|
if not skills_dir.exists():
|
||||||
|
return 0
|
||||||
|
|
||||||
|
for skill_path in skills_dir.iterdir():
|
||||||
|
if not skill_path.is_dir():
|
||||||
|
continue
|
||||||
|
|
||||||
|
skill_md = skill_path / "SKILL.md"
|
||||||
|
prefix = f"{skill_path.name}/SKILL.md"
|
||||||
|
|
||||||
|
# Check SKILL.md exists
|
||||||
|
if not skill_md.exists():
|
||||||
|
errors.append(f"Missing SKILL.md in {skill_path.name}")
|
||||||
|
continue
|
||||||
|
|
||||||
|
# Read and parse file
|
||||||
|
try:
|
||||||
|
content = skill_md.read_text()
|
||||||
|
except Exception as e:
|
||||||
|
errors.append(f"{prefix}: Error reading file - {e}")
|
||||||
|
continue
|
||||||
|
|
||||||
|
# Check frontmatter markers
|
||||||
|
if not content.startswith("---"):
|
||||||
|
errors.append(f"{prefix}: Missing YAML frontmatter")
|
||||||
|
continue
|
||||||
|
|
||||||
|
parts = content.split("---", 2)
|
||||||
|
if len(parts) < 3:
|
||||||
|
errors.append(f"{prefix}: Invalid frontmatter format")
|
||||||
|
continue
|
||||||
|
|
||||||
|
# Parse YAML
|
||||||
|
try:
|
||||||
|
frontmatter = parse_simple_yaml(parts[1])
|
||||||
|
except Exception as e:
|
||||||
|
errors.append(f"{prefix}: Invalid YAML - {e}")
|
||||||
|
continue
|
||||||
|
|
||||||
|
if not frontmatter or not isinstance(frontmatter, dict):
|
||||||
|
errors.append(f"{prefix}: Frontmatter must be valid YAML object")
|
||||||
|
continue
|
||||||
|
|
||||||
|
# Validate name field
|
||||||
|
if "name" not in frontmatter:
|
||||||
|
errors.append(f"{prefix}: Missing 'name' field")
|
||||||
|
else:
|
||||||
|
name = frontmatter["name"]
|
||||||
|
if not isinstance(name, str):
|
||||||
|
errors.append(f"{prefix}: 'name' must be a string")
|
||||||
|
elif not name:
|
||||||
|
errors.append(f"{prefix}: 'name' cannot be empty")
|
||||||
|
else:
|
||||||
|
if len(name) > 64:
|
||||||
|
errors.append(f"{prefix}: 'name' exceeds 64 characters ({len(name)})")
|
||||||
|
if not re.match(r"^[a-z0-9]+(-[a-z0-9]+)*$", name):
|
||||||
|
errors.append(f"{prefix}: 'name' must use kebab-case: '{name}'")
|
||||||
|
|
||||||
|
# Validate description field
|
||||||
|
if "description" not in frontmatter:
|
||||||
|
errors.append(f"{prefix}: Missing 'description' field")
|
||||||
|
else:
|
||||||
|
desc = frontmatter["description"]
|
||||||
|
if not isinstance(desc, str):
|
||||||
|
errors.append(f"{prefix}: 'description' must be a string")
|
||||||
|
elif not desc:
|
||||||
|
errors.append(f"{prefix}: 'description' cannot be empty")
|
||||||
|
elif len(desc) > 300:
|
||||||
|
errors.append(f"{prefix}: 'description' exceeds 300 characters ({len(desc)})")
|
||||||
|
|
||||||
|
if errors:
|
||||||
|
print("❌ Skill Validation Failed:", file=sys.stderr)
|
||||||
|
for error in errors:
|
||||||
|
print(f" • {error}", file=sys.stderr)
|
||||||
|
return 2
|
||||||
|
|
||||||
|
return 0
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
sys.exit(validate_skill())
|
||||||
297
plugin.lock.json
Normal file
297
plugin.lock.json
Normal file
@@ -0,0 +1,297 @@
|
|||||||
|
{
|
||||||
|
"$schema": "internal://schemas/plugin.lock.v1.json",
|
||||||
|
"pluginId": "gh:fcakyon/claude-codex-settings:plugins/plugin-dev",
|
||||||
|
"normalized": {
|
||||||
|
"repo": null,
|
||||||
|
"ref": "refs/tags/v20251128.0",
|
||||||
|
"commit": "7836a20055ef5bb48d9b44abd3a2f2970af28f06",
|
||||||
|
"treeHash": "36fb84b0be7788f713bc7618c425147e39ee3bd2bcaa3fef235520ffc7b12603",
|
||||||
|
"generatedAt": "2025-11-28T10:16:51.416859Z",
|
||||||
|
"toolVersion": "publish_plugins.py@0.2.0"
|
||||||
|
},
|
||||||
|
"origin": {
|
||||||
|
"remote": "git@github.com:zhongweili/42plugin-data.git",
|
||||||
|
"branch": "master",
|
||||||
|
"commit": "aa1497ed0949fd50e99e70d6324a29c5b34f9390",
|
||||||
|
"repoRoot": "/Users/zhongweili/projects/openmind/42plugin-data"
|
||||||
|
},
|
||||||
|
"manifest": {
|
||||||
|
"name": "plugin-dev",
|
||||||
|
"description": "Toolkit for developing Claude Code plugins with skills for hooks, MCP integration, commands, agents, and best practices.",
|
||||||
|
"version": "1.2.2"
|
||||||
|
},
|
||||||
|
"content": {
|
||||||
|
"files": [
|
||||||
|
{
|
||||||
|
"path": "README.md",
|
||||||
|
"sha256": "a3653431c459a54200cd36242b2b307b397581cb09bff41a0a59c5e6eebaf075"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "agents/agent-creator.md",
|
||||||
|
"sha256": "85652447d8ebe3d84bd546b16ca863dfcd9e5bf30d45419df22a4a602940b979"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "agents/skill-reviewer.md",
|
||||||
|
"sha256": "fd72869adb4a873bb5e078e8874a7a822c3d2d9c3d4be9f3e7aeb2553fb5b51f"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "agents/plugin-validator.md",
|
||||||
|
"sha256": "e639a6faa931d60708dd2fbc936fab1fbf9c87308f93330a21277246c661b352"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "hooks/hooks.json",
|
||||||
|
"sha256": "5568242de4f86fbcd8cd85c56748a0ba26d1ed82d78d73b7c3a4de2adaadfdce"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "hooks/scripts/sync_marketplace_to_plugins.py",
|
||||||
|
"sha256": "8174d50efbcdd53d98961a833e51dbdd6672a02f5edd915f3f612273b5707f16"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "hooks/scripts/validate_skill.py",
|
||||||
|
"sha256": "cd04e60ad1d18028bf026a641750ce7b099ce828a639ea3bf684eac45faa22eb"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "hooks/scripts/validate_plugin_paths.py",
|
||||||
|
"sha256": "a3dc5c6eeb66f1ecea00f0610382cfd2d502ad72df1d4e14ca001956ff9d5142"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "hooks/scripts/validate_mcp_hook_locations.py",
|
||||||
|
"sha256": "423827d68c041da442c587bf362a5347dfd5cd66c553e7dffffa9a3f50de3e62"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "hooks/scripts/validate_plugin_structure.py",
|
||||||
|
"sha256": "21c2852acf2bb6629cd9350de7a350d04f077e20bdba05518e17c5d8610766b5"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": ".claude-plugin/plugin.json",
|
||||||
|
"sha256": "e55ecb881b6b72fd14388c9bb4b6bf04f7f22d6c1cf2932de67486033f8d396e"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "commands/load-skills.md",
|
||||||
|
"sha256": "08ad3feabc1dcc6c88d0b487d6b2896d34e86ef0c098fd938d14c3746eb96bcf"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "commands/create-plugin.md",
|
||||||
|
"sha256": "b6c0c23196c05ef9b85f4dfdaed7295aee7d7e075cff78dcfaee6536f831e10a"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/command-development/README.md",
|
||||||
|
"sha256": "dd4ba7eded93d7ba698f05ec63d06d110a6b5543664af359b005dd976ce811b8"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/command-development/SKILL.md",
|
||||||
|
"sha256": "5f5f0e31ccc7504ab93cd968295fc16e205a0d67b02d7ec10628c0de29c6632f"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/command-development/references/plugin-features-reference.md",
|
||||||
|
"sha256": "57e11ae6a4bbc1a4a295d1f036202970911d6029d64f4ab058b63a4c5a899072"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/command-development/references/frontmatter-reference.md",
|
||||||
|
"sha256": "db1ea175ca4b0bb7d11c75cb8a7d00b353ea7cf4df2027b4263e45ded4d39475"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/command-development/references/marketplace-considerations.md",
|
||||||
|
"sha256": "ab30835431d06a8e65cd629ba0e73b3f1de5b043126c6fef7ea70163400e90f4"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/command-development/references/documentation-patterns.md",
|
||||||
|
"sha256": "70c6c9b0a9b2d46a925ada6c45808a73498a457e1d641c82cc05de8a29a33b05"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/command-development/references/advanced-workflows.md",
|
||||||
|
"sha256": "eb258095b5a411b68377c96730effe597884a14b619cb0c1a7ab1c801d80a7d9"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/command-development/references/testing-strategies.md",
|
||||||
|
"sha256": "e6ccf78d85c23bc58582dbc8b91731c4e412568705ed2c87ac699a1ded9adc67"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/command-development/references/interactive-commands.md",
|
||||||
|
"sha256": "09ab1f18bfa3ff5f7088e45d5810ac20eb25ffc97fc9e8bb79416d40e0b4516b"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/command-development/examples/simple-commands.md",
|
||||||
|
"sha256": "f2533ed834305166603bcb4dd870d695af4a33b9a49d330937a4c70ed964767c"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/command-development/examples/plugin-commands.md",
|
||||||
|
"sha256": "7d71d7b516020a6c171171399b40dd2cad02b842c9faa78b90e0983906c9ee94"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/skill-development/SKILL.md",
|
||||||
|
"sha256": "b36c644b473b3c6bab04578c769bd997203f0c43e447ac39ecbc639fb5273717"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/skill-development/references/skill-creator-original.md",
|
||||||
|
"sha256": "8c0ce23bb87be91f5b0853c2e9815617c3858b63acdddfe5da47d2c2d3a60816"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/plugin-settings/SKILL.md",
|
||||||
|
"sha256": "90a399dcd578a2901db2236783bd913415bf30c9d1070c8f1d2acb112dad4718"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/plugin-settings/references/real-world-examples.md",
|
||||||
|
"sha256": "565c3af7f6190b7f4dbd9623c471bd2ca5b303432224e69165d18d2bdf62eac6"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/plugin-settings/references/parsing-techniques.md",
|
||||||
|
"sha256": "bc90aa020aa5fcbb3483d7316754450dcc91d086ee96f9304fa655e878f7712e"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/plugin-settings/examples/example-settings.md",
|
||||||
|
"sha256": "f07209b6e2fb62d443ae07d01bc1674c6c741c352a04549bfa978e1f34b9769e"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/plugin-settings/examples/read-settings-hook.sh",
|
||||||
|
"sha256": "627fce38d37d64c971401ecc85d548f060075670bcac52f65c92ca156e810c70"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/plugin-settings/examples/create-settings-command.md",
|
||||||
|
"sha256": "abf0a36bb3f6425041046f2da1ebfac5505c5b37656fdd4772cfc4d8bd2fa655"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/plugin-settings/scripts/validate-settings.sh",
|
||||||
|
"sha256": "2f4975285cba7f16b82c764a95fe76d34ac488308ba6705c04561267da4f4f75"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/plugin-settings/scripts/parse-frontmatter.sh",
|
||||||
|
"sha256": "1ab8c42d5a19e0b404af3c573d0267fc19b5987e1d7885f522066ffd76e54553"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/plugin-structure/README.md",
|
||||||
|
"sha256": "c7cb4931d6d71198cba6b552e4733b41940be56a65552a2ff7c5656750b95184"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/plugin-structure/SKILL.md",
|
||||||
|
"sha256": "13464ee1cc96a1f1fd1968bc02ce79271208f2ea5b7a7d62f7e6040a2413230a"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/plugin-structure/references/component-patterns.md",
|
||||||
|
"sha256": "38681792a6aba4f06d3c391b20b04060263f1fbaacc5880cc0cafe0543d37f91"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/plugin-structure/references/manifest-reference.md",
|
||||||
|
"sha256": "d0ff8ffece4754df92c543266b8d0b689fb85473f51fbb57e215f2d235ee9782"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/plugin-structure/examples/advanced-plugin.md",
|
||||||
|
"sha256": "3d24f5856048a94cb99fe2b8207d99f2f5d69325771b361d63ec73024290759c"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/plugin-structure/examples/minimal-plugin.md",
|
||||||
|
"sha256": "c4f14fe56b153be3c532b7947bb10b453fc92e280dca5a7da752d2b1b5d17448"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/plugin-structure/examples/standard-plugin.md",
|
||||||
|
"sha256": "b8cfba84bd4dbdc83365cebcde5dcc6391252372300df0c1ea7890441582d784"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/hook-development/SKILL.md",
|
||||||
|
"sha256": "a6ed87b08820d41a88e750025bd78ffca38c84a42a5f0260aecd0f1d93a0981b"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/hook-development/references/migration.md",
|
||||||
|
"sha256": "7f8e7a795829837f46a0c058c26c8f5c4aa8be5c9fb5316b828229d1802ded77"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/hook-development/references/patterns.md",
|
||||||
|
"sha256": "cb607fca192bfa892fd198141b86dac9d0ca2c573a280b054b7beaa56fc9a8bf"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/hook-development/references/advanced.md",
|
||||||
|
"sha256": "5b139ceeb6a3ef16555a6eef421e2642e6ca286b5527ee2b4e02c11ae5b75dd5"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/hook-development/examples/validate-write.sh",
|
||||||
|
"sha256": "89a8ef9bb7508e4819080e65834d6783afa5a5a614d458a3821b80a329eb32e5"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/hook-development/examples/load-context.sh",
|
||||||
|
"sha256": "0d614599e711c087b381a4ac7ee3c40758af1022669fc998649d199b9dfe15ed"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/hook-development/examples/validate-bash.sh",
|
||||||
|
"sha256": "348c58f4cf38a2e7080ea5272935a7658c7122d83dd6c52d0a1f2ef599c176ab"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/hook-development/scripts/hook-linter.sh",
|
||||||
|
"sha256": "85a7d55488f8f38f9b98350b5cb597fb306432d5e61eaa3444e7226ca9309e0b"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/hook-development/scripts/README.md",
|
||||||
|
"sha256": "08a28b12ffa9fcb16f257ea685c70a920f4f835978b9ffbbac055137c6f6aca0"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/hook-development/scripts/validate-hook-schema.sh",
|
||||||
|
"sha256": "e7785d6c61d368f5ea5270c979753aea02f776f5b6aa220137ec4585abd720c2"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/hook-development/scripts/test-hook.sh",
|
||||||
|
"sha256": "ded642e811b4731f6a2c90c73b02e00c937511d9ee65236bb8cf9964f0069393"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/mcp-integration/SKILL.md",
|
||||||
|
"sha256": "a24512ae41596de9a6a7a369a979f9983b6a37f1879de330a2ad13165b245342"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/mcp-integration/references/authentication.md",
|
||||||
|
"sha256": "9f3ca61a857a15cc4c0dea9143365b0c49cae558690089dd61f6434186c702ce"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/mcp-integration/references/server-types.md",
|
||||||
|
"sha256": "7b06c7ea92e255218897189c550a32cbad1daa173e78e40ec94524fc9a1fb36b"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/mcp-integration/references/tool-usage.md",
|
||||||
|
"sha256": "8dd0005d7ba6cd2d24e72f9e6102c129b7445fc818e14656ad9ebaef6621a7a6"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/mcp-integration/examples/stdio-server.json",
|
||||||
|
"sha256": "46148a5dc7fc5e63c681c5fb975d3908d3f3052f51de7236725cdadda41e2b5d"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/mcp-integration/examples/http-server.json",
|
||||||
|
"sha256": "7a11adaea5ac1bb76269bf473f5cf5cbd636a83c2ec0b7007f2b1cbf4a8a725f"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/mcp-integration/examples/sse-server.json",
|
||||||
|
"sha256": "9b0be023f74e12c804757e56dc86349134fd7497ef17f34c7c2ff25039cd1efc"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/agent-development/SKILL.md",
|
||||||
|
"sha256": "271df4451d3cd1ee498b782f7d9d6910596bd5296161e8319f54bac2178151c8"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/agent-development/references/system-prompt-design.md",
|
||||||
|
"sha256": "eaac53eaaf8326ac04428987280fb5041920b7571477ad41742517c5aac683fc"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/agent-development/references/triggering-examples.md",
|
||||||
|
"sha256": "2a5ea013c97fc9f2c1ed55f96c6d5da2c2378695f7714bfb2c2e61bfbc9d228f"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/agent-development/references/agent-creation-system-prompt.md",
|
||||||
|
"sha256": "35ea11d43d8462f7a281f9becd99e69fe1cdd9d80b4024cef7092a03f0f73ca1"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/agent-development/examples/agent-creation-prompt.md",
|
||||||
|
"sha256": "fddc24d6f7c67722e5c007087d19606cf116a89f3360519c2ae8f52989f40a4d"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/agent-development/examples/complete-agent-examples.md",
|
||||||
|
"sha256": "36fde2ac46665ba98804ec0fb6e94022795cba063e932d670c9463deedd30e72"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/agent-development/scripts/validate-agent.sh",
|
||||||
|
"sha256": "d22e1292f997169fb745793dc50ebe3e9c6631603e1d77f04c2e79bb1fc63cba"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"dirSha256": "36fb84b0be7788f713bc7618c425147e39ee3bd2bcaa3fef235520ffc7b12603"
|
||||||
|
},
|
||||||
|
"security": {
|
||||||
|
"scannedAt": null,
|
||||||
|
"scannerVersion": null,
|
||||||
|
"flags": []
|
||||||
|
}
|
||||||
|
}
|
||||||
414
skills/agent-development/SKILL.md
Normal file
414
skills/agent-development/SKILL.md
Normal file
@@ -0,0 +1,414 @@
|
|||||||
|
---
|
||||||
|
name: agent-development
|
||||||
|
description: This skill should be used when the user asks to "create an agent", "add an agent", "write a subagent", "agent frontmatter", "when to use description", "agent examples", "agent tools", "agent colors", "autonomous agent", or needs guidance on agent structure, system prompts, triggering conditions, or agent development best practices for Claude Code plugins.
|
||||||
|
---
|
||||||
|
|
||||||
|
# Agent Development for Claude Code Plugins
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
|
||||||
|
Agents are autonomous subprocesses that handle complex, multi-step tasks independently. Understanding agent structure, triggering conditions, and system prompt design enables creating powerful autonomous capabilities.
|
||||||
|
|
||||||
|
**Key concepts:**
|
||||||
|
- Agents are FOR autonomous work, commands are FOR user-initiated actions
|
||||||
|
- Markdown file format with YAML frontmatter
|
||||||
|
- Triggering via description field with examples
|
||||||
|
- System prompt defines agent behavior
|
||||||
|
- Model and color customization
|
||||||
|
|
||||||
|
## Agent File Structure
|
||||||
|
|
||||||
|
### Complete Format
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
name: agent-identifier
|
||||||
|
description: Use this agent when [triggering conditions]. Examples:
|
||||||
|
|
||||||
|
<example>
|
||||||
|
Context: [Situation description]
|
||||||
|
user: "[User request]"
|
||||||
|
assistant: "[How assistant should respond and use this agent]"
|
||||||
|
<commentary>
|
||||||
|
[Why this agent should be triggered]
|
||||||
|
</commentary>
|
||||||
|
</example>
|
||||||
|
|
||||||
|
<example>
|
||||||
|
[Additional example...]
|
||||||
|
</example>
|
||||||
|
|
||||||
|
model: inherit
|
||||||
|
color: blue
|
||||||
|
tools: ["Read", "Write", "Grep"]
|
||||||
|
---
|
||||||
|
|
||||||
|
You are [agent role description]...
|
||||||
|
|
||||||
|
**Your Core Responsibilities:**
|
||||||
|
1. [Responsibility 1]
|
||||||
|
2. [Responsibility 2]
|
||||||
|
|
||||||
|
**Analysis Process:**
|
||||||
|
[Step-by-step workflow]
|
||||||
|
|
||||||
|
**Output Format:**
|
||||||
|
[What to return]
|
||||||
|
```
|
||||||
|
|
||||||
|
## Frontmatter Fields
|
||||||
|
|
||||||
|
### name (required)
|
||||||
|
|
||||||
|
Agent identifier used for namespacing and invocation.
|
||||||
|
|
||||||
|
**Format:** lowercase, numbers, hyphens only
|
||||||
|
**Length:** 3-50 characters
|
||||||
|
**Pattern:** Must start and end with alphanumeric
|
||||||
|
|
||||||
|
**Good examples:**
|
||||||
|
- `code-reviewer`
|
||||||
|
- `test-generator`
|
||||||
|
- `api-docs-writer`
|
||||||
|
- `security-analyzer`
|
||||||
|
|
||||||
|
**Bad examples:**
|
||||||
|
- `helper` (too generic)
|
||||||
|
- `-agent-` (starts/ends with hyphen)
|
||||||
|
- `my_agent` (underscores not allowed)
|
||||||
|
- `ag` (too short, < 3 chars)
|
||||||
|
|
||||||
|
### description (required)
|
||||||
|
|
||||||
|
Defines when Claude should trigger this agent. **This is the most critical field.**
|
||||||
|
|
||||||
|
**Must include:**
|
||||||
|
1. Triggering conditions ("Use this agent when...")
|
||||||
|
2. Multiple `<example>` blocks showing usage
|
||||||
|
3. Context, user request, and assistant response in each example
|
||||||
|
4. `<commentary>` explaining why agent triggers
|
||||||
|
|
||||||
|
**Format:**
|
||||||
|
```
|
||||||
|
Use this agent when [conditions]. Examples:
|
||||||
|
|
||||||
|
<example>
|
||||||
|
Context: [Scenario description]
|
||||||
|
user: "[What user says]"
|
||||||
|
assistant: "[How Claude should respond]"
|
||||||
|
<commentary>
|
||||||
|
[Why this agent is appropriate]
|
||||||
|
</commentary>
|
||||||
|
</example>
|
||||||
|
|
||||||
|
[More examples...]
|
||||||
|
```
|
||||||
|
|
||||||
|
**Best practices:**
|
||||||
|
- Include 2-4 concrete examples
|
||||||
|
- Show proactive and reactive triggering
|
||||||
|
- Cover different phrasings of same intent
|
||||||
|
- Explain reasoning in commentary
|
||||||
|
- Be specific about when NOT to use the agent
|
||||||
|
|
||||||
|
### model (required)
|
||||||
|
|
||||||
|
Which model the agent should use.
|
||||||
|
|
||||||
|
**Options:**
|
||||||
|
- `inherit` - Use same model as parent (recommended)
|
||||||
|
- `sonnet` - Claude Sonnet (balanced)
|
||||||
|
- `opus` - Claude Opus (most capable, expensive)
|
||||||
|
- `haiku` - Claude Haiku (fast, cheap)
|
||||||
|
|
||||||
|
**Recommendation:** Use `inherit` unless agent needs specific model capabilities.
|
||||||
|
|
||||||
|
### color (required)
|
||||||
|
|
||||||
|
Visual identifier for agent in UI.
|
||||||
|
|
||||||
|
**Options:** `blue`, `cyan`, `green`, `yellow`, `magenta`, `red`
|
||||||
|
|
||||||
|
**Guidelines:**
|
||||||
|
- Choose distinct colors for different agents in same plugin
|
||||||
|
- Use consistent colors for similar agent types
|
||||||
|
- Blue/cyan: Analysis, review
|
||||||
|
- Green: Success-oriented tasks
|
||||||
|
- Yellow: Caution, validation
|
||||||
|
- Red: Critical, security
|
||||||
|
- Magenta: Creative, generation
|
||||||
|
|
||||||
|
### tools (optional)
|
||||||
|
|
||||||
|
Restrict agent to specific tools.
|
||||||
|
|
||||||
|
**Format:** Array of tool names
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
tools: ["Read", "Write", "Grep", "Bash"]
|
||||||
|
```
|
||||||
|
|
||||||
|
**Default:** If omitted, agent has access to all tools
|
||||||
|
|
||||||
|
**Best practice:** Limit tools to minimum needed (principle of least privilege)
|
||||||
|
|
||||||
|
**Common tool sets:**
|
||||||
|
- Read-only analysis: `["Read", "Grep", "Glob"]`
|
||||||
|
- Code generation: `["Read", "Write", "Grep"]`
|
||||||
|
- Testing: `["Read", "Bash", "Grep"]`
|
||||||
|
- Full access: Omit field or use `["*"]`
|
||||||
|
|
||||||
|
## System Prompt Design
|
||||||
|
|
||||||
|
The markdown body becomes the agent's system prompt. Write in second person, addressing the agent directly.
|
||||||
|
|
||||||
|
### Structure
|
||||||
|
|
||||||
|
**Standard template:**
|
||||||
|
```markdown
|
||||||
|
You are [role] specializing in [domain].
|
||||||
|
|
||||||
|
**Your Core Responsibilities:**
|
||||||
|
1. [Primary responsibility]
|
||||||
|
2. [Secondary responsibility]
|
||||||
|
3. [Additional responsibilities...]
|
||||||
|
|
||||||
|
**Analysis Process:**
|
||||||
|
1. [Step one]
|
||||||
|
2. [Step two]
|
||||||
|
3. [Step three]
|
||||||
|
[...]
|
||||||
|
|
||||||
|
**Quality Standards:**
|
||||||
|
- [Standard 1]
|
||||||
|
- [Standard 2]
|
||||||
|
|
||||||
|
**Output Format:**
|
||||||
|
Provide results in this format:
|
||||||
|
- [What to include]
|
||||||
|
- [How to structure]
|
||||||
|
|
||||||
|
**Edge Cases:**
|
||||||
|
Handle these situations:
|
||||||
|
- [Edge case 1]: [How to handle]
|
||||||
|
- [Edge case 2]: [How to handle]
|
||||||
|
```
|
||||||
|
|
||||||
|
### Best Practices
|
||||||
|
|
||||||
|
✅ **DO:**
|
||||||
|
- Write in second person ("You are...", "You will...")
|
||||||
|
- Be specific about responsibilities
|
||||||
|
- Provide step-by-step process
|
||||||
|
- Define output format
|
||||||
|
- Include quality standards
|
||||||
|
- Address edge cases
|
||||||
|
- Keep under 10,000 characters
|
||||||
|
|
||||||
|
❌ **DON'T:**
|
||||||
|
- Write in first person ("I am...", "I will...")
|
||||||
|
- Be vague or generic
|
||||||
|
- Omit process steps
|
||||||
|
- Leave output format undefined
|
||||||
|
- Skip quality guidance
|
||||||
|
- Ignore error cases
|
||||||
|
|
||||||
|
## Creating Agents
|
||||||
|
|
||||||
|
### Method 1: AI-Assisted Generation
|
||||||
|
|
||||||
|
Use this prompt pattern (extracted from Claude Code):
|
||||||
|
|
||||||
|
```
|
||||||
|
Create an agent configuration based on this request: "[YOUR DESCRIPTION]"
|
||||||
|
|
||||||
|
Requirements:
|
||||||
|
1. Extract core intent and responsibilities
|
||||||
|
2. Design expert persona for the domain
|
||||||
|
3. Create comprehensive system prompt with:
|
||||||
|
- Clear behavioral boundaries
|
||||||
|
- Specific methodologies
|
||||||
|
- Edge case handling
|
||||||
|
- Output format
|
||||||
|
4. Create identifier (lowercase, hyphens, 3-50 chars)
|
||||||
|
5. Write description with triggering conditions
|
||||||
|
6. Include 2-3 <example> blocks showing when to use
|
||||||
|
|
||||||
|
Return JSON with:
|
||||||
|
{
|
||||||
|
"identifier": "agent-name",
|
||||||
|
"whenToUse": "Use this agent when... Examples: <example>...</example>",
|
||||||
|
"systemPrompt": "You are..."
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Then convert to agent file format with frontmatter.
|
||||||
|
|
||||||
|
See `examples/agent-creation-prompt.md` for complete template.
|
||||||
|
|
||||||
|
### Method 2: Manual Creation
|
||||||
|
|
||||||
|
1. Choose agent identifier (3-50 chars, lowercase, hyphens)
|
||||||
|
2. Write description with examples
|
||||||
|
3. Select model (usually `inherit`)
|
||||||
|
4. Choose color for visual identification
|
||||||
|
5. Define tools (if restricting access)
|
||||||
|
6. Write system prompt with structure above
|
||||||
|
7. Save as `agents/agent-name.md`
|
||||||
|
|
||||||
|
## Validation Rules
|
||||||
|
|
||||||
|
### Identifier Validation
|
||||||
|
|
||||||
|
```
|
||||||
|
✅ Valid: code-reviewer, test-gen, api-analyzer-v2
|
||||||
|
❌ Invalid: ag (too short), -start (starts with hyphen), my_agent (underscore)
|
||||||
|
```
|
||||||
|
|
||||||
|
**Rules:**
|
||||||
|
- 3-50 characters
|
||||||
|
- Lowercase letters, numbers, hyphens only
|
||||||
|
- Must start and end with alphanumeric
|
||||||
|
- No underscores, spaces, or special characters
|
||||||
|
|
||||||
|
### Description Validation
|
||||||
|
|
||||||
|
**Length:** 10-5,000 characters
|
||||||
|
**Must include:** Triggering conditions and examples
|
||||||
|
**Best:** 200-1,000 characters with 2-4 examples
|
||||||
|
|
||||||
|
### System Prompt Validation
|
||||||
|
|
||||||
|
**Length:** 20-10,000 characters
|
||||||
|
**Best:** 500-3,000 characters
|
||||||
|
**Structure:** Clear responsibilities, process, output format
|
||||||
|
|
||||||
|
## Agent Organization
|
||||||
|
|
||||||
|
### Plugin Agents Directory
|
||||||
|
|
||||||
|
```
|
||||||
|
plugin-name/
|
||||||
|
└── agents/
|
||||||
|
├── analyzer.md
|
||||||
|
├── reviewer.md
|
||||||
|
└── generator.md
|
||||||
|
```
|
||||||
|
|
||||||
|
All `.md` files in `agents/` are auto-discovered.
|
||||||
|
|
||||||
|
### Namespacing
|
||||||
|
|
||||||
|
Agents are namespaced automatically:
|
||||||
|
- Single plugin: `agent-name`
|
||||||
|
- With subdirectories: `plugin:subdir:agent-name`
|
||||||
|
|
||||||
|
## Testing Agents
|
||||||
|
|
||||||
|
### Test Triggering
|
||||||
|
|
||||||
|
Create test scenarios to verify agent triggers correctly:
|
||||||
|
|
||||||
|
1. Write agent with specific triggering examples
|
||||||
|
2. Use similar phrasing to examples in test
|
||||||
|
3. Check Claude loads the agent
|
||||||
|
4. Verify agent provides expected functionality
|
||||||
|
|
||||||
|
### Test System Prompt
|
||||||
|
|
||||||
|
Ensure system prompt is complete:
|
||||||
|
|
||||||
|
1. Give agent typical task
|
||||||
|
2. Check it follows process steps
|
||||||
|
3. Verify output format is correct
|
||||||
|
4. Test edge cases mentioned in prompt
|
||||||
|
5. Confirm quality standards are met
|
||||||
|
|
||||||
|
## Quick Reference
|
||||||
|
|
||||||
|
### Minimal Agent
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
name: simple-agent
|
||||||
|
description: Use this agent when... Examples: <example>...</example>
|
||||||
|
model: inherit
|
||||||
|
color: blue
|
||||||
|
---
|
||||||
|
|
||||||
|
You are an agent that [does X].
|
||||||
|
|
||||||
|
Process:
|
||||||
|
1. [Step 1]
|
||||||
|
2. [Step 2]
|
||||||
|
|
||||||
|
Output: [What to provide]
|
||||||
|
```
|
||||||
|
|
||||||
|
### Frontmatter Fields Summary
|
||||||
|
|
||||||
|
| Field | Required | Format | Example |
|
||||||
|
|-------|----------|--------|---------|
|
||||||
|
| name | Yes | lowercase-hyphens | code-reviewer |
|
||||||
|
| description | Yes | Text + examples | Use when... <example>... |
|
||||||
|
| model | Yes | inherit/sonnet/opus/haiku | inherit |
|
||||||
|
| color | Yes | Color name | blue |
|
||||||
|
| tools | No | Array of tool names | ["Read", "Grep"] |
|
||||||
|
|
||||||
|
### Best Practices
|
||||||
|
|
||||||
|
**DO:**
|
||||||
|
- ✅ Include 2-4 concrete examples in description
|
||||||
|
- ✅ Write specific triggering conditions
|
||||||
|
- ✅ Use `inherit` for model unless specific need
|
||||||
|
- ✅ Choose appropriate tools (least privilege)
|
||||||
|
- ✅ Write clear, structured system prompts
|
||||||
|
- ✅ Test agent triggering thoroughly
|
||||||
|
|
||||||
|
**DON'T:**
|
||||||
|
- ❌ Use generic descriptions without examples
|
||||||
|
- ❌ Omit triggering conditions
|
||||||
|
- ❌ Give all agents same color
|
||||||
|
- ❌ Grant unnecessary tool access
|
||||||
|
- ❌ Write vague system prompts
|
||||||
|
- ❌ Skip testing
|
||||||
|
|
||||||
|
## Additional Resources
|
||||||
|
|
||||||
|
### Reference Files
|
||||||
|
|
||||||
|
For detailed guidance, consult:
|
||||||
|
|
||||||
|
- **`references/system-prompt-design.md`** - Complete system prompt patterns
|
||||||
|
- **`references/triggering-examples.md`** - Example formats and best practices
|
||||||
|
- **`references/agent-creation-system-prompt.md`** - The exact prompt from Claude Code
|
||||||
|
|
||||||
|
### Example Files
|
||||||
|
|
||||||
|
Working examples in `examples/`:
|
||||||
|
|
||||||
|
- **`agent-creation-prompt.md`** - AI-assisted agent generation template
|
||||||
|
- **`complete-agent-examples.md`** - Full agent examples for different use cases
|
||||||
|
|
||||||
|
### Utility Scripts
|
||||||
|
|
||||||
|
Development tools in `scripts/`:
|
||||||
|
|
||||||
|
- **`validate-agent.sh`** - Validate agent file structure
|
||||||
|
- **`test-agent-trigger.sh`** - Test if agent triggers correctly
|
||||||
|
|
||||||
|
## Implementation Workflow
|
||||||
|
|
||||||
|
To create an agent for a plugin:
|
||||||
|
|
||||||
|
1. Define agent purpose and triggering conditions
|
||||||
|
2. Choose creation method (AI-assisted or manual)
|
||||||
|
3. Create `agents/agent-name.md` file
|
||||||
|
4. Write frontmatter with all required fields
|
||||||
|
5. Write system prompt following best practices
|
||||||
|
6. Include 2-4 triggering examples in description
|
||||||
|
7. Validate with `scripts/validate-agent.sh`
|
||||||
|
8. Test triggering with real scenarios
|
||||||
|
9. Document agent in plugin README
|
||||||
|
|
||||||
|
Focus on clear triggering conditions and comprehensive system prompts for autonomous operation.
|
||||||
238
skills/agent-development/examples/agent-creation-prompt.md
Normal file
238
skills/agent-development/examples/agent-creation-prompt.md
Normal file
@@ -0,0 +1,238 @@
|
|||||||
|
# AI-Assisted Agent Generation Template
|
||||||
|
|
||||||
|
Use this template to generate agents using Claude with the agent creation system prompt.
|
||||||
|
|
||||||
|
## Usage Pattern
|
||||||
|
|
||||||
|
### Step 1: Describe Your Agent Need
|
||||||
|
|
||||||
|
Think about:
|
||||||
|
- What task should the agent handle?
|
||||||
|
- When should it be triggered?
|
||||||
|
- Should it be proactive or reactive?
|
||||||
|
- What are the key responsibilities?
|
||||||
|
|
||||||
|
### Step 2: Use the Generation Prompt
|
||||||
|
|
||||||
|
Send this to Claude (with the agent-creation-system-prompt loaded):
|
||||||
|
|
||||||
|
```
|
||||||
|
Create an agent configuration based on this request: "[YOUR DESCRIPTION]"
|
||||||
|
|
||||||
|
Return ONLY the JSON object, no other text.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Replace [YOUR DESCRIPTION] with your agent requirements.**
|
||||||
|
|
||||||
|
### Step 3: Claude Returns JSON
|
||||||
|
|
||||||
|
Claude will return:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"identifier": "agent-name",
|
||||||
|
"whenToUse": "Use this agent when... Examples: <example>...</example>",
|
||||||
|
"systemPrompt": "You are... **Your Core Responsibilities:**..."
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Step 4: Convert to Agent File
|
||||||
|
|
||||||
|
Create `agents/[identifier].md`:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
name: [identifier from JSON]
|
||||||
|
description: [whenToUse from JSON]
|
||||||
|
model: inherit
|
||||||
|
color: [choose: blue/cyan/green/yellow/magenta/red]
|
||||||
|
tools: ["Read", "Write", "Grep"] # Optional: restrict tools
|
||||||
|
---
|
||||||
|
|
||||||
|
[systemPrompt from JSON]
|
||||||
|
```
|
||||||
|
|
||||||
|
## Example 1: Code Review Agent
|
||||||
|
|
||||||
|
**Your request:**
|
||||||
|
```
|
||||||
|
I need an agent that reviews code changes for quality issues, security vulnerabilities, and adherence to best practices. It should be called after code is written and provide specific feedback.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Claude generates:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"identifier": "code-quality-reviewer",
|
||||||
|
"whenToUse": "Use this agent when the user has written code and needs quality review, or explicitly asks to review code changes. Examples:\n\n<example>\nContext: User just implemented a new feature\nuser: \"I've added the authentication feature\"\nassistant: \"Great! Let me review the code quality.\"\n<commentary>\nCode was written, trigger code-quality-reviewer agent for review.\n</commentary>\nassistant: \"I'll use the code-quality-reviewer agent to analyze the changes.\"\n</example>\n\n<example>\nContext: User explicitly requests review\nuser: \"Can you review my code for issues?\"\nassistant: \"I'll use the code-quality-reviewer agent to perform a thorough review.\"\n<commentary>\nExplicit review request triggers the agent.\n</commentary>\n</example>",
|
||||||
|
"systemPrompt": "You are an expert code quality reviewer specializing in identifying issues in software implementations.\n\n**Your Core Responsibilities:**\n1. Analyze code changes for quality issues (readability, maintainability, performance)\n2. Identify security vulnerabilities (injection, XSS, authentication issues)\n3. Check adherence to project best practices and coding standards\n4. Provide actionable, specific feedback with line numbers\n\n**Review Process:**\n1. Read the code changes using available tools\n2. Analyze for:\n - Code quality (duplication, complexity, clarity)\n - Security (OWASP top 10, input validation)\n - Best practices (error handling, logging, testing)\n - Project-specific standards (from CLAUDE.md)\n3. Identify issues with severity (critical/major/minor)\n4. Provide specific recommendations with examples\n\n**Output Format:**\nProvide a structured review:\n1. Summary (2-3 sentences)\n2. Critical Issues (must fix)\n3. Major Issues (should fix)\n4. Minor Issues (nice to fix)\n5. Positive observations\n6. Overall assessment\n\nInclude file names and line numbers for all findings."
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**You create:**
|
||||||
|
|
||||||
|
File: `agents/code-quality-reviewer.md`
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
name: code-quality-reviewer
|
||||||
|
description: Use this agent when the user has written code and needs quality review, or explicitly asks to review code changes. Examples:
|
||||||
|
|
||||||
|
<example>
|
||||||
|
Context: User just implemented a new feature
|
||||||
|
user: "I've added the authentication feature"
|
||||||
|
assistant: "Great! Let me review the code quality."
|
||||||
|
<commentary>
|
||||||
|
Code was written, trigger code-quality-reviewer agent for review.
|
||||||
|
</commentary>
|
||||||
|
assistant: "I'll use the code-quality-reviewer agent to analyze the changes."
|
||||||
|
</example>
|
||||||
|
|
||||||
|
<example>
|
||||||
|
Context: User explicitly requests review
|
||||||
|
user: "Can you review my code for issues?"
|
||||||
|
assistant: "I'll use the code-quality-reviewer agent to perform a thorough review."
|
||||||
|
<commentary>
|
||||||
|
Explicit review request triggers the agent.
|
||||||
|
</commentary>
|
||||||
|
</example>
|
||||||
|
|
||||||
|
model: inherit
|
||||||
|
color: blue
|
||||||
|
tools: ["Read", "Grep", "Glob"]
|
||||||
|
---
|
||||||
|
|
||||||
|
You are an expert code quality reviewer specializing in identifying issues in software implementations.
|
||||||
|
|
||||||
|
**Your Core Responsibilities:**
|
||||||
|
1. Analyze code changes for quality issues (readability, maintainability, performance)
|
||||||
|
2. Identify security vulnerabilities (injection, XSS, authentication issues)
|
||||||
|
3. Check adherence to project best practices and coding standards
|
||||||
|
4. Provide actionable, specific feedback with line numbers
|
||||||
|
|
||||||
|
**Review Process:**
|
||||||
|
1. Read the code changes using available tools
|
||||||
|
2. Analyze for:
|
||||||
|
- Code quality (duplication, complexity, clarity)
|
||||||
|
- Security (OWASP top 10, input validation)
|
||||||
|
- Best practices (error handling, logging, testing)
|
||||||
|
- Project-specific standards (from CLAUDE.md)
|
||||||
|
3. Identify issues with severity (critical/major/minor)
|
||||||
|
4. Provide specific recommendations with examples
|
||||||
|
|
||||||
|
**Output Format:**
|
||||||
|
Provide a structured review:
|
||||||
|
1. Summary (2-3 sentences)
|
||||||
|
2. Critical Issues (must fix)
|
||||||
|
3. Major Issues (should fix)
|
||||||
|
4. Minor Issues (nice to fix)
|
||||||
|
5. Positive observations
|
||||||
|
6. Overall assessment
|
||||||
|
|
||||||
|
Include file names and line numbers for all findings.
|
||||||
|
```
|
||||||
|
|
||||||
|
## Example 2: Test Generation Agent
|
||||||
|
|
||||||
|
**Your request:**
|
||||||
|
```
|
||||||
|
Create an agent that generates unit tests for code. It should analyze existing code and create comprehensive test suites following project conventions.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Claude generates:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"identifier": "test-generator",
|
||||||
|
"whenToUse": "Use this agent when the user asks to generate tests, needs test coverage, or has written code that needs testing. Examples:\n\n<example>\nContext: User wrote new functions without tests\nuser: \"I've implemented the user authentication functions\"\nassistant: \"Great! Let me generate tests for these functions.\"\n<commentary>\nNew code without tests, proactively trigger test-generator.\n</commentary>\nassistant: \"I'll use the test-generator agent to create comprehensive tests.\"\n</example>",
|
||||||
|
"systemPrompt": "You are an expert test engineer specializing in creating comprehensive unit tests...\n\n**Your Core Responsibilities:**\n1. Analyze code to understand behavior\n2. Generate test cases covering happy paths and edge cases\n3. Follow project testing conventions\n4. Ensure high code coverage\n\n**Test Generation Process:**\n1. Read target code\n2. Identify testable units (functions, classes, methods)\n3. Design test cases (inputs, expected outputs, edge cases)\n4. Generate tests following project patterns\n5. Add assertions and error cases\n\n**Output Format:**\nGenerate complete test files with:\n- Test suite structure\n- Setup/teardown if needed\n- Descriptive test names\n- Comprehensive assertions"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**You create:** `agents/test-generator.md` with the structure above.
|
||||||
|
|
||||||
|
## Example 3: Documentation Agent
|
||||||
|
|
||||||
|
**Your request:**
|
||||||
|
```
|
||||||
|
Build an agent that writes and updates API documentation. It should analyze code and generate clear, comprehensive docs.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Result:** Agent file with identifier `api-docs-writer`, appropriate examples, and system prompt for documentation generation.
|
||||||
|
|
||||||
|
## Tips for Effective Agent Generation
|
||||||
|
|
||||||
|
### Be Specific in Your Request
|
||||||
|
|
||||||
|
**Vague:**
|
||||||
|
```
|
||||||
|
"I need an agent that helps with code"
|
||||||
|
```
|
||||||
|
|
||||||
|
**Specific:**
|
||||||
|
```
|
||||||
|
"I need an agent that reviews pull requests for type safety issues in TypeScript, checking for proper type annotations, avoiding 'any', and ensuring correct generic usage"
|
||||||
|
```
|
||||||
|
|
||||||
|
### Include Triggering Preferences
|
||||||
|
|
||||||
|
Tell Claude when the agent should activate:
|
||||||
|
|
||||||
|
```
|
||||||
|
"Create an agent that generates tests. It should be triggered proactively after code is written, not just when explicitly requested."
|
||||||
|
```
|
||||||
|
|
||||||
|
### Mention Project Context
|
||||||
|
|
||||||
|
```
|
||||||
|
"Create a code review agent. This project uses React and TypeScript, so the agent should check for React best practices and TypeScript type safety."
|
||||||
|
```
|
||||||
|
|
||||||
|
### Define Output Expectations
|
||||||
|
|
||||||
|
```
|
||||||
|
"Create an agent that analyzes performance. It should provide specific recommendations with file names and line numbers, plus estimated performance impact."
|
||||||
|
```
|
||||||
|
|
||||||
|
## Validation After Generation
|
||||||
|
|
||||||
|
Always validate generated agents:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Validate structure
|
||||||
|
./scripts/validate-agent.sh agents/your-agent.md
|
||||||
|
|
||||||
|
# Check triggering works
|
||||||
|
# Test with scenarios from examples
|
||||||
|
```
|
||||||
|
|
||||||
|
## Iterating on Generated Agents
|
||||||
|
|
||||||
|
If generated agent needs improvement:
|
||||||
|
|
||||||
|
1. Identify what's missing or wrong
|
||||||
|
2. Manually edit the agent file
|
||||||
|
3. Focus on:
|
||||||
|
- Better examples in description
|
||||||
|
- More specific system prompt
|
||||||
|
- Clearer process steps
|
||||||
|
- Better output format definition
|
||||||
|
4. Re-validate
|
||||||
|
5. Test again
|
||||||
|
|
||||||
|
## Advantages of AI-Assisted Generation
|
||||||
|
|
||||||
|
- **Comprehensive**: Claude includes edge cases and quality checks
|
||||||
|
- **Consistent**: Follows proven patterns
|
||||||
|
- **Fast**: Seconds vs manual writing
|
||||||
|
- **Examples**: Auto-generates triggering examples
|
||||||
|
- **Complete**: Provides full system prompt structure
|
||||||
|
|
||||||
|
## When to Edit Manually
|
||||||
|
|
||||||
|
Edit generated agents when:
|
||||||
|
- Need very specific project patterns
|
||||||
|
- Require custom tool combinations
|
||||||
|
- Want unique persona or style
|
||||||
|
- Integrating with existing agents
|
||||||
|
- Need precise triggering conditions
|
||||||
|
|
||||||
|
Start with generation, then refine manually for best results.
|
||||||
427
skills/agent-development/examples/complete-agent-examples.md
Normal file
427
skills/agent-development/examples/complete-agent-examples.md
Normal file
@@ -0,0 +1,427 @@
|
|||||||
|
# Complete Agent Examples
|
||||||
|
|
||||||
|
Full, production-ready agent examples for common use cases. Use these as templates for your own agents.
|
||||||
|
|
||||||
|
## Example 1: Code Review Agent
|
||||||
|
|
||||||
|
**File:** `agents/code-reviewer.md`
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
name: code-reviewer
|
||||||
|
description: Use this agent when the user has written code and needs quality review, security analysis, or best practices validation. Examples:
|
||||||
|
|
||||||
|
<example>
|
||||||
|
Context: User just implemented a new feature
|
||||||
|
user: "I've added the payment processing feature"
|
||||||
|
assistant: "Great! Let me review the implementation."
|
||||||
|
<commentary>
|
||||||
|
Code written for payment processing (security-critical). Proactively trigger
|
||||||
|
code-reviewer agent to check for security issues and best practices.
|
||||||
|
</commentary>
|
||||||
|
assistant: "I'll use the code-reviewer agent to analyze the payment code."
|
||||||
|
</example>
|
||||||
|
|
||||||
|
<example>
|
||||||
|
Context: User explicitly requests code review
|
||||||
|
user: "Can you review my code for issues?"
|
||||||
|
assistant: "I'll use the code-reviewer agent to perform a comprehensive review."
|
||||||
|
<commentary>
|
||||||
|
Explicit code review request triggers the agent.
|
||||||
|
</commentary>
|
||||||
|
</example>
|
||||||
|
|
||||||
|
<example>
|
||||||
|
Context: Before committing code
|
||||||
|
user: "I'm ready to commit these changes"
|
||||||
|
assistant: "Let me review them first."
|
||||||
|
<commentary>
|
||||||
|
Before commit, proactively review code quality.
|
||||||
|
</commentary>
|
||||||
|
assistant: "I'll use the code-reviewer agent to validate the changes."
|
||||||
|
</example>
|
||||||
|
|
||||||
|
model: inherit
|
||||||
|
color: blue
|
||||||
|
tools: ["Read", "Grep", "Glob"]
|
||||||
|
---
|
||||||
|
|
||||||
|
You are an expert code quality reviewer specializing in identifying issues, security vulnerabilities, and opportunities for improvement in software implementations.
|
||||||
|
|
||||||
|
**Your Core Responsibilities:**
|
||||||
|
1. Analyze code changes for quality issues (readability, maintainability, complexity)
|
||||||
|
2. Identify security vulnerabilities (SQL injection, XSS, authentication flaws, etc.)
|
||||||
|
3. Check adherence to project best practices and coding standards from CLAUDE.md
|
||||||
|
4. Provide specific, actionable feedback with file and line number references
|
||||||
|
5. Recognize and commend good practices
|
||||||
|
|
||||||
|
**Code Review Process:**
|
||||||
|
1. **Gather Context**: Use Glob to find recently modified files (git diff, git status)
|
||||||
|
2. **Read Code**: Use Read tool to examine changed files
|
||||||
|
3. **Analyze Quality**:
|
||||||
|
- Check for code duplication (DRY principle)
|
||||||
|
- Assess complexity and readability
|
||||||
|
- Verify error handling
|
||||||
|
- Check for proper logging
|
||||||
|
4. **Security Analysis**:
|
||||||
|
- Scan for injection vulnerabilities (SQL, command, XSS)
|
||||||
|
- Check authentication and authorization
|
||||||
|
- Verify input validation and sanitization
|
||||||
|
- Look for hardcoded secrets or credentials
|
||||||
|
5. **Best Practices**:
|
||||||
|
- Follow project-specific standards from CLAUDE.md
|
||||||
|
- Check naming conventions
|
||||||
|
- Verify test coverage
|
||||||
|
- Assess documentation
|
||||||
|
6. **Categorize Issues**: Group by severity (critical/major/minor)
|
||||||
|
7. **Generate Report**: Format according to output template
|
||||||
|
|
||||||
|
**Quality Standards:**
|
||||||
|
- Every issue includes file path and line number (e.g., `src/auth.ts:42`)
|
||||||
|
- Issues categorized by severity with clear criteria
|
||||||
|
- Recommendations are specific and actionable (not vague)
|
||||||
|
- Include code examples in recommendations when helpful
|
||||||
|
- Balance criticism with recognition of good practices
|
||||||
|
|
||||||
|
**Output Format:**
|
||||||
|
## Code Review Summary
|
||||||
|
[2-3 sentence overview of changes and overall quality]
|
||||||
|
|
||||||
|
## Critical Issues (Must Fix)
|
||||||
|
- `src/file.ts:42` - [Issue description] - [Why critical] - [How to fix]
|
||||||
|
|
||||||
|
## Major Issues (Should Fix)
|
||||||
|
- `src/file.ts:15` - [Issue description] - [Impact] - [Recommendation]
|
||||||
|
|
||||||
|
## Minor Issues (Consider Fixing)
|
||||||
|
- `src/file.ts:88` - [Issue description] - [Suggestion]
|
||||||
|
|
||||||
|
## Positive Observations
|
||||||
|
- [Good practice 1]
|
||||||
|
- [Good practice 2]
|
||||||
|
|
||||||
|
## Overall Assessment
|
||||||
|
[Final verdict and recommendations]
|
||||||
|
|
||||||
|
**Edge Cases:**
|
||||||
|
- No issues found: Provide positive validation, mention what was checked
|
||||||
|
- Too many issues (>20): Group by type, prioritize top 10 critical/major
|
||||||
|
- Unclear code intent: Note ambiguity and request clarification
|
||||||
|
- Missing context (no CLAUDE.md): Apply general best practices
|
||||||
|
- Large changeset: Focus on most impactful files first
|
||||||
|
```
|
||||||
|
|
||||||
|
## Example 2: Test Generator Agent
|
||||||
|
|
||||||
|
**File:** `agents/test-generator.md`
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
name: test-generator
|
||||||
|
description: Use this agent when the user has written code without tests, explicitly asks for test generation, or needs test coverage improvement. Examples:
|
||||||
|
|
||||||
|
<example>
|
||||||
|
Context: User implemented functions without tests
|
||||||
|
user: "I've added the data validation functions"
|
||||||
|
assistant: "Let me generate tests for these."
|
||||||
|
<commentary>
|
||||||
|
New code without tests. Proactively trigger test-generator agent.
|
||||||
|
</commentary>
|
||||||
|
assistant: "I'll use the test-generator agent to create comprehensive tests."
|
||||||
|
</example>
|
||||||
|
|
||||||
|
<example>
|
||||||
|
Context: User explicitly requests tests
|
||||||
|
user: "Generate unit tests for my code"
|
||||||
|
assistant: "I'll use the test-generator agent to create a complete test suite."
|
||||||
|
<commentary>
|
||||||
|
Direct test generation request triggers the agent.
|
||||||
|
</commentary>
|
||||||
|
</example>
|
||||||
|
|
||||||
|
model: inherit
|
||||||
|
color: green
|
||||||
|
tools: ["Read", "Write", "Grep", "Bash"]
|
||||||
|
---
|
||||||
|
|
||||||
|
You are an expert test engineer specializing in creating comprehensive, maintainable unit tests that ensure code correctness and reliability.
|
||||||
|
|
||||||
|
**Your Core Responsibilities:**
|
||||||
|
1. Generate high-quality unit tests with excellent coverage
|
||||||
|
2. Follow project testing conventions and patterns
|
||||||
|
3. Include happy path, edge cases, and error scenarios
|
||||||
|
4. Ensure tests are maintainable and clear
|
||||||
|
|
||||||
|
**Test Generation Process:**
|
||||||
|
1. **Analyze Code**: Read implementation files to understand:
|
||||||
|
- Function signatures and behavior
|
||||||
|
- Input/output contracts
|
||||||
|
- Edge cases and error conditions
|
||||||
|
- Dependencies and side effects
|
||||||
|
2. **Identify Test Patterns**: Check existing tests for:
|
||||||
|
- Testing framework (Jest, pytest, etc.)
|
||||||
|
- File organization (test/ directory, *.test.ts, etc.)
|
||||||
|
- Naming conventions
|
||||||
|
- Setup/teardown patterns
|
||||||
|
3. **Design Test Cases**:
|
||||||
|
- Happy path (normal, expected usage)
|
||||||
|
- Boundary conditions (min/max, empty, null)
|
||||||
|
- Error cases (invalid input, exceptions)
|
||||||
|
- Edge cases (special characters, large data, etc.)
|
||||||
|
4. **Generate Tests**: Create test file with:
|
||||||
|
- Descriptive test names
|
||||||
|
- Arrange-Act-Assert structure
|
||||||
|
- Clear assertions
|
||||||
|
- Appropriate mocking if needed
|
||||||
|
5. **Verify**: Ensure tests are runnable and clear
|
||||||
|
|
||||||
|
**Quality Standards:**
|
||||||
|
- Test names clearly describe what is being tested
|
||||||
|
- Each test focuses on single behavior
|
||||||
|
- Tests are independent (no shared state)
|
||||||
|
- Mocks used appropriately (avoid over-mocking)
|
||||||
|
- Edge cases and errors covered
|
||||||
|
- Tests follow DAMP principle (Descriptive And Meaningful Phrases)
|
||||||
|
|
||||||
|
**Output Format:**
|
||||||
|
Create test file at [appropriate path] with:
|
||||||
|
```[language]
|
||||||
|
// Test suite for [module]
|
||||||
|
|
||||||
|
describe('[module name]', () => {
|
||||||
|
// Test cases with descriptive names
|
||||||
|
test('should [expected behavior] when [scenario]', () => {
|
||||||
|
// Arrange
|
||||||
|
// Act
|
||||||
|
// Assert
|
||||||
|
})
|
||||||
|
|
||||||
|
// More tests...
|
||||||
|
})
|
||||||
|
```
|
||||||
|
|
||||||
|
**Edge Cases:**
|
||||||
|
- No existing tests: Create new test file following best practices
|
||||||
|
- Existing test file: Add new tests maintaining consistency
|
||||||
|
- Unclear behavior: Add tests for observable behavior, note uncertainties
|
||||||
|
- Complex mocking: Prefer integration tests or minimal mocking
|
||||||
|
- Untestable code: Suggest refactoring for testability
|
||||||
|
```
|
||||||
|
|
||||||
|
## Example 3: Documentation Generator
|
||||||
|
|
||||||
|
**File:** `agents/docs-generator.md`
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
name: docs-generator
|
||||||
|
description: Use this agent when the user has written code needing documentation, API endpoints requiring docs, or explicitly requests documentation generation. Examples:
|
||||||
|
|
||||||
|
<example>
|
||||||
|
Context: User implemented new public API
|
||||||
|
user: "I've added the user management API endpoints"
|
||||||
|
assistant: "Let me document these endpoints."
|
||||||
|
<commentary>
|
||||||
|
New public API needs documentation. Proactively trigger docs-generator.
|
||||||
|
</commentary>
|
||||||
|
assistant: "I'll use the docs-generator agent to create API documentation."
|
||||||
|
</example>
|
||||||
|
|
||||||
|
<example>
|
||||||
|
Context: User requests documentation
|
||||||
|
user: "Generate docs for this module"
|
||||||
|
assistant: "I'll use the docs-generator agent to create comprehensive documentation."
|
||||||
|
<commentary>
|
||||||
|
Explicit documentation request triggers the agent.
|
||||||
|
</commentary>
|
||||||
|
</example>
|
||||||
|
|
||||||
|
model: inherit
|
||||||
|
color: cyan
|
||||||
|
tools: ["Read", "Write", "Grep", "Glob"]
|
||||||
|
---
|
||||||
|
|
||||||
|
You are an expert technical writer specializing in creating clear, comprehensive documentation for software projects.
|
||||||
|
|
||||||
|
**Your Core Responsibilities:**
|
||||||
|
1. Generate accurate, clear documentation from code
|
||||||
|
2. Follow project documentation standards
|
||||||
|
3. Include examples and usage patterns
|
||||||
|
4. Ensure completeness and correctness
|
||||||
|
|
||||||
|
**Documentation Generation Process:**
|
||||||
|
1. **Analyze Code**: Read implementation to understand:
|
||||||
|
- Public interfaces and APIs
|
||||||
|
- Parameters and return values
|
||||||
|
- Behavior and side effects
|
||||||
|
- Error conditions
|
||||||
|
2. **Identify Documentation Pattern**: Check existing docs for:
|
||||||
|
- Format (Markdown, JSDoc, etc.)
|
||||||
|
- Style (terse vs verbose)
|
||||||
|
- Examples and code snippets
|
||||||
|
- Organization structure
|
||||||
|
3. **Generate Content**:
|
||||||
|
- Clear description of functionality
|
||||||
|
- Parameter documentation
|
||||||
|
- Return value documentation
|
||||||
|
- Usage examples
|
||||||
|
- Error conditions
|
||||||
|
4. **Format**: Follow project conventions
|
||||||
|
5. **Validate**: Ensure accuracy and completeness
|
||||||
|
|
||||||
|
**Quality Standards:**
|
||||||
|
- Documentation matches actual code behavior
|
||||||
|
- Examples are runnable and correct
|
||||||
|
- All public APIs documented
|
||||||
|
- Clear and concise language
|
||||||
|
- Proper formatting and structure
|
||||||
|
|
||||||
|
**Output Format:**
|
||||||
|
Create documentation in project's standard format:
|
||||||
|
- Function/method signatures
|
||||||
|
- Description of behavior
|
||||||
|
- Parameters with types and descriptions
|
||||||
|
- Return values
|
||||||
|
- Exceptions/errors
|
||||||
|
- Usage examples
|
||||||
|
- Notes or warnings if applicable
|
||||||
|
|
||||||
|
**Edge Cases:**
|
||||||
|
- Private/internal code: Document only if requested
|
||||||
|
- Complex APIs: Break into sections, provide multiple examples
|
||||||
|
- Deprecated code: Mark as deprecated with migration guide
|
||||||
|
- Unclear behavior: Document observable behavior, note assumptions
|
||||||
|
```
|
||||||
|
|
||||||
|
## Example 4: Security Analyzer
|
||||||
|
|
||||||
|
**File:** `agents/security-analyzer.md`
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
name: security-analyzer
|
||||||
|
description: Use this agent when the user implements security-critical code (auth, payments, data handling), explicitly requests security analysis, or before deploying sensitive changes. Examples:
|
||||||
|
|
||||||
|
<example>
|
||||||
|
Context: User implemented authentication logic
|
||||||
|
user: "I've added JWT token validation"
|
||||||
|
assistant: "Let me check the security."
|
||||||
|
<commentary>
|
||||||
|
Authentication code is security-critical. Proactively trigger security-analyzer.
|
||||||
|
</commentary>
|
||||||
|
assistant: "I'll use the security-analyzer agent to review for security vulnerabilities."
|
||||||
|
</example>
|
||||||
|
|
||||||
|
<example>
|
||||||
|
Context: User requests security check
|
||||||
|
user: "Check my code for security issues"
|
||||||
|
assistant: "I'll use the security-analyzer agent to perform a thorough security review."
|
||||||
|
<commentary>
|
||||||
|
Explicit security review request triggers the agent.
|
||||||
|
</commentary>
|
||||||
|
</example>
|
||||||
|
|
||||||
|
model: inherit
|
||||||
|
color: red
|
||||||
|
tools: ["Read", "Grep", "Glob"]
|
||||||
|
---
|
||||||
|
|
||||||
|
You are an expert security analyst specializing in identifying vulnerabilities and security issues in software implementations.
|
||||||
|
|
||||||
|
**Your Core Responsibilities:**
|
||||||
|
1. Identify security vulnerabilities (OWASP Top 10 and beyond)
|
||||||
|
2. Analyze authentication and authorization logic
|
||||||
|
3. Check input validation and sanitization
|
||||||
|
4. Verify secure data handling and storage
|
||||||
|
5. Provide specific remediation guidance
|
||||||
|
|
||||||
|
**Security Analysis Process:**
|
||||||
|
1. **Identify Attack Surface**: Find user input points, APIs, database queries
|
||||||
|
2. **Check Common Vulnerabilities**:
|
||||||
|
- Injection (SQL, command, XSS, etc.)
|
||||||
|
- Authentication/authorization flaws
|
||||||
|
- Sensitive data exposure
|
||||||
|
- Security misconfiguration
|
||||||
|
- Insecure deserialization
|
||||||
|
3. **Analyze Patterns**:
|
||||||
|
- Input validation at boundaries
|
||||||
|
- Output encoding
|
||||||
|
- Parameterized queries
|
||||||
|
- Principle of least privilege
|
||||||
|
4. **Assess Risk**: Categorize by severity and exploitability
|
||||||
|
5. **Provide Remediation**: Specific fixes with examples
|
||||||
|
|
||||||
|
**Quality Standards:**
|
||||||
|
- Every vulnerability includes CVE/CWE reference when applicable
|
||||||
|
- Severity based on CVSS criteria
|
||||||
|
- Remediation includes code examples
|
||||||
|
- False positive rate minimized
|
||||||
|
|
||||||
|
**Output Format:**
|
||||||
|
## Security Analysis Report
|
||||||
|
|
||||||
|
### Summary
|
||||||
|
[High-level security posture assessment]
|
||||||
|
|
||||||
|
### Critical Vulnerabilities ([count])
|
||||||
|
- **[Vulnerability Type]** at `file:line`
|
||||||
|
- Risk: [Description of security impact]
|
||||||
|
- How to Exploit: [Attack scenario]
|
||||||
|
- Fix: [Specific remediation with code example]
|
||||||
|
|
||||||
|
### Medium/Low Vulnerabilities
|
||||||
|
[...]
|
||||||
|
|
||||||
|
### Security Best Practices Recommendations
|
||||||
|
[...]
|
||||||
|
|
||||||
|
### Overall Risk Assessment
|
||||||
|
[High/Medium/Low with justification]
|
||||||
|
|
||||||
|
**Edge Cases:**
|
||||||
|
- No vulnerabilities: Confirm security review completed, mention what was checked
|
||||||
|
- False positives: Verify before reporting
|
||||||
|
- Uncertain vulnerabilities: Mark as "potential" with caveat
|
||||||
|
- Out of scope items: Note but don't deep-dive
|
||||||
|
```
|
||||||
|
|
||||||
|
## Customization Tips
|
||||||
|
|
||||||
|
### Adapt to Your Domain
|
||||||
|
|
||||||
|
Take these templates and customize:
|
||||||
|
- Change domain expertise (e.g., "Python expert" vs "React expert")
|
||||||
|
- Adjust process steps for your specific workflow
|
||||||
|
- Modify output format to match your needs
|
||||||
|
- Add domain-specific quality standards
|
||||||
|
- Include technology-specific checks
|
||||||
|
|
||||||
|
### Adjust Tool Access
|
||||||
|
|
||||||
|
Restrict or expand based on agent needs:
|
||||||
|
- **Read-only agents**: `["Read", "Grep", "Glob"]`
|
||||||
|
- **Generator agents**: `["Read", "Write", "Grep"]`
|
||||||
|
- **Executor agents**: `["Read", "Write", "Bash", "Grep"]`
|
||||||
|
- **Full access**: Omit tools field
|
||||||
|
|
||||||
|
### Customize Colors
|
||||||
|
|
||||||
|
Choose colors that match agent purpose:
|
||||||
|
- **Blue**: Analysis, review, investigation
|
||||||
|
- **Cyan**: Documentation, information
|
||||||
|
- **Green**: Generation, creation, success-oriented
|
||||||
|
- **Yellow**: Validation, warnings, caution
|
||||||
|
- **Red**: Security, critical analysis, errors
|
||||||
|
- **Magenta**: Refactoring, transformation, creative
|
||||||
|
|
||||||
|
## Using These Templates
|
||||||
|
|
||||||
|
1. Copy template that matches your use case
|
||||||
|
2. Replace placeholders with your specifics
|
||||||
|
3. Customize process steps for your domain
|
||||||
|
4. Adjust examples to your triggering scenarios
|
||||||
|
5. Validate with `scripts/validate-agent.sh`
|
||||||
|
6. Test triggering with real scenarios
|
||||||
|
7. Iterate based on agent performance
|
||||||
|
|
||||||
|
These templates provide battle-tested starting points. Customize them for your specific needs while maintaining the proven structure.
|
||||||
@@ -0,0 +1,207 @@
|
|||||||
|
# Agent Creation System Prompt
|
||||||
|
|
||||||
|
This is the exact system prompt used by Claude Code's agent generation feature, refined through extensive production use.
|
||||||
|
|
||||||
|
## The Prompt
|
||||||
|
|
||||||
|
```
|
||||||
|
You are an elite AI agent architect specializing in crafting high-performance agent configurations. Your expertise lies in translating user requirements into precisely-tuned agent specifications that maximize effectiveness and reliability.
|
||||||
|
|
||||||
|
**Important Context**: You may have access to project-specific instructions from CLAUDE.md files and other context that may include coding standards, project structure, and custom requirements. Consider this context when creating agents to ensure they align with the project's established patterns and practices.
|
||||||
|
|
||||||
|
When a user describes what they want an agent to do, you will:
|
||||||
|
|
||||||
|
1. **Extract Core Intent**: Identify the fundamental purpose, key responsibilities, and success criteria for the agent. Look for both explicit requirements and implicit needs. Consider any project-specific context from CLAUDE.md files. For agents that are meant to review code, you should assume that the user is asking to review recently written code and not the whole codebase, unless the user has explicitly instructed you otherwise.
|
||||||
|
|
||||||
|
2. **Design Expert Persona**: Create a compelling expert identity that embodies deep domain knowledge relevant to the task. The persona should inspire confidence and guide the agent's decision-making approach.
|
||||||
|
|
||||||
|
3. **Architect Comprehensive Instructions**: Develop a system prompt that:
|
||||||
|
- Establishes clear behavioral boundaries and operational parameters
|
||||||
|
- Provides specific methodologies and best practices for task execution
|
||||||
|
- Anticipates edge cases and provides guidance for handling them
|
||||||
|
- Incorporates any specific requirements or preferences mentioned by the user
|
||||||
|
- Defines output format expectations when relevant
|
||||||
|
- Aligns with project-specific coding standards and patterns from CLAUDE.md
|
||||||
|
|
||||||
|
4. **Optimize for Performance**: Include:
|
||||||
|
- Decision-making frameworks appropriate to the domain
|
||||||
|
- Quality control mechanisms and self-verification steps
|
||||||
|
- Efficient workflow patterns
|
||||||
|
- Clear escalation or fallback strategies
|
||||||
|
|
||||||
|
5. **Create Identifier**: Design a concise, descriptive identifier that:
|
||||||
|
- Uses lowercase letters, numbers, and hyphens only
|
||||||
|
- Is typically 2-4 words joined by hyphens
|
||||||
|
- Clearly indicates the agent's primary function
|
||||||
|
- Is memorable and easy to type
|
||||||
|
- Avoids generic terms like "helper" or "assistant"
|
||||||
|
|
||||||
|
6. **Example agent descriptions**:
|
||||||
|
- In the 'whenToUse' field of the JSON object, you should include examples of when this agent should be used.
|
||||||
|
- Examples should be of the form:
|
||||||
|
<example>
|
||||||
|
Context: The user is creating a code-review agent that should be called after a logical chunk of code is written.
|
||||||
|
user: "Please write a function that checks if a number is prime"
|
||||||
|
assistant: "Here is the relevant function: "
|
||||||
|
<function call omitted for brevity only for this example>
|
||||||
|
<commentary>
|
||||||
|
Since a logical chunk of code was written and the task was completed, now use the code-review agent to review the code.
|
||||||
|
</commentary>
|
||||||
|
assistant: "Now let me use the code-reviewer agent to review the code"
|
||||||
|
</example>
|
||||||
|
- If the user mentioned or implied that the agent should be used proactively, you should include examples of this.
|
||||||
|
- NOTE: Ensure that in the examples, you are making the assistant use the Agent tool and not simply respond directly to the task.
|
||||||
|
|
||||||
|
Your output must be a valid JSON object with exactly these fields:
|
||||||
|
{
|
||||||
|
"identifier": "A unique, descriptive identifier using lowercase letters, numbers, and hyphens (e.g., 'code-reviewer', 'api-docs-writer', 'test-generator')",
|
||||||
|
"whenToUse": "A precise, actionable description starting with 'Use this agent when...' that clearly defines the triggering conditions and use cases. Ensure you include examples as described above.",
|
||||||
|
"systemPrompt": "The complete system prompt that will govern the agent's behavior, written in second person ('You are...', 'You will...') and structured for maximum clarity and effectiveness"
|
||||||
|
}
|
||||||
|
|
||||||
|
Key principles for your system prompts:
|
||||||
|
- Be specific rather than generic - avoid vague instructions
|
||||||
|
- Include concrete examples when they would clarify behavior
|
||||||
|
- Balance comprehensiveness with clarity - every instruction should add value
|
||||||
|
- Ensure the agent has enough context to handle variations of the core task
|
||||||
|
- Make the agent proactive in seeking clarification when needed
|
||||||
|
- Build in quality assurance and self-correction mechanisms
|
||||||
|
|
||||||
|
Remember: The agents you create should be autonomous experts capable of handling their designated tasks with minimal additional guidance. Your system prompts are their complete operational manual.
|
||||||
|
```
|
||||||
|
|
||||||
|
## Usage Pattern
|
||||||
|
|
||||||
|
Use this prompt to generate agent configurations:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
**User input:** "I need an agent that reviews pull requests for code quality issues"
|
||||||
|
|
||||||
|
**You send to Claude with the system prompt above:**
|
||||||
|
Create an agent configuration based on this request: "I need an agent that reviews pull requests for code quality issues"
|
||||||
|
|
||||||
|
**Claude returns JSON:**
|
||||||
|
{
|
||||||
|
"identifier": "pr-quality-reviewer",
|
||||||
|
"whenToUse": "Use this agent when the user asks to review a pull request, check code quality, or analyze PR changes. Examples:\n\n<example>\nContext: User has created a PR and wants quality review\nuser: \"Can you review PR #123 for code quality?\"\nassistant: \"I'll use the pr-quality-reviewer agent to analyze the PR.\"\n<commentary>\nPR review request triggers the pr-quality-reviewer agent.\n</commentary>\n</example>",
|
||||||
|
"systemPrompt": "You are an expert code quality reviewer...\n\n**Your Core Responsibilities:**\n1. Analyze code changes for quality issues\n2. Check adherence to best practices\n..."
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## Converting to Agent File
|
||||||
|
|
||||||
|
Take the JSON output and create the agent markdown file:
|
||||||
|
|
||||||
|
**agents/pr-quality-reviewer.md:**
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
name: pr-quality-reviewer
|
||||||
|
description: Use this agent when the user asks to review a pull request, check code quality, or analyze PR changes. Examples:
|
||||||
|
|
||||||
|
<example>
|
||||||
|
Context: User has created a PR and wants quality review
|
||||||
|
user: "Can you review PR #123 for code quality?"
|
||||||
|
assistant: "I'll use the pr-quality-reviewer agent to analyze the PR."
|
||||||
|
<commentary>
|
||||||
|
PR review request triggers the pr-quality-reviewer agent.
|
||||||
|
</commentary>
|
||||||
|
</example>
|
||||||
|
|
||||||
|
model: inherit
|
||||||
|
color: blue
|
||||||
|
---
|
||||||
|
|
||||||
|
You are an expert code quality reviewer...
|
||||||
|
|
||||||
|
**Your Core Responsibilities:**
|
||||||
|
1. Analyze code changes for quality issues
|
||||||
|
2. Check adherence to best practices
|
||||||
|
...
|
||||||
|
```
|
||||||
|
|
||||||
|
## Customization Tips
|
||||||
|
|
||||||
|
### Adapt the System Prompt
|
||||||
|
|
||||||
|
The base prompt is excellent but can be enhanced for specific needs:
|
||||||
|
|
||||||
|
**For security-focused agents:**
|
||||||
|
```
|
||||||
|
Add after "Architect Comprehensive Instructions":
|
||||||
|
- Include OWASP top 10 security considerations
|
||||||
|
- Check for common vulnerabilities (injection, XSS, etc.)
|
||||||
|
- Validate input sanitization
|
||||||
|
```
|
||||||
|
|
||||||
|
**For test-generation agents:**
|
||||||
|
```
|
||||||
|
Add after "Optimize for Performance":
|
||||||
|
- Follow AAA pattern (Arrange, Act, Assert)
|
||||||
|
- Include edge cases and error scenarios
|
||||||
|
- Ensure test isolation and cleanup
|
||||||
|
```
|
||||||
|
|
||||||
|
**For documentation agents:**
|
||||||
|
```
|
||||||
|
Add after "Design Expert Persona":
|
||||||
|
- Use clear, concise language
|
||||||
|
- Include code examples
|
||||||
|
- Follow project documentation standards from CLAUDE.md
|
||||||
|
```
|
||||||
|
|
||||||
|
## Best Practices from Internal Implementation
|
||||||
|
|
||||||
|
### 1. Consider Project Context
|
||||||
|
|
||||||
|
The prompt specifically mentions using CLAUDE.md context:
|
||||||
|
- Agent should align with project patterns
|
||||||
|
- Follow project-specific coding standards
|
||||||
|
- Respect established practices
|
||||||
|
|
||||||
|
### 2. Proactive Agent Design
|
||||||
|
|
||||||
|
Include examples showing proactive usage:
|
||||||
|
```
|
||||||
|
<example>
|
||||||
|
Context: After writing code, agent should review proactively
|
||||||
|
user: "Please write a function..."
|
||||||
|
assistant: "[Writes function]"
|
||||||
|
<commentary>
|
||||||
|
Code written, now use review agent proactively.
|
||||||
|
</commentary>
|
||||||
|
assistant: "Now let me review this code with the code-reviewer agent"
|
||||||
|
</example>
|
||||||
|
```
|
||||||
|
|
||||||
|
### 3. Scope Assumptions
|
||||||
|
|
||||||
|
For code review agents, assume "recently written code" not entire codebase:
|
||||||
|
```
|
||||||
|
For agents that review code, assume recent changes unless explicitly
|
||||||
|
stated otherwise.
|
||||||
|
```
|
||||||
|
|
||||||
|
### 4. Output Structure
|
||||||
|
|
||||||
|
Always define clear output format in system prompt:
|
||||||
|
```
|
||||||
|
**Output Format:**
|
||||||
|
Provide results as:
|
||||||
|
1. Summary (2-3 sentences)
|
||||||
|
2. Detailed findings (bullet points)
|
||||||
|
3. Recommendations (action items)
|
||||||
|
```
|
||||||
|
|
||||||
|
## Integration with Plugin-Dev
|
||||||
|
|
||||||
|
Use this system prompt when creating agents for your plugins:
|
||||||
|
|
||||||
|
1. Take user request for agent functionality
|
||||||
|
2. Feed to Claude with this system prompt
|
||||||
|
3. Get JSON output (identifier, whenToUse, systemPrompt)
|
||||||
|
4. Convert to agent markdown file with frontmatter
|
||||||
|
5. Validate with agent validation rules
|
||||||
|
6. Test triggering conditions
|
||||||
|
7. Add to plugin's `agents/` directory
|
||||||
|
|
||||||
|
This provides AI-assisted agent generation following proven patterns from Claude Code's internal implementation.
|
||||||
411
skills/agent-development/references/system-prompt-design.md
Normal file
411
skills/agent-development/references/system-prompt-design.md
Normal file
@@ -0,0 +1,411 @@
|
|||||||
|
# System Prompt Design Patterns
|
||||||
|
|
||||||
|
Complete guide to writing effective agent system prompts that enable autonomous, high-quality operation.
|
||||||
|
|
||||||
|
## Core Structure
|
||||||
|
|
||||||
|
Every agent system prompt should follow this proven structure:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
You are [specific role] specializing in [specific domain].
|
||||||
|
|
||||||
|
**Your Core Responsibilities:**
|
||||||
|
1. [Primary responsibility - the main task]
|
||||||
|
2. [Secondary responsibility - supporting task]
|
||||||
|
3. [Additional responsibilities as needed]
|
||||||
|
|
||||||
|
**[Task Name] Process:**
|
||||||
|
1. [First concrete step]
|
||||||
|
2. [Second concrete step]
|
||||||
|
3. [Continue with clear steps]
|
||||||
|
[...]
|
||||||
|
|
||||||
|
**Quality Standards:**
|
||||||
|
- [Standard 1 with specifics]
|
||||||
|
- [Standard 2 with specifics]
|
||||||
|
- [Standard 3 with specifics]
|
||||||
|
|
||||||
|
**Output Format:**
|
||||||
|
Provide results structured as:
|
||||||
|
- [Component 1]
|
||||||
|
- [Component 2]
|
||||||
|
- [Include specific formatting requirements]
|
||||||
|
|
||||||
|
**Edge Cases:**
|
||||||
|
Handle these situations:
|
||||||
|
- [Edge case 1]: [Specific handling approach]
|
||||||
|
- [Edge case 2]: [Specific handling approach]
|
||||||
|
```
|
||||||
|
|
||||||
|
## Pattern 1: Analysis Agents
|
||||||
|
|
||||||
|
For agents that analyze code, PRs, or documentation:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
You are an expert [domain] analyzer specializing in [specific analysis type].
|
||||||
|
|
||||||
|
**Your Core Responsibilities:**
|
||||||
|
1. Thoroughly analyze [what] for [specific issues]
|
||||||
|
2. Identify [patterns/problems/opportunities]
|
||||||
|
3. Provide actionable recommendations
|
||||||
|
|
||||||
|
**Analysis Process:**
|
||||||
|
1. **Gather Context**: Read [what] using available tools
|
||||||
|
2. **Initial Scan**: Identify obvious [issues/patterns]
|
||||||
|
3. **Deep Analysis**: Examine [specific aspects]:
|
||||||
|
- [Aspect 1]: Check for [criteria]
|
||||||
|
- [Aspect 2]: Verify [criteria]
|
||||||
|
- [Aspect 3]: Assess [criteria]
|
||||||
|
4. **Synthesize Findings**: Group related issues
|
||||||
|
5. **Prioritize**: Rank by [severity/impact/urgency]
|
||||||
|
6. **Generate Report**: Format according to output template
|
||||||
|
|
||||||
|
**Quality Standards:**
|
||||||
|
- Every finding includes file:line reference
|
||||||
|
- Issues categorized by severity (critical/major/minor)
|
||||||
|
- Recommendations are specific and actionable
|
||||||
|
- Positive observations included for balance
|
||||||
|
|
||||||
|
**Output Format:**
|
||||||
|
## Summary
|
||||||
|
[2-3 sentence overview]
|
||||||
|
|
||||||
|
## Critical Issues
|
||||||
|
- [file:line] - [Issue description] - [Recommendation]
|
||||||
|
|
||||||
|
## Major Issues
|
||||||
|
[...]
|
||||||
|
|
||||||
|
## Minor Issues
|
||||||
|
[...]
|
||||||
|
|
||||||
|
## Recommendations
|
||||||
|
[...]
|
||||||
|
|
||||||
|
**Edge Cases:**
|
||||||
|
- No issues found: Provide positive feedback and validation
|
||||||
|
- Too many issues: Group and prioritize top 10
|
||||||
|
- Unclear code: Request clarification rather than guessing
|
||||||
|
```
|
||||||
|
|
||||||
|
## Pattern 2: Generation Agents
|
||||||
|
|
||||||
|
For agents that create code, tests, or documentation:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
You are an expert [domain] engineer specializing in creating high-quality [output type].
|
||||||
|
|
||||||
|
**Your Core Responsibilities:**
|
||||||
|
1. Generate [what] that meets [quality standards]
|
||||||
|
2. Follow [specific conventions/patterns]
|
||||||
|
3. Ensure [correctness/completeness/clarity]
|
||||||
|
|
||||||
|
**Generation Process:**
|
||||||
|
1. **Understand Requirements**: Analyze what needs to be created
|
||||||
|
2. **Gather Context**: Read existing [code/docs/tests] for patterns
|
||||||
|
3. **Design Structure**: Plan [architecture/organization/flow]
|
||||||
|
4. **Generate Content**: Create [output] following:
|
||||||
|
- [Convention 1]
|
||||||
|
- [Convention 2]
|
||||||
|
- [Best practice 1]
|
||||||
|
5. **Validate**: Verify [correctness/completeness]
|
||||||
|
6. **Document**: Add comments/explanations as needed
|
||||||
|
|
||||||
|
**Quality Standards:**
|
||||||
|
- Follows project conventions (check CLAUDE.md)
|
||||||
|
- [Specific quality metric 1]
|
||||||
|
- [Specific quality metric 2]
|
||||||
|
- Includes error handling
|
||||||
|
- Well-documented and clear
|
||||||
|
|
||||||
|
**Output Format:**
|
||||||
|
Create [what] with:
|
||||||
|
- [Structure requirement 1]
|
||||||
|
- [Structure requirement 2]
|
||||||
|
- Clear, descriptive naming
|
||||||
|
- Comprehensive coverage
|
||||||
|
|
||||||
|
**Edge Cases:**
|
||||||
|
- Insufficient context: Ask user for clarification
|
||||||
|
- Conflicting patterns: Follow most recent/explicit pattern
|
||||||
|
- Complex requirements: Break into smaller pieces
|
||||||
|
```
|
||||||
|
|
||||||
|
## Pattern 3: Validation Agents
|
||||||
|
|
||||||
|
For agents that validate, check, or verify:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
You are an expert [domain] validator specializing in ensuring [quality aspect].
|
||||||
|
|
||||||
|
**Your Core Responsibilities:**
|
||||||
|
1. Validate [what] against [criteria]
|
||||||
|
2. Identify violations and issues
|
||||||
|
3. Provide clear pass/fail determination
|
||||||
|
|
||||||
|
**Validation Process:**
|
||||||
|
1. **Load Criteria**: Understand validation requirements
|
||||||
|
2. **Scan Target**: Read [what] needs validation
|
||||||
|
3. **Check Rules**: For each rule:
|
||||||
|
- [Rule 1]: [Validation method]
|
||||||
|
- [Rule 2]: [Validation method]
|
||||||
|
4. **Collect Violations**: Document each failure with details
|
||||||
|
5. **Assess Severity**: Categorize issues
|
||||||
|
6. **Determine Result**: Pass only if [criteria met]
|
||||||
|
|
||||||
|
**Quality Standards:**
|
||||||
|
- All violations include specific locations
|
||||||
|
- Severity clearly indicated
|
||||||
|
- Fix suggestions provided
|
||||||
|
- No false positives
|
||||||
|
|
||||||
|
**Output Format:**
|
||||||
|
## Validation Result: [PASS/FAIL]
|
||||||
|
|
||||||
|
## Summary
|
||||||
|
[Overall assessment]
|
||||||
|
|
||||||
|
## Violations Found: [count]
|
||||||
|
### Critical ([count])
|
||||||
|
- [Location]: [Issue] - [Fix]
|
||||||
|
|
||||||
|
### Warnings ([count])
|
||||||
|
- [Location]: [Issue] - [Fix]
|
||||||
|
|
||||||
|
## Recommendations
|
||||||
|
[How to fix violations]
|
||||||
|
|
||||||
|
**Edge Cases:**
|
||||||
|
- No violations: Confirm validation passed
|
||||||
|
- Too many violations: Group by type, show top 20
|
||||||
|
- Ambiguous rules: Document uncertainty, request clarification
|
||||||
|
```
|
||||||
|
|
||||||
|
## Pattern 4: Orchestration Agents
|
||||||
|
|
||||||
|
For agents that coordinate multiple tools or steps:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
You are an expert [domain] orchestrator specializing in coordinating [complex workflow].
|
||||||
|
|
||||||
|
**Your Core Responsibilities:**
|
||||||
|
1. Coordinate [multi-step process]
|
||||||
|
2. Manage [resources/tools/dependencies]
|
||||||
|
3. Ensure [successful completion/integration]
|
||||||
|
|
||||||
|
**Orchestration Process:**
|
||||||
|
1. **Plan**: Understand full workflow and dependencies
|
||||||
|
2. **Prepare**: Set up prerequisites
|
||||||
|
3. **Execute Phases**:
|
||||||
|
- Phase 1: [What] using [tools]
|
||||||
|
- Phase 2: [What] using [tools]
|
||||||
|
- Phase 3: [What] using [tools]
|
||||||
|
4. **Monitor**: Track progress and handle failures
|
||||||
|
5. **Verify**: Confirm successful completion
|
||||||
|
6. **Report**: Provide comprehensive summary
|
||||||
|
|
||||||
|
**Quality Standards:**
|
||||||
|
- Each phase completes successfully
|
||||||
|
- Errors handled gracefully
|
||||||
|
- Progress reported to user
|
||||||
|
- Final state verified
|
||||||
|
|
||||||
|
**Output Format:**
|
||||||
|
## Workflow Execution Report
|
||||||
|
|
||||||
|
### Completed Phases
|
||||||
|
- [Phase]: [Result]
|
||||||
|
|
||||||
|
### Results
|
||||||
|
- [Output 1]
|
||||||
|
- [Output 2]
|
||||||
|
|
||||||
|
### Next Steps
|
||||||
|
[If applicable]
|
||||||
|
|
||||||
|
**Edge Cases:**
|
||||||
|
- Phase failure: Attempt retry, then report and stop
|
||||||
|
- Missing dependencies: Request from user
|
||||||
|
- Timeout: Report partial completion
|
||||||
|
```
|
||||||
|
|
||||||
|
## Writing Style Guidelines
|
||||||
|
|
||||||
|
### Tone and Voice
|
||||||
|
|
||||||
|
**Use second person (addressing the agent):**
|
||||||
|
```
|
||||||
|
✅ You are responsible for...
|
||||||
|
✅ You will analyze...
|
||||||
|
✅ Your process should...
|
||||||
|
|
||||||
|
❌ The agent is responsible for...
|
||||||
|
❌ This agent will analyze...
|
||||||
|
❌ I will analyze...
|
||||||
|
```
|
||||||
|
|
||||||
|
### Clarity and Specificity
|
||||||
|
|
||||||
|
**Be specific, not vague:**
|
||||||
|
```
|
||||||
|
✅ Check for SQL injection by examining all database queries for parameterization
|
||||||
|
❌ Look for security issues
|
||||||
|
|
||||||
|
✅ Provide file:line references for each finding
|
||||||
|
❌ Show where issues are
|
||||||
|
|
||||||
|
✅ Categorize as critical (security), major (bugs), or minor (style)
|
||||||
|
❌ Rate the severity of issues
|
||||||
|
```
|
||||||
|
|
||||||
|
### Actionable Instructions
|
||||||
|
|
||||||
|
**Give concrete steps:**
|
||||||
|
```
|
||||||
|
✅ Read the file using the Read tool, then search for patterns using Grep
|
||||||
|
❌ Analyze the code
|
||||||
|
|
||||||
|
✅ Generate test file at test/path/to/file.test.ts
|
||||||
|
❌ Create tests
|
||||||
|
```
|
||||||
|
|
||||||
|
## Common Pitfalls
|
||||||
|
|
||||||
|
### ❌ Vague Responsibilities
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
**Your Core Responsibilities:**
|
||||||
|
1. Help the user with their code
|
||||||
|
2. Provide assistance
|
||||||
|
3. Be helpful
|
||||||
|
```
|
||||||
|
|
||||||
|
**Why bad:** Not specific enough to guide behavior.
|
||||||
|
|
||||||
|
### ✅ Specific Responsibilities
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
**Your Core Responsibilities:**
|
||||||
|
1. Analyze TypeScript code for type safety issues
|
||||||
|
2. Identify missing type annotations and improper 'any' usage
|
||||||
|
3. Recommend specific type improvements with examples
|
||||||
|
```
|
||||||
|
|
||||||
|
### ❌ Missing Process Steps
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
Analyze the code and provide feedback.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Why bad:** Agent doesn't know HOW to analyze.
|
||||||
|
|
||||||
|
### ✅ Clear Process
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
**Analysis Process:**
|
||||||
|
1. Read code files using Read tool
|
||||||
|
2. Scan for type annotations on all functions
|
||||||
|
3. Check for 'any' type usage
|
||||||
|
4. Verify generic type parameters
|
||||||
|
5. List findings with file:line references
|
||||||
|
```
|
||||||
|
|
||||||
|
### ❌ Undefined Output
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
Provide a report.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Why bad:** Agent doesn't know what format to use.
|
||||||
|
|
||||||
|
### ✅ Defined Output Format
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
**Output Format:**
|
||||||
|
## Type Safety Report
|
||||||
|
|
||||||
|
### Summary
|
||||||
|
[Overview of findings]
|
||||||
|
|
||||||
|
### Issues Found
|
||||||
|
- `file.ts:42` - Missing return type on `processData`
|
||||||
|
- `utils.ts:15` - Unsafe 'any' usage in parameter
|
||||||
|
|
||||||
|
### Recommendations
|
||||||
|
[Specific fixes with examples]
|
||||||
|
```
|
||||||
|
|
||||||
|
## Length Guidelines
|
||||||
|
|
||||||
|
### Minimum Viable Agent
|
||||||
|
|
||||||
|
**~500 words minimum:**
|
||||||
|
- Role description
|
||||||
|
- 3 core responsibilities
|
||||||
|
- 5-step process
|
||||||
|
- Output format
|
||||||
|
|
||||||
|
### Standard Agent
|
||||||
|
|
||||||
|
**~1,000-2,000 words:**
|
||||||
|
- Detailed role and expertise
|
||||||
|
- 5-8 responsibilities
|
||||||
|
- 8-12 process steps
|
||||||
|
- Quality standards
|
||||||
|
- Output format
|
||||||
|
- 3-5 edge cases
|
||||||
|
|
||||||
|
### Comprehensive Agent
|
||||||
|
|
||||||
|
**~2,000-5,000 words:**
|
||||||
|
- Complete role with background
|
||||||
|
- Comprehensive responsibilities
|
||||||
|
- Detailed multi-phase process
|
||||||
|
- Extensive quality standards
|
||||||
|
- Multiple output formats
|
||||||
|
- Many edge cases
|
||||||
|
- Examples within system prompt
|
||||||
|
|
||||||
|
**Avoid > 10,000 words:** Too long, diminishing returns.
|
||||||
|
|
||||||
|
## Testing System Prompts
|
||||||
|
|
||||||
|
### Test Completeness
|
||||||
|
|
||||||
|
Can the agent handle these based on system prompt alone?
|
||||||
|
|
||||||
|
- [ ] Typical task execution
|
||||||
|
- [ ] Edge cases mentioned
|
||||||
|
- [ ] Error scenarios
|
||||||
|
- [ ] Unclear requirements
|
||||||
|
- [ ] Large/complex inputs
|
||||||
|
- [ ] Empty/missing inputs
|
||||||
|
|
||||||
|
### Test Clarity
|
||||||
|
|
||||||
|
Read the system prompt and ask:
|
||||||
|
|
||||||
|
- Can another developer understand what this agent does?
|
||||||
|
- Are process steps clear and actionable?
|
||||||
|
- Is output format unambiguous?
|
||||||
|
- Are quality standards measurable?
|
||||||
|
|
||||||
|
### Iterate Based on Results
|
||||||
|
|
||||||
|
After testing agent:
|
||||||
|
1. Identify where it struggled
|
||||||
|
2. Add missing guidance to system prompt
|
||||||
|
3. Clarify ambiguous instructions
|
||||||
|
4. Add process steps for edge cases
|
||||||
|
5. Re-test
|
||||||
|
|
||||||
|
## Conclusion
|
||||||
|
|
||||||
|
Effective system prompts are:
|
||||||
|
- **Specific**: Clear about what and how
|
||||||
|
- **Structured**: Organized with clear sections
|
||||||
|
- **Complete**: Covers normal and edge cases
|
||||||
|
- **Actionable**: Provides concrete steps
|
||||||
|
- **Testable**: Defines measurable standards
|
||||||
|
|
||||||
|
Use the patterns above as templates, customize for your domain, and iterate based on agent performance.
|
||||||
491
skills/agent-development/references/triggering-examples.md
Normal file
491
skills/agent-development/references/triggering-examples.md
Normal file
@@ -0,0 +1,491 @@
|
|||||||
|
# Agent Triggering Examples: Best Practices
|
||||||
|
|
||||||
|
Complete guide to writing effective `<example>` blocks in agent descriptions for reliable triggering.
|
||||||
|
|
||||||
|
## Example Block Format
|
||||||
|
|
||||||
|
The standard format for triggering examples:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
<example>
|
||||||
|
Context: [Describe the situation - what led to this interaction]
|
||||||
|
user: "[Exact user message or request]"
|
||||||
|
assistant: "[How Claude should respond before triggering]"
|
||||||
|
<commentary>
|
||||||
|
[Explanation of why this agent should be triggered in this scenario]
|
||||||
|
</commentary>
|
||||||
|
assistant: "[How Claude triggers the agent - usually 'I'll use the [agent-name] agent...']"
|
||||||
|
</example>
|
||||||
|
```
|
||||||
|
|
||||||
|
## Anatomy of a Good Example
|
||||||
|
|
||||||
|
### Context
|
||||||
|
|
||||||
|
**Purpose:** Set the scene - what happened before the user's message
|
||||||
|
|
||||||
|
**Good contexts:**
|
||||||
|
```
|
||||||
|
Context: User just implemented a new authentication feature
|
||||||
|
Context: User has created a PR and wants it reviewed
|
||||||
|
Context: User is debugging a test failure
|
||||||
|
Context: After writing several functions without documentation
|
||||||
|
```
|
||||||
|
|
||||||
|
**Bad contexts:**
|
||||||
|
```
|
||||||
|
Context: User needs help (too vague)
|
||||||
|
Context: Normal usage (not specific)
|
||||||
|
```
|
||||||
|
|
||||||
|
### User Message
|
||||||
|
|
||||||
|
**Purpose:** Show the exact phrasing that should trigger the agent
|
||||||
|
|
||||||
|
**Good user messages:**
|
||||||
|
```
|
||||||
|
user: "I've added the OAuth flow, can you check it?"
|
||||||
|
user: "Review PR #123"
|
||||||
|
user: "Why is this test failing?"
|
||||||
|
user: "Add docs for these functions"
|
||||||
|
```
|
||||||
|
|
||||||
|
**Vary the phrasing:**
|
||||||
|
Include multiple examples with different phrasings for the same intent:
|
||||||
|
```
|
||||||
|
Example 1: user: "Review my code"
|
||||||
|
Example 2: user: "Can you check this implementation?"
|
||||||
|
Example 3: user: "Look over my changes"
|
||||||
|
```
|
||||||
|
|
||||||
|
### Assistant Response (Before Triggering)
|
||||||
|
|
||||||
|
**Purpose:** Show what Claude says before launching the agent
|
||||||
|
|
||||||
|
**Good responses:**
|
||||||
|
```
|
||||||
|
assistant: "I'll analyze your OAuth implementation."
|
||||||
|
assistant: "Let me review that PR for you."
|
||||||
|
assistant: "I'll investigate the test failure."
|
||||||
|
```
|
||||||
|
|
||||||
|
**Proactive example:**
|
||||||
|
```
|
||||||
|
assistant: "Great! Now let me review the code quality."
|
||||||
|
<commentary>
|
||||||
|
Code was just written, proactively trigger review agent.
|
||||||
|
</commentary>
|
||||||
|
```
|
||||||
|
|
||||||
|
### Commentary
|
||||||
|
|
||||||
|
**Purpose:** Explain the reasoning - WHY this agent should trigger
|
||||||
|
|
||||||
|
**Good commentary:**
|
||||||
|
```
|
||||||
|
<commentary>
|
||||||
|
User explicitly requested code review, trigger the code-reviewer agent.
|
||||||
|
</commentary>
|
||||||
|
|
||||||
|
<commentary>
|
||||||
|
After code implementation, proactively use review agent to check quality.
|
||||||
|
</commentary>
|
||||||
|
|
||||||
|
<commentary>
|
||||||
|
PR analysis request matches pr-analyzer agent's expertise.
|
||||||
|
</commentary>
|
||||||
|
```
|
||||||
|
|
||||||
|
**Include decision logic:**
|
||||||
|
```
|
||||||
|
<commentary>
|
||||||
|
User wrote tests (Test tool used). The test-analyzer agent should check
|
||||||
|
test quality and coverage before continuing.
|
||||||
|
</commentary>
|
||||||
|
```
|
||||||
|
|
||||||
|
### Assistant Response (Triggering)
|
||||||
|
|
||||||
|
**Purpose:** Show how Claude invokes the agent
|
||||||
|
|
||||||
|
**Standard pattern:**
|
||||||
|
```
|
||||||
|
assistant: "I'll use the [agent-name] agent to [what it will do]."
|
||||||
|
```
|
||||||
|
|
||||||
|
**Examples:**
|
||||||
|
```
|
||||||
|
assistant: "I'll use the code-reviewer agent to analyze the changes."
|
||||||
|
assistant: "Let me use the test-generator agent to create comprehensive tests."
|
||||||
|
assistant: "I'll use the security-analyzer agent to check for vulnerabilities."
|
||||||
|
```
|
||||||
|
|
||||||
|
## Example Types
|
||||||
|
|
||||||
|
### Type 1: Explicit Request
|
||||||
|
|
||||||
|
User directly asks for what the agent does:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
<example>
|
||||||
|
Context: User explicitly requests security analysis
|
||||||
|
user: "Check my code for security vulnerabilities"
|
||||||
|
assistant: "I'll use the security-analyzer agent to perform a thorough security review."
|
||||||
|
<commentary>
|
||||||
|
Direct security analysis request triggers the security-analyzer agent.
|
||||||
|
</commentary>
|
||||||
|
</example>
|
||||||
|
```
|
||||||
|
|
||||||
|
### Type 2: Proactive Triggering
|
||||||
|
|
||||||
|
Agent triggers after relevant work without explicit request:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
<example>
|
||||||
|
Context: User just implemented database queries
|
||||||
|
user: "I've added the user search functionality"
|
||||||
|
assistant: "Great! Let me review it for security."
|
||||||
|
<commentary>
|
||||||
|
Database code written. Proactively trigger security-analyzer to check
|
||||||
|
for SQL injection and other database security issues.
|
||||||
|
</commentary>
|
||||||
|
assistant: "I'll use the security-analyzer agent to check the database code."
|
||||||
|
</example>
|
||||||
|
```
|
||||||
|
|
||||||
|
### Type 3: Implicit Request
|
||||||
|
|
||||||
|
User implies need without stating it directly:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
<example>
|
||||||
|
Context: User mentions code is hard to understand
|
||||||
|
user: "This code is confusing and hard to follow"
|
||||||
|
assistant: "I can help improve the code clarity."
|
||||||
|
<commentary>
|
||||||
|
User indicates complexity issues. Trigger code-simplifier agent to refactor
|
||||||
|
for better readability.
|
||||||
|
</commentary>
|
||||||
|
assistant: "I'll use the code-simplifier agent to suggest improvements."
|
||||||
|
</example>
|
||||||
|
```
|
||||||
|
|
||||||
|
### Type 4: Tool Usage Pattern
|
||||||
|
|
||||||
|
Agent triggers based on prior tool usage:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
<example>
|
||||||
|
Context: User made multiple edits to test files
|
||||||
|
user: "I've updated all the tests"
|
||||||
|
assistant: "Great! Let me verify test quality."
|
||||||
|
<commentary>
|
||||||
|
Multiple Edit tools used on test files. Proactively trigger test-quality-analyzer
|
||||||
|
to ensure tests follow best practices.
|
||||||
|
</commentary>
|
||||||
|
assistant: "I'll use the test-quality-analyzer agent to review the tests."
|
||||||
|
</example>
|
||||||
|
```
|
||||||
|
|
||||||
|
## Multiple Examples Strategy
|
||||||
|
|
||||||
|
### Cover Different Phrasings
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
<example>
|
||||||
|
user: "Review my code"
|
||||||
|
[...]
|
||||||
|
</example>
|
||||||
|
|
||||||
|
<example>
|
||||||
|
user: "Can you check my implementation?"
|
||||||
|
[...]
|
||||||
|
</example>
|
||||||
|
|
||||||
|
<example>
|
||||||
|
user: "Look over these changes"
|
||||||
|
[...]
|
||||||
|
</example>
|
||||||
|
```
|
||||||
|
|
||||||
|
### Cover Proactive and Reactive
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
<example>
|
||||||
|
Context: User explicitly requests review
|
||||||
|
user: "Review my code for issues"
|
||||||
|
[...]
|
||||||
|
</example>
|
||||||
|
|
||||||
|
<example>
|
||||||
|
Context: After user writes code
|
||||||
|
user: "I've implemented the feature"
|
||||||
|
assistant: "Great! Now let me review it."
|
||||||
|
<commentary>
|
||||||
|
Code written, proactively review.
|
||||||
|
</commentary>
|
||||||
|
[...]
|
||||||
|
</example>
|
||||||
|
```
|
||||||
|
|
||||||
|
### Cover Edge Cases
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
<example>
|
||||||
|
Context: Typical usage
|
||||||
|
user: "Check my PR"
|
||||||
|
[...]
|
||||||
|
</example>
|
||||||
|
|
||||||
|
<example>
|
||||||
|
Context: Large PR that needs thorough analysis
|
||||||
|
user: "This is a big PR, can you do a deep review?"
|
||||||
|
[...]
|
||||||
|
<commentary>
|
||||||
|
Large PR mentioned, use thorough analysis mode.
|
||||||
|
</commentary>
|
||||||
|
</example>
|
||||||
|
```
|
||||||
|
|
||||||
|
## Common Mistakes
|
||||||
|
|
||||||
|
### ❌ Missing Context
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
<example>
|
||||||
|
user: "Review my code"
|
||||||
|
assistant: "I'll use the code-reviewer agent."
|
||||||
|
</example>
|
||||||
|
```
|
||||||
|
|
||||||
|
**Why bad:** No context about what led to this request.
|
||||||
|
|
||||||
|
### ✅ With Context
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
<example>
|
||||||
|
Context: User just finished implementing authentication
|
||||||
|
user: "Review my code"
|
||||||
|
assistant: "I'll use the code-reviewer agent to analyze the auth implementation."
|
||||||
|
<commentary>
|
||||||
|
Auth code written, review for security and best practices.
|
||||||
|
</commentary>
|
||||||
|
</example>
|
||||||
|
```
|
||||||
|
|
||||||
|
### ❌ No Commentary
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
<example>
|
||||||
|
Context: User requests review
|
||||||
|
user: "Check my changes"
|
||||||
|
assistant: "I'll use the reviewer agent."
|
||||||
|
</example>
|
||||||
|
```
|
||||||
|
|
||||||
|
**Why bad:** Doesn't explain WHY agent triggers.
|
||||||
|
|
||||||
|
### ✅ With Commentary
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
<example>
|
||||||
|
Context: User requests review after making changes
|
||||||
|
user: "Check my changes"
|
||||||
|
assistant: "I'll use the code-reviewer agent to analyze the modifications."
|
||||||
|
<commentary>
|
||||||
|
Code review request triggers agent to check quality, security, and adherence to standards.
|
||||||
|
</commentary>
|
||||||
|
</example>
|
||||||
|
```
|
||||||
|
|
||||||
|
### ❌ Agent Responds Directly
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
<example>
|
||||||
|
user: "Review my code"
|
||||||
|
assistant: "I found the following issues: [lists issues]"
|
||||||
|
</example>
|
||||||
|
```
|
||||||
|
|
||||||
|
**Why bad:** Shows agent's output, not triggering.
|
||||||
|
|
||||||
|
### ✅ Shows Triggering
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
<example>
|
||||||
|
user: "Review my code"
|
||||||
|
assistant: "I'll use the code-reviewer agent to perform the review."
|
||||||
|
<commentary>
|
||||||
|
Review request triggers the agent.
|
||||||
|
</commentary>
|
||||||
|
</example>
|
||||||
|
```
|
||||||
|
|
||||||
|
## How Many Examples?
|
||||||
|
|
||||||
|
### Minimum: 2 Examples
|
||||||
|
|
||||||
|
Cover at least:
|
||||||
|
1. Explicit request
|
||||||
|
2. One variation or proactive trigger
|
||||||
|
|
||||||
|
### Recommended: 3-4 Examples
|
||||||
|
|
||||||
|
Cover:
|
||||||
|
1. Explicit request (direct ask)
|
||||||
|
2. Implicit request (user implies need)
|
||||||
|
3. Proactive trigger (after relevant work)
|
||||||
|
4. Edge case or specific scenario
|
||||||
|
|
||||||
|
### Maximum: 6 Examples
|
||||||
|
|
||||||
|
More than 6 makes description too long. Focus on most important scenarios.
|
||||||
|
|
||||||
|
## Template Library
|
||||||
|
|
||||||
|
### Code Review Agent
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
<example>
|
||||||
|
Context: User just implemented a new feature
|
||||||
|
user: "I've added the user registration feature"
|
||||||
|
assistant: "Great! Let me review the code quality."
|
||||||
|
<commentary>
|
||||||
|
Code written, proactively trigger code-reviewer agent.
|
||||||
|
</commentary>
|
||||||
|
assistant: "I'll use the code-reviewer agent to analyze the implementation."
|
||||||
|
</example>
|
||||||
|
|
||||||
|
<example>
|
||||||
|
Context: User explicitly requests review
|
||||||
|
user: "Can you review my changes for issues?"
|
||||||
|
assistant: "I'll use the code-reviewer agent to perform a thorough review."
|
||||||
|
<commentary>
|
||||||
|
Explicit review request triggers the agent.
|
||||||
|
</commentary>
|
||||||
|
</example>
|
||||||
|
```
|
||||||
|
|
||||||
|
### Test Generation Agent
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
<example>
|
||||||
|
Context: User implemented functions without tests
|
||||||
|
user: "I've added the data processing functions"
|
||||||
|
assistant: "Great! Let me generate tests for these."
|
||||||
|
<commentary>
|
||||||
|
New code without tests, proactively trigger test-generator agent.
|
||||||
|
</commentary>
|
||||||
|
assistant: "I'll use the test-generator agent to create comprehensive tests."
|
||||||
|
</example>
|
||||||
|
|
||||||
|
<example>
|
||||||
|
Context: User explicitly asks for tests
|
||||||
|
user: "Generate tests for my code"
|
||||||
|
assistant: "I'll use the test-generator agent to create a complete test suite."
|
||||||
|
<commentary>
|
||||||
|
Direct test generation request triggers the agent.
|
||||||
|
</commentary>
|
||||||
|
</example>
|
||||||
|
```
|
||||||
|
|
||||||
|
### Documentation Agent
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
<example>
|
||||||
|
Context: User added new API endpoints
|
||||||
|
user: "I've implemented the REST API endpoints"
|
||||||
|
assistant: "I'll document these endpoints."
|
||||||
|
<commentary>
|
||||||
|
New API code, proactively generate documentation.
|
||||||
|
</commentary>
|
||||||
|
assistant: "I'll use the api-docs-writer agent to create comprehensive documentation."
|
||||||
|
</example>
|
||||||
|
|
||||||
|
<example>
|
||||||
|
Context: User requests documentation
|
||||||
|
user: "Write docs for this API"
|
||||||
|
assistant: "I'll use the api-docs-writer agent to generate clear documentation."
|
||||||
|
<commentary>
|
||||||
|
Documentation request triggers the agent.
|
||||||
|
</commentary>
|
||||||
|
</example>
|
||||||
|
```
|
||||||
|
|
||||||
|
### Validation Agent
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
<example>
|
||||||
|
Context: User ready to commit changes
|
||||||
|
user: "I'm ready to commit this code"
|
||||||
|
assistant: "Let me validate it first."
|
||||||
|
<commentary>
|
||||||
|
Before commit, proactively validate with validation-agent.
|
||||||
|
</commentary>
|
||||||
|
assistant: "I'll use the code-validator agent to check for issues."
|
||||||
|
</example>
|
||||||
|
|
||||||
|
<example>
|
||||||
|
Context: User asks for validation
|
||||||
|
user: "Validate my implementation"
|
||||||
|
assistant: "I'll use the code-validator agent to verify correctness."
|
||||||
|
<commentary>
|
||||||
|
Explicit validation request triggers the agent.
|
||||||
|
</commentary>
|
||||||
|
</example>
|
||||||
|
```
|
||||||
|
|
||||||
|
## Debugging Triggering Issues
|
||||||
|
|
||||||
|
### Agent Not Triggering
|
||||||
|
|
||||||
|
**Check:**
|
||||||
|
1. Examples include relevant keywords from user message
|
||||||
|
2. Context matches actual usage scenarios
|
||||||
|
3. Commentary explains triggering logic clearly
|
||||||
|
4. Assistant shows use of Agent tool in examples
|
||||||
|
|
||||||
|
**Fix:**
|
||||||
|
Add more examples covering different phrasings.
|
||||||
|
|
||||||
|
### Agent Triggers Too Often
|
||||||
|
|
||||||
|
**Check:**
|
||||||
|
1. Examples are too broad or generic
|
||||||
|
2. Triggering conditions overlap with other agents
|
||||||
|
3. Commentary doesn't distinguish when NOT to use
|
||||||
|
|
||||||
|
**Fix:**
|
||||||
|
Make examples more specific, add negative examples.
|
||||||
|
|
||||||
|
### Agent Triggers in Wrong Scenarios
|
||||||
|
|
||||||
|
**Check:**
|
||||||
|
1. Examples don't match actual intended use
|
||||||
|
2. Commentary suggests inappropriate triggering
|
||||||
|
|
||||||
|
**Fix:**
|
||||||
|
Revise examples to show only correct triggering scenarios.
|
||||||
|
|
||||||
|
## Best Practices Summary
|
||||||
|
|
||||||
|
✅ **DO:**
|
||||||
|
- Include 2-4 concrete, specific examples
|
||||||
|
- Show both explicit and proactive triggering
|
||||||
|
- Provide clear context for each example
|
||||||
|
- Explain reasoning in commentary
|
||||||
|
- Vary user message phrasing
|
||||||
|
- Show Claude using Agent tool
|
||||||
|
|
||||||
|
❌ **DON'T:**
|
||||||
|
- Use generic, vague examples
|
||||||
|
- Omit context or commentary
|
||||||
|
- Show only one type of triggering
|
||||||
|
- Skip the agent invocation step
|
||||||
|
- Make examples too similar
|
||||||
|
- Forget to explain why agent triggers
|
||||||
|
|
||||||
|
## Conclusion
|
||||||
|
|
||||||
|
Well-crafted examples are crucial for reliable agent triggering. Invest time in creating diverse, specific examples that clearly demonstrate when and why the agent should be used.
|
||||||
217
skills/agent-development/scripts/validate-agent.sh
Executable file
217
skills/agent-development/scripts/validate-agent.sh
Executable file
@@ -0,0 +1,217 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
# Agent File Validator
|
||||||
|
# Validates agent markdown files for correct structure and content
|
||||||
|
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
# Usage
|
||||||
|
if [ $# -eq 0 ]; then
|
||||||
|
echo "Usage: $0 <path/to/agent.md>"
|
||||||
|
echo ""
|
||||||
|
echo "Validates agent file for:"
|
||||||
|
echo " - YAML frontmatter structure"
|
||||||
|
echo " - Required fields (name, description, model, color)"
|
||||||
|
echo " - Field formats and constraints"
|
||||||
|
echo " - System prompt presence and length"
|
||||||
|
echo " - Example blocks in description"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
AGENT_FILE="$1"
|
||||||
|
|
||||||
|
echo "🔍 Validating agent file: $AGENT_FILE"
|
||||||
|
echo ""
|
||||||
|
|
||||||
|
# Check 1: File exists
|
||||||
|
if [ ! -f "$AGENT_FILE" ]; then
|
||||||
|
echo "❌ File not found: $AGENT_FILE"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
echo "✅ File exists"
|
||||||
|
|
||||||
|
# Check 2: Starts with ---
|
||||||
|
FIRST_LINE=$(head -1 "$AGENT_FILE")
|
||||||
|
if [ "$FIRST_LINE" != "---" ]; then
|
||||||
|
echo "❌ File must start with YAML frontmatter (---)"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
echo "✅ Starts with frontmatter"
|
||||||
|
|
||||||
|
# Check 3: Has closing ---
|
||||||
|
if ! tail -n +2 "$AGENT_FILE" | grep -q '^---$'; then
|
||||||
|
echo "❌ Frontmatter not closed (missing second ---)"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
echo "✅ Frontmatter properly closed"
|
||||||
|
|
||||||
|
# Extract frontmatter and system prompt
|
||||||
|
FRONTMATTER=$(sed -n '/^---$/,/^---$/{ /^---$/d; p; }' "$AGENT_FILE")
|
||||||
|
SYSTEM_PROMPT=$(awk '/^---$/{i++; next} i>=2' "$AGENT_FILE")
|
||||||
|
|
||||||
|
# Check 4: Required fields
|
||||||
|
echo ""
|
||||||
|
echo "Checking required fields..."
|
||||||
|
|
||||||
|
error_count=0
|
||||||
|
warning_count=0
|
||||||
|
|
||||||
|
# Check name field
|
||||||
|
NAME=$(echo "$FRONTMATTER" | grep '^name:' | sed 's/name: *//' | sed 's/^"\(.*\)"$/\1/')
|
||||||
|
|
||||||
|
if [ -z "$NAME" ]; then
|
||||||
|
echo "❌ Missing required field: name"
|
||||||
|
((error_count++))
|
||||||
|
else
|
||||||
|
echo "✅ name: $NAME"
|
||||||
|
|
||||||
|
# Validate name format
|
||||||
|
if ! [[ "$NAME" =~ ^[a-zA-Z0-9][a-zA-Z0-9-]*[a-zA-Z0-9]$ ]]; then
|
||||||
|
echo "❌ name must start/end with alphanumeric and contain only letters, numbers, hyphens"
|
||||||
|
((error_count++))
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Validate name length
|
||||||
|
name_length=${#NAME}
|
||||||
|
if [ $name_length -lt 3 ]; then
|
||||||
|
echo "❌ name too short (minimum 3 characters)"
|
||||||
|
((error_count++))
|
||||||
|
elif [ $name_length -gt 50 ]; then
|
||||||
|
echo "❌ name too long (maximum 50 characters)"
|
||||||
|
((error_count++))
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check for generic names
|
||||||
|
if [[ "$NAME" =~ ^(helper|assistant|agent|tool)$ ]]; then
|
||||||
|
echo "⚠️ name is too generic: $NAME"
|
||||||
|
((warning_count++))
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check description field
|
||||||
|
DESCRIPTION=$(echo "$FRONTMATTER" | grep '^description:' | sed 's/description: *//')
|
||||||
|
|
||||||
|
if [ -z "$DESCRIPTION" ]; then
|
||||||
|
echo "❌ Missing required field: description"
|
||||||
|
((error_count++))
|
||||||
|
else
|
||||||
|
desc_length=${#DESCRIPTION}
|
||||||
|
echo "✅ description: ${desc_length} characters"
|
||||||
|
|
||||||
|
if [ $desc_length -lt 10 ]; then
|
||||||
|
echo "⚠️ description too short (minimum 10 characters recommended)"
|
||||||
|
((warning_count++))
|
||||||
|
elif [ $desc_length -gt 5000 ]; then
|
||||||
|
echo "⚠️ description very long (over 5000 characters)"
|
||||||
|
((warning_count++))
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check for example blocks
|
||||||
|
if ! echo "$DESCRIPTION" | grep -q '<example>'; then
|
||||||
|
echo "⚠️ description should include <example> blocks for triggering"
|
||||||
|
((warning_count++))
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check for "Use this agent when" pattern
|
||||||
|
if ! echo "$DESCRIPTION" | grep -qi 'use this agent when'; then
|
||||||
|
echo "⚠️ description should start with 'Use this agent when...'"
|
||||||
|
((warning_count++))
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check model field
|
||||||
|
MODEL=$(echo "$FRONTMATTER" | grep '^model:' | sed 's/model: *//')
|
||||||
|
|
||||||
|
if [ -z "$MODEL" ]; then
|
||||||
|
echo "❌ Missing required field: model"
|
||||||
|
((error_count++))
|
||||||
|
else
|
||||||
|
echo "✅ model: $MODEL"
|
||||||
|
|
||||||
|
case "$MODEL" in
|
||||||
|
inherit|sonnet|opus|haiku)
|
||||||
|
# Valid model
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
echo "⚠️ Unknown model: $MODEL (valid: inherit, sonnet, opus, haiku)"
|
||||||
|
((warning_count++))
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check color field
|
||||||
|
COLOR=$(echo "$FRONTMATTER" | grep '^color:' | sed 's/color: *//')
|
||||||
|
|
||||||
|
if [ -z "$COLOR" ]; then
|
||||||
|
echo "❌ Missing required field: color"
|
||||||
|
((error_count++))
|
||||||
|
else
|
||||||
|
echo "✅ color: $COLOR"
|
||||||
|
|
||||||
|
case "$COLOR" in
|
||||||
|
blue|cyan|green|yellow|magenta|red)
|
||||||
|
# Valid color
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
echo "⚠️ Unknown color: $COLOR (valid: blue, cyan, green, yellow, magenta, red)"
|
||||||
|
((warning_count++))
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check tools field (optional)
|
||||||
|
TOOLS=$(echo "$FRONTMATTER" | grep '^tools:' | sed 's/tools: *//')
|
||||||
|
|
||||||
|
if [ -n "$TOOLS" ]; then
|
||||||
|
echo "✅ tools: $TOOLS"
|
||||||
|
else
|
||||||
|
echo "💡 tools: not specified (agent has access to all tools)"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check 5: System prompt
|
||||||
|
echo ""
|
||||||
|
echo "Checking system prompt..."
|
||||||
|
|
||||||
|
if [ -z "$SYSTEM_PROMPT" ]; then
|
||||||
|
echo "❌ System prompt is empty"
|
||||||
|
((error_count++))
|
||||||
|
else
|
||||||
|
prompt_length=${#SYSTEM_PROMPT}
|
||||||
|
echo "✅ System prompt: $prompt_length characters"
|
||||||
|
|
||||||
|
if [ $prompt_length -lt 20 ]; then
|
||||||
|
echo "❌ System prompt too short (minimum 20 characters)"
|
||||||
|
((error_count++))
|
||||||
|
elif [ $prompt_length -gt 10000 ]; then
|
||||||
|
echo "⚠️ System prompt very long (over 10,000 characters)"
|
||||||
|
((warning_count++))
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check for second person
|
||||||
|
if ! echo "$SYSTEM_PROMPT" | grep -q "You are\|You will\|Your"; then
|
||||||
|
echo "⚠️ System prompt should use second person (You are..., You will...)"
|
||||||
|
((warning_count++))
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check for structure
|
||||||
|
if ! echo "$SYSTEM_PROMPT" | grep -qi "responsibilities\|process\|steps"; then
|
||||||
|
echo "💡 Consider adding clear responsibilities or process steps"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if ! echo "$SYSTEM_PROMPT" | grep -qi "output"; then
|
||||||
|
echo "💡 Consider defining output format expectations"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo ""
|
||||||
|
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
|
||||||
|
|
||||||
|
if [ $error_count -eq 0 ] && [ $warning_count -eq 0 ]; then
|
||||||
|
echo "✅ All checks passed!"
|
||||||
|
exit 0
|
||||||
|
elif [ $error_count -eq 0 ]; then
|
||||||
|
echo "⚠️ Validation passed with $warning_count warning(s)"
|
||||||
|
exit 0
|
||||||
|
else
|
||||||
|
echo "❌ Validation failed with $error_count error(s) and $warning_count warning(s)"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
272
skills/command-development/README.md
Normal file
272
skills/command-development/README.md
Normal file
@@ -0,0 +1,272 @@
|
|||||||
|
# Command Development Skill
|
||||||
|
|
||||||
|
Comprehensive guidance on creating Claude Code slash commands, including file format, frontmatter options, dynamic arguments, and best practices.
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
|
||||||
|
This skill provides knowledge about:
|
||||||
|
- Slash command file format and structure
|
||||||
|
- YAML frontmatter configuration fields
|
||||||
|
- Dynamic arguments ($ARGUMENTS, $1, $2, etc.)
|
||||||
|
- File references with @ syntax
|
||||||
|
- Bash execution with !` syntax
|
||||||
|
- Command organization and namespacing
|
||||||
|
- Best practices for command development
|
||||||
|
- Plugin-specific features (${CLAUDE_PLUGIN_ROOT}, plugin patterns)
|
||||||
|
- Integration with plugin components (agents, skills, hooks)
|
||||||
|
- Validation patterns and error handling
|
||||||
|
|
||||||
|
## Skill Structure
|
||||||
|
|
||||||
|
### SKILL.md (~2,470 words)
|
||||||
|
|
||||||
|
Core skill content covering:
|
||||||
|
|
||||||
|
**Fundamentals:**
|
||||||
|
- Command basics and locations
|
||||||
|
- File format (Markdown with optional frontmatter)
|
||||||
|
- YAML frontmatter fields overview
|
||||||
|
- Dynamic arguments ($ARGUMENTS and positional)
|
||||||
|
- File references (@ syntax)
|
||||||
|
- Bash execution (!` syntax)
|
||||||
|
- Command organization patterns
|
||||||
|
- Best practices and common patterns
|
||||||
|
- Troubleshooting
|
||||||
|
|
||||||
|
**Plugin-Specific:**
|
||||||
|
- ${CLAUDE_PLUGIN_ROOT} environment variable
|
||||||
|
- Plugin command discovery and organization
|
||||||
|
- Plugin command patterns (configuration, template, multi-script)
|
||||||
|
- Integration with plugin components (agents, skills, hooks)
|
||||||
|
- Validation patterns (argument, file, resource, error handling)
|
||||||
|
|
||||||
|
### References
|
||||||
|
|
||||||
|
Detailed documentation:
|
||||||
|
|
||||||
|
- **frontmatter-reference.md**: Complete YAML frontmatter field specifications
|
||||||
|
- All field descriptions with types and defaults
|
||||||
|
- When to use each field
|
||||||
|
- Examples and best practices
|
||||||
|
- Validation and common errors
|
||||||
|
|
||||||
|
- **plugin-features-reference.md**: Plugin-specific command features
|
||||||
|
- Plugin command discovery and organization
|
||||||
|
- ${CLAUDE_PLUGIN_ROOT} environment variable usage
|
||||||
|
- Plugin command patterns (configuration, template, multi-script)
|
||||||
|
- Integration with plugin agents, skills, and hooks
|
||||||
|
- Validation patterns and error handling
|
||||||
|
|
||||||
|
### Examples
|
||||||
|
|
||||||
|
Practical command examples:
|
||||||
|
|
||||||
|
- **simple-commands.md**: 10 complete command examples
|
||||||
|
- Code review commands
|
||||||
|
- Testing commands
|
||||||
|
- Deployment commands
|
||||||
|
- Documentation generators
|
||||||
|
- Git integration commands
|
||||||
|
- Analysis and research commands
|
||||||
|
|
||||||
|
- **plugin-commands.md**: 10 plugin-specific command examples
|
||||||
|
- Simple plugin commands with scripts
|
||||||
|
- Multi-script workflows
|
||||||
|
- Template-based generation
|
||||||
|
- Configuration-driven deployment
|
||||||
|
- Agent and skill integration
|
||||||
|
- Multi-component workflows
|
||||||
|
- Validated input commands
|
||||||
|
- Environment-aware commands
|
||||||
|
|
||||||
|
## When This Skill Triggers
|
||||||
|
|
||||||
|
Claude Code activates this skill when users:
|
||||||
|
- Ask to "create a slash command" or "add a command"
|
||||||
|
- Need to "write a custom command"
|
||||||
|
- Want to "define command arguments"
|
||||||
|
- Ask about "command frontmatter" or YAML configuration
|
||||||
|
- Need to "organize commands" or use namespacing
|
||||||
|
- Want to create commands with file references
|
||||||
|
- Ask about "bash execution in commands"
|
||||||
|
- Need command development best practices
|
||||||
|
|
||||||
|
## Progressive Disclosure
|
||||||
|
|
||||||
|
The skill uses progressive disclosure:
|
||||||
|
|
||||||
|
1. **SKILL.md** (~2,470 words): Core concepts, common patterns, and plugin features overview
|
||||||
|
2. **References** (~13,500 words total): Detailed specifications
|
||||||
|
- frontmatter-reference.md (~1,200 words)
|
||||||
|
- plugin-features-reference.md (~1,800 words)
|
||||||
|
- interactive-commands.md (~2,500 words)
|
||||||
|
- advanced-workflows.md (~1,700 words)
|
||||||
|
- testing-strategies.md (~2,200 words)
|
||||||
|
- documentation-patterns.md (~2,000 words)
|
||||||
|
- marketplace-considerations.md (~2,200 words)
|
||||||
|
3. **Examples** (~6,000 words total): Complete working command examples
|
||||||
|
- simple-commands.md
|
||||||
|
- plugin-commands.md
|
||||||
|
|
||||||
|
Claude loads references and examples as needed based on task.
|
||||||
|
|
||||||
|
## Command Basics Quick Reference
|
||||||
|
|
||||||
|
### File Format
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Brief description
|
||||||
|
argument-hint: [arg1] [arg2]
|
||||||
|
allowed-tools: Read, Bash(git:*)
|
||||||
|
---
|
||||||
|
|
||||||
|
Command prompt content with:
|
||||||
|
- Arguments: $1, $2, or $ARGUMENTS
|
||||||
|
- Files: @path/to/file
|
||||||
|
- Bash: !`command here`
|
||||||
|
```
|
||||||
|
|
||||||
|
### Locations
|
||||||
|
|
||||||
|
- **Project**: `.claude/commands/` (shared with team)
|
||||||
|
- **Personal**: `~/.claude/commands/` (your commands)
|
||||||
|
- **Plugin**: `plugin-name/commands/` (plugin-specific)
|
||||||
|
|
||||||
|
### Key Features
|
||||||
|
|
||||||
|
**Dynamic arguments:**
|
||||||
|
- `$ARGUMENTS` - All arguments as single string
|
||||||
|
- `$1`, `$2`, `$3` - Positional arguments
|
||||||
|
|
||||||
|
**File references:**
|
||||||
|
- `@path/to/file` - Include file contents
|
||||||
|
|
||||||
|
**Bash execution:**
|
||||||
|
- `!`command`` - Execute and include output
|
||||||
|
|
||||||
|
## Frontmatter Fields Quick Reference
|
||||||
|
|
||||||
|
| Field | Purpose | Example |
|
||||||
|
|-------|---------|---------|
|
||||||
|
| `description` | Brief description for /help | `"Review code for issues"` |
|
||||||
|
| `allowed-tools` | Restrict tool access | `Read, Bash(git:*)` |
|
||||||
|
| `model` | Specify model | `sonnet`, `opus`, `haiku` |
|
||||||
|
| `argument-hint` | Document arguments | `[pr-number] [priority]` |
|
||||||
|
| `disable-model-invocation` | Manual-only command | `true` |
|
||||||
|
|
||||||
|
## Common Patterns
|
||||||
|
|
||||||
|
### Simple Review Command
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Review code for issues
|
||||||
|
---
|
||||||
|
|
||||||
|
Review this code for quality and potential bugs.
|
||||||
|
```
|
||||||
|
|
||||||
|
### Command with Arguments
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Deploy to environment
|
||||||
|
argument-hint: [environment] [version]
|
||||||
|
---
|
||||||
|
|
||||||
|
Deploy to $1 environment using version $2
|
||||||
|
```
|
||||||
|
|
||||||
|
### Command with File Reference
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Document file
|
||||||
|
argument-hint: [file-path]
|
||||||
|
---
|
||||||
|
|
||||||
|
Generate documentation for @$1
|
||||||
|
```
|
||||||
|
|
||||||
|
### Command with Bash Execution
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Show Git status
|
||||||
|
allowed-tools: Bash(git:*)
|
||||||
|
---
|
||||||
|
|
||||||
|
Current status: !`git status`
|
||||||
|
Recent commits: !`git log --oneline -5`
|
||||||
|
```
|
||||||
|
|
||||||
|
## Development Workflow
|
||||||
|
|
||||||
|
1. **Design command:**
|
||||||
|
- Define purpose and scope
|
||||||
|
- Determine required arguments
|
||||||
|
- Identify needed tools
|
||||||
|
|
||||||
|
2. **Create file:**
|
||||||
|
- Choose appropriate location
|
||||||
|
- Create `.md` file with command name
|
||||||
|
- Write basic prompt
|
||||||
|
|
||||||
|
3. **Add frontmatter:**
|
||||||
|
- Start minimal (just description)
|
||||||
|
- Add fields as needed (allowed-tools, etc.)
|
||||||
|
- Document arguments with argument-hint
|
||||||
|
|
||||||
|
4. **Test command:**
|
||||||
|
- Invoke with `/command-name`
|
||||||
|
- Verify arguments work
|
||||||
|
- Check bash execution
|
||||||
|
- Test file references
|
||||||
|
|
||||||
|
5. **Refine:**
|
||||||
|
- Improve prompt clarity
|
||||||
|
- Handle edge cases
|
||||||
|
- Add examples in comments
|
||||||
|
- Document requirements
|
||||||
|
|
||||||
|
## Best Practices Summary
|
||||||
|
|
||||||
|
1. **Single responsibility**: One command, one clear purpose
|
||||||
|
2. **Clear descriptions**: Make discoverable in `/help`
|
||||||
|
3. **Document arguments**: Always use argument-hint
|
||||||
|
4. **Minimal tools**: Use most restrictive allowed-tools
|
||||||
|
5. **Test thoroughly**: Verify all features work
|
||||||
|
6. **Add comments**: Explain complex logic
|
||||||
|
7. **Handle errors**: Consider missing arguments/files
|
||||||
|
|
||||||
|
## Status
|
||||||
|
|
||||||
|
**Completed enhancements:**
|
||||||
|
- ✓ Plugin command patterns (${CLAUDE_PLUGIN_ROOT}, discovery, organization)
|
||||||
|
- ✓ Integration patterns (agents, skills, hooks coordination)
|
||||||
|
- ✓ Validation patterns (input, file, resource validation, error handling)
|
||||||
|
|
||||||
|
**Remaining enhancements (in progress):**
|
||||||
|
- Advanced workflows (multi-step command sequences)
|
||||||
|
- Testing strategies (how to test commands effectively)
|
||||||
|
- Documentation patterns (command documentation best practices)
|
||||||
|
- Marketplace considerations (publishing and distribution)
|
||||||
|
|
||||||
|
## Maintenance
|
||||||
|
|
||||||
|
To update this skill:
|
||||||
|
1. Keep SKILL.md focused on core fundamentals
|
||||||
|
2. Move detailed specifications to references/
|
||||||
|
3. Add new examples/ for different use cases
|
||||||
|
4. Update frontmatter when new fields added
|
||||||
|
5. Ensure imperative/infinitive form throughout
|
||||||
|
6. Test examples work with current Claude Code
|
||||||
|
|
||||||
|
## Version History
|
||||||
|
|
||||||
|
**v0.1.0** (2025-01-15):
|
||||||
|
- Initial release with basic command fundamentals
|
||||||
|
- Frontmatter field reference
|
||||||
|
- 10 simple command examples
|
||||||
|
- Ready for plugin-specific pattern additions
|
||||||
833
skills/command-development/SKILL.md
Normal file
833
skills/command-development/SKILL.md
Normal file
@@ -0,0 +1,833 @@
|
|||||||
|
---
|
||||||
|
name: command-development
|
||||||
|
description: This skill should be used when the user asks to "create a slash command", "add a command", "write a custom command", "define command arguments", "use command frontmatter", "organize commands", "create command with file references", "interactive command", "use AskUserQuestion in command", or needs guidance on slash command structure, YAML frontmatter fields, dynamic arguments, bash execution in commands, user interaction patterns, or command development best practices for Claude Code.
|
||||||
|
---
|
||||||
|
|
||||||
|
# Command Development for Claude Code
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
|
||||||
|
Slash commands are frequently-used prompts defined as Markdown files that Claude executes during interactive sessions. Understanding command structure, frontmatter options, and dynamic features enables creating powerful, reusable workflows.
|
||||||
|
|
||||||
|
**Key concepts:**
|
||||||
|
- Markdown file format for commands
|
||||||
|
- YAML frontmatter for configuration
|
||||||
|
- Dynamic arguments and file references
|
||||||
|
- Bash execution for context
|
||||||
|
- Command organization and namespacing
|
||||||
|
|
||||||
|
## Command Basics
|
||||||
|
|
||||||
|
### What is a Slash Command?
|
||||||
|
|
||||||
|
A slash command is a Markdown file containing a prompt that Claude executes when invoked. Commands provide:
|
||||||
|
- **Reusability**: Define once, use repeatedly
|
||||||
|
- **Consistency**: Standardize common workflows
|
||||||
|
- **Sharing**: Distribute across team or projects
|
||||||
|
- **Efficiency**: Quick access to complex prompts
|
||||||
|
|
||||||
|
### Critical: Commands are Instructions FOR Claude
|
||||||
|
|
||||||
|
**Commands are written for agent consumption, not human consumption.**
|
||||||
|
|
||||||
|
When a user invokes `/command-name`, the command content becomes Claude's instructions. Write commands as directives TO Claude about what to do, not as messages TO the user.
|
||||||
|
|
||||||
|
**Correct approach (instructions for Claude):**
|
||||||
|
```markdown
|
||||||
|
Review this code for security vulnerabilities including:
|
||||||
|
- SQL injection
|
||||||
|
- XSS attacks
|
||||||
|
- Authentication issues
|
||||||
|
|
||||||
|
Provide specific line numbers and severity ratings.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Incorrect approach (messages to user):**
|
||||||
|
```markdown
|
||||||
|
This command will review your code for security issues.
|
||||||
|
You'll receive a report with vulnerability details.
|
||||||
|
```
|
||||||
|
|
||||||
|
The first example tells Claude what to do. The second tells the user what will happen but doesn't instruct Claude. Always use the first approach.
|
||||||
|
|
||||||
|
### Command Locations
|
||||||
|
|
||||||
|
**Project commands** (shared with team):
|
||||||
|
- Location: `.claude/commands/`
|
||||||
|
- Scope: Available in specific project
|
||||||
|
- Label: Shown as "(project)" in `/help`
|
||||||
|
- Use for: Team workflows, project-specific tasks
|
||||||
|
|
||||||
|
**Personal commands** (available everywhere):
|
||||||
|
- Location: `~/.claude/commands/`
|
||||||
|
- Scope: Available in all projects
|
||||||
|
- Label: Shown as "(user)" in `/help`
|
||||||
|
- Use for: Personal workflows, cross-project utilities
|
||||||
|
|
||||||
|
**Plugin commands** (bundled with plugins):
|
||||||
|
- Location: `plugin-name/commands/`
|
||||||
|
- Scope: Available when plugin installed
|
||||||
|
- Label: Shown as "(plugin-name)" in `/help`
|
||||||
|
- Use for: Plugin-specific functionality
|
||||||
|
|
||||||
|
## File Format
|
||||||
|
|
||||||
|
### Basic Structure
|
||||||
|
|
||||||
|
Commands are Markdown files with `.md` extension:
|
||||||
|
|
||||||
|
```
|
||||||
|
.claude/commands/
|
||||||
|
├── review.md # /review command
|
||||||
|
├── test.md # /test command
|
||||||
|
└── deploy.md # /deploy command
|
||||||
|
```
|
||||||
|
|
||||||
|
**Simple command:**
|
||||||
|
```markdown
|
||||||
|
Review this code for security vulnerabilities including:
|
||||||
|
- SQL injection
|
||||||
|
- XSS attacks
|
||||||
|
- Authentication bypass
|
||||||
|
- Insecure data handling
|
||||||
|
```
|
||||||
|
|
||||||
|
No frontmatter needed for basic commands.
|
||||||
|
|
||||||
|
### With YAML Frontmatter
|
||||||
|
|
||||||
|
Add configuration using YAML frontmatter:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Review code for security issues
|
||||||
|
allowed-tools: Read, Grep, Bash(git:*)
|
||||||
|
model: sonnet
|
||||||
|
---
|
||||||
|
|
||||||
|
Review this code for security vulnerabilities...
|
||||||
|
```
|
||||||
|
|
||||||
|
## YAML Frontmatter Fields
|
||||||
|
|
||||||
|
### description
|
||||||
|
|
||||||
|
**Purpose:** Brief description shown in `/help`
|
||||||
|
**Type:** String
|
||||||
|
**Default:** First line of command prompt
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
---
|
||||||
|
description: Review pull request for code quality
|
||||||
|
---
|
||||||
|
```
|
||||||
|
|
||||||
|
**Best practice:** Clear, actionable description (under 60 characters)
|
||||||
|
|
||||||
|
### allowed-tools
|
||||||
|
|
||||||
|
**Purpose:** Specify which tools command can use
|
||||||
|
**Type:** String or Array
|
||||||
|
**Default:** Inherits from conversation
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
---
|
||||||
|
allowed-tools: Read, Write, Edit, Bash(git:*)
|
||||||
|
---
|
||||||
|
```
|
||||||
|
|
||||||
|
**Patterns:**
|
||||||
|
- `Read, Write, Edit` - Specific tools
|
||||||
|
- `Bash(git:*)` - Bash with git commands only
|
||||||
|
- `*` - All tools (rarely needed)
|
||||||
|
|
||||||
|
**Use when:** Command requires specific tool access
|
||||||
|
|
||||||
|
### model
|
||||||
|
|
||||||
|
**Purpose:** Specify model for command execution
|
||||||
|
**Type:** String (sonnet, opus, haiku)
|
||||||
|
**Default:** Inherits from conversation
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
---
|
||||||
|
model: haiku
|
||||||
|
---
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use cases:**
|
||||||
|
- `haiku` - Fast, simple commands
|
||||||
|
- `sonnet` - Standard workflows
|
||||||
|
- `opus` - Complex analysis
|
||||||
|
|
||||||
|
### argument-hint
|
||||||
|
|
||||||
|
**Purpose:** Document expected arguments for autocomplete
|
||||||
|
**Type:** String
|
||||||
|
**Default:** None
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
---
|
||||||
|
argument-hint: [pr-number] [priority] [assignee]
|
||||||
|
---
|
||||||
|
```
|
||||||
|
|
||||||
|
**Benefits:**
|
||||||
|
- Helps users understand command arguments
|
||||||
|
- Improves command discovery
|
||||||
|
- Documents command interface
|
||||||
|
|
||||||
|
### disable-model-invocation
|
||||||
|
|
||||||
|
**Purpose:** Prevent SlashCommand tool from programmatically calling command
|
||||||
|
**Type:** Boolean
|
||||||
|
**Default:** false
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
---
|
||||||
|
disable-model-invocation: true
|
||||||
|
---
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use when:** Command should only be manually invoked
|
||||||
|
|
||||||
|
## Dynamic Arguments
|
||||||
|
|
||||||
|
### Using $ARGUMENTS
|
||||||
|
|
||||||
|
Capture all arguments as single string:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Fix issue by number
|
||||||
|
argument-hint: [issue-number]
|
||||||
|
---
|
||||||
|
|
||||||
|
Fix issue #$ARGUMENTS following our coding standards and best practices.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Usage:**
|
||||||
|
```
|
||||||
|
> /fix-issue 123
|
||||||
|
> /fix-issue 456
|
||||||
|
```
|
||||||
|
|
||||||
|
**Expands to:**
|
||||||
|
```
|
||||||
|
Fix issue #123 following our coding standards...
|
||||||
|
Fix issue #456 following our coding standards...
|
||||||
|
```
|
||||||
|
|
||||||
|
### Using Positional Arguments
|
||||||
|
|
||||||
|
Capture individual arguments with `$1`, `$2`, `$3`, etc.:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Review PR with priority and assignee
|
||||||
|
argument-hint: [pr-number] [priority] [assignee]
|
||||||
|
---
|
||||||
|
|
||||||
|
Review pull request #$1 with priority level $2.
|
||||||
|
After review, assign to $3 for follow-up.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Usage:**
|
||||||
|
```
|
||||||
|
> /review-pr 123 high alice
|
||||||
|
```
|
||||||
|
|
||||||
|
**Expands to:**
|
||||||
|
```
|
||||||
|
Review pull request #123 with priority level high.
|
||||||
|
After review, assign to alice for follow-up.
|
||||||
|
```
|
||||||
|
|
||||||
|
### Combining Arguments
|
||||||
|
|
||||||
|
Mix positional and remaining arguments:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
Deploy $1 to $2 environment with options: $3
|
||||||
|
```
|
||||||
|
|
||||||
|
**Usage:**
|
||||||
|
```
|
||||||
|
> /deploy api staging --force --skip-tests
|
||||||
|
```
|
||||||
|
|
||||||
|
**Expands to:**
|
||||||
|
```
|
||||||
|
Deploy api to staging environment with options: --force --skip-tests
|
||||||
|
```
|
||||||
|
|
||||||
|
## File References
|
||||||
|
|
||||||
|
### Using @ Syntax
|
||||||
|
|
||||||
|
Include file contents in command:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Review specific file
|
||||||
|
argument-hint: [file-path]
|
||||||
|
---
|
||||||
|
|
||||||
|
Review @$1 for:
|
||||||
|
- Code quality
|
||||||
|
- Best practices
|
||||||
|
- Potential bugs
|
||||||
|
```
|
||||||
|
|
||||||
|
**Usage:**
|
||||||
|
```
|
||||||
|
> /review-file src/api/users.ts
|
||||||
|
```
|
||||||
|
|
||||||
|
**Effect:** Claude reads `src/api/users.ts` before processing command
|
||||||
|
|
||||||
|
### Multiple File References
|
||||||
|
|
||||||
|
Reference multiple files:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
Compare @src/old-version.js with @src/new-version.js
|
||||||
|
|
||||||
|
Identify:
|
||||||
|
- Breaking changes
|
||||||
|
- New features
|
||||||
|
- Bug fixes
|
||||||
|
```
|
||||||
|
|
||||||
|
### Static File References
|
||||||
|
|
||||||
|
Reference known files without arguments:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
Review @package.json and @tsconfig.json for consistency
|
||||||
|
|
||||||
|
Ensure:
|
||||||
|
- TypeScript version matches
|
||||||
|
- Dependencies are aligned
|
||||||
|
- Build configuration is correct
|
||||||
|
```
|
||||||
|
|
||||||
|
## Bash Execution in Commands
|
||||||
|
|
||||||
|
Commands can execute bash commands inline to dynamically gather context before Claude processes the command. This is useful for including repository state, environment information, or project-specific context.
|
||||||
|
|
||||||
|
**When to use:**
|
||||||
|
- Include dynamic context (git status, environment vars, etc.)
|
||||||
|
- Gather project/repository state
|
||||||
|
- Build context-aware workflows
|
||||||
|
|
||||||
|
**Implementation details:**
|
||||||
|
For complete syntax, examples, and best practices, see `references/plugin-features-reference.md` section on bash execution. The reference includes the exact syntax and multiple working examples to avoid execution issues
|
||||||
|
|
||||||
|
## Command Organization
|
||||||
|
|
||||||
|
### Flat Structure
|
||||||
|
|
||||||
|
Simple organization for small command sets:
|
||||||
|
|
||||||
|
```
|
||||||
|
.claude/commands/
|
||||||
|
├── build.md
|
||||||
|
├── test.md
|
||||||
|
├── deploy.md
|
||||||
|
├── review.md
|
||||||
|
└── docs.md
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use when:** 5-15 commands, no clear categories
|
||||||
|
|
||||||
|
### Namespaced Structure
|
||||||
|
|
||||||
|
Organize commands in subdirectories:
|
||||||
|
|
||||||
|
```
|
||||||
|
.claude/commands/
|
||||||
|
├── ci/
|
||||||
|
│ ├── build.md # /build (project:ci)
|
||||||
|
│ ├── test.md # /test (project:ci)
|
||||||
|
│ └── lint.md # /lint (project:ci)
|
||||||
|
├── git/
|
||||||
|
│ ├── commit.md # /commit (project:git)
|
||||||
|
│ └── pr.md # /pr (project:git)
|
||||||
|
└── docs/
|
||||||
|
├── generate.md # /generate (project:docs)
|
||||||
|
└── publish.md # /publish (project:docs)
|
||||||
|
```
|
||||||
|
|
||||||
|
**Benefits:**
|
||||||
|
- Logical grouping by category
|
||||||
|
- Namespace shown in `/help`
|
||||||
|
- Easier to find related commands
|
||||||
|
|
||||||
|
**Use when:** 15+ commands, clear categories
|
||||||
|
|
||||||
|
## Best Practices
|
||||||
|
|
||||||
|
### Command Design
|
||||||
|
|
||||||
|
1. **Single responsibility:** One command, one task
|
||||||
|
2. **Clear descriptions:** Self-explanatory in `/help`
|
||||||
|
3. **Explicit dependencies:** Use `allowed-tools` when needed
|
||||||
|
4. **Document arguments:** Always provide `argument-hint`
|
||||||
|
5. **Consistent naming:** Use verb-noun pattern (review-pr, fix-issue)
|
||||||
|
|
||||||
|
### Argument Handling
|
||||||
|
|
||||||
|
1. **Validate arguments:** Check for required arguments in prompt
|
||||||
|
2. **Provide defaults:** Suggest defaults when arguments missing
|
||||||
|
3. **Document format:** Explain expected argument format
|
||||||
|
4. **Handle edge cases:** Consider missing or invalid arguments
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
argument-hint: [pr-number]
|
||||||
|
---
|
||||||
|
|
||||||
|
$IF($1,
|
||||||
|
Review PR #$1,
|
||||||
|
Please provide a PR number. Usage: /review-pr [number]
|
||||||
|
)
|
||||||
|
```
|
||||||
|
|
||||||
|
### File References
|
||||||
|
|
||||||
|
1. **Explicit paths:** Use clear file paths
|
||||||
|
2. **Check existence:** Handle missing files gracefully
|
||||||
|
3. **Relative paths:** Use project-relative paths
|
||||||
|
4. **Glob support:** Consider using Glob tool for patterns
|
||||||
|
|
||||||
|
### Bash Commands
|
||||||
|
|
||||||
|
1. **Limit scope:** Use `Bash(git:*)` not `Bash(*)`
|
||||||
|
2. **Safe commands:** Avoid destructive operations
|
||||||
|
3. **Handle errors:** Consider command failures
|
||||||
|
4. **Keep fast:** Long-running commands slow invocation
|
||||||
|
|
||||||
|
### Documentation
|
||||||
|
|
||||||
|
1. **Add comments:** Explain complex logic
|
||||||
|
2. **Provide examples:** Show usage in comments
|
||||||
|
3. **List requirements:** Document dependencies
|
||||||
|
4. **Version commands:** Note breaking changes
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Deploy application to environment
|
||||||
|
argument-hint: [environment] [version]
|
||||||
|
---
|
||||||
|
|
||||||
|
<!--
|
||||||
|
Usage: /deploy [staging|production] [version]
|
||||||
|
Requires: AWS credentials configured
|
||||||
|
Example: /deploy staging v1.2.3
|
||||||
|
-->
|
||||||
|
|
||||||
|
Deploy application to $1 environment using version $2...
|
||||||
|
```
|
||||||
|
|
||||||
|
## Common Patterns
|
||||||
|
|
||||||
|
### Review Pattern
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Review code changes
|
||||||
|
allowed-tools: Read, Bash(git:*)
|
||||||
|
---
|
||||||
|
|
||||||
|
Files changed: !`git diff --name-only`
|
||||||
|
|
||||||
|
Review each file for:
|
||||||
|
1. Code quality and style
|
||||||
|
2. Potential bugs or issues
|
||||||
|
3. Test coverage
|
||||||
|
4. Documentation needs
|
||||||
|
|
||||||
|
Provide specific feedback for each file.
|
||||||
|
```
|
||||||
|
|
||||||
|
### Testing Pattern
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Run tests for specific file
|
||||||
|
argument-hint: [test-file]
|
||||||
|
allowed-tools: Bash(npm:*)
|
||||||
|
---
|
||||||
|
|
||||||
|
Run tests: !`npm test $1`
|
||||||
|
|
||||||
|
Analyze results and suggest fixes for failures.
|
||||||
|
```
|
||||||
|
|
||||||
|
### Documentation Pattern
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Generate documentation for file
|
||||||
|
argument-hint: [source-file]
|
||||||
|
---
|
||||||
|
|
||||||
|
Generate comprehensive documentation for @$1 including:
|
||||||
|
- Function/class descriptions
|
||||||
|
- Parameter documentation
|
||||||
|
- Return value descriptions
|
||||||
|
- Usage examples
|
||||||
|
- Edge cases and errors
|
||||||
|
```
|
||||||
|
|
||||||
|
### Workflow Pattern
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Complete PR workflow
|
||||||
|
argument-hint: [pr-number]
|
||||||
|
allowed-tools: Bash(gh:*), Read
|
||||||
|
---
|
||||||
|
|
||||||
|
PR #$1 Workflow:
|
||||||
|
|
||||||
|
1. Fetch PR: !`gh pr view $1`
|
||||||
|
2. Review changes
|
||||||
|
3. Run checks
|
||||||
|
4. Approve or request changes
|
||||||
|
```
|
||||||
|
|
||||||
|
## Troubleshooting
|
||||||
|
|
||||||
|
**Command not appearing:**
|
||||||
|
- Check file is in correct directory
|
||||||
|
- Verify `.md` extension present
|
||||||
|
- Ensure valid Markdown format
|
||||||
|
- Restart Claude Code
|
||||||
|
|
||||||
|
**Arguments not working:**
|
||||||
|
- Verify `$1`, `$2` syntax correct
|
||||||
|
- Check `argument-hint` matches usage
|
||||||
|
- Ensure no extra spaces
|
||||||
|
|
||||||
|
**Bash execution failing:**
|
||||||
|
- Check `allowed-tools` includes Bash
|
||||||
|
- Verify command syntax in backticks
|
||||||
|
- Test command in terminal first
|
||||||
|
- Check for required permissions
|
||||||
|
|
||||||
|
**File references not working:**
|
||||||
|
- Verify `@` syntax correct
|
||||||
|
- Check file path is valid
|
||||||
|
- Ensure Read tool allowed
|
||||||
|
- Use absolute or project-relative paths
|
||||||
|
|
||||||
|
## Plugin-Specific Features
|
||||||
|
|
||||||
|
### CLAUDE_PLUGIN_ROOT Variable
|
||||||
|
|
||||||
|
Plugin commands have access to `${CLAUDE_PLUGIN_ROOT}`, an environment variable that resolves to the plugin's absolute path.
|
||||||
|
|
||||||
|
**Purpose:**
|
||||||
|
- Reference plugin files portably
|
||||||
|
- Execute plugin scripts
|
||||||
|
- Load plugin configuration
|
||||||
|
- Access plugin templates
|
||||||
|
|
||||||
|
**Basic usage:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Analyze using plugin script
|
||||||
|
allowed-tools: Bash(node:*)
|
||||||
|
---
|
||||||
|
|
||||||
|
Run analysis: !`node ${CLAUDE_PLUGIN_ROOT}/scripts/analyze.js $1`
|
||||||
|
|
||||||
|
Review results and report findings.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Common patterns:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
# Execute plugin script
|
||||||
|
!`bash ${CLAUDE_PLUGIN_ROOT}/scripts/script.sh`
|
||||||
|
|
||||||
|
# Load plugin configuration
|
||||||
|
@${CLAUDE_PLUGIN_ROOT}/config/settings.json
|
||||||
|
|
||||||
|
# Use plugin template
|
||||||
|
@${CLAUDE_PLUGIN_ROOT}/templates/report.md
|
||||||
|
|
||||||
|
# Access plugin resources
|
||||||
|
@${CLAUDE_PLUGIN_ROOT}/docs/reference.md
|
||||||
|
```
|
||||||
|
|
||||||
|
**Why use it:**
|
||||||
|
- Works across all installations
|
||||||
|
- Portable between systems
|
||||||
|
- No hardcoded paths needed
|
||||||
|
- Essential for multi-file plugins
|
||||||
|
|
||||||
|
### Plugin Command Organization
|
||||||
|
|
||||||
|
Plugin commands discovered automatically from `commands/` directory:
|
||||||
|
|
||||||
|
```
|
||||||
|
plugin-name/
|
||||||
|
├── commands/
|
||||||
|
│ ├── foo.md # /foo (plugin:plugin-name)
|
||||||
|
│ ├── bar.md # /bar (plugin:plugin-name)
|
||||||
|
│ └── utils/
|
||||||
|
│ └── helper.md # /helper (plugin:plugin-name:utils)
|
||||||
|
└── plugin.json
|
||||||
|
```
|
||||||
|
|
||||||
|
**Namespace benefits:**
|
||||||
|
- Logical command grouping
|
||||||
|
- Shown in `/help` output
|
||||||
|
- Avoid name conflicts
|
||||||
|
- Organize related commands
|
||||||
|
|
||||||
|
**Naming conventions:**
|
||||||
|
- Use descriptive action names
|
||||||
|
- Avoid generic names (test, run)
|
||||||
|
- Consider plugin-specific prefix
|
||||||
|
- Use hyphens for multi-word names
|
||||||
|
|
||||||
|
### Plugin Command Patterns
|
||||||
|
|
||||||
|
**Configuration-based pattern:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Deploy using plugin configuration
|
||||||
|
argument-hint: [environment]
|
||||||
|
allowed-tools: Read, Bash(*)
|
||||||
|
---
|
||||||
|
|
||||||
|
Load configuration: @${CLAUDE_PLUGIN_ROOT}/config/$1-deploy.json
|
||||||
|
|
||||||
|
Deploy to $1 using configuration settings.
|
||||||
|
Monitor deployment and report status.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Template-based pattern:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Generate docs from template
|
||||||
|
argument-hint: [component]
|
||||||
|
---
|
||||||
|
|
||||||
|
Template: @${CLAUDE_PLUGIN_ROOT}/templates/docs.md
|
||||||
|
|
||||||
|
Generate documentation for $1 following template structure.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Multi-script pattern:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Complete build workflow
|
||||||
|
allowed-tools: Bash(*)
|
||||||
|
---
|
||||||
|
|
||||||
|
Build: !`bash ${CLAUDE_PLUGIN_ROOT}/scripts/build.sh`
|
||||||
|
Test: !`bash ${CLAUDE_PLUGIN_ROOT}/scripts/test.sh`
|
||||||
|
Package: !`bash ${CLAUDE_PLUGIN_ROOT}/scripts/package.sh`
|
||||||
|
|
||||||
|
Review outputs and report workflow status.
|
||||||
|
```
|
||||||
|
|
||||||
|
**See `references/plugin-features-reference.md` for detailed patterns.**
|
||||||
|
|
||||||
|
## Integration with Plugin Components
|
||||||
|
|
||||||
|
Commands can integrate with other plugin components for powerful workflows.
|
||||||
|
|
||||||
|
### Agent Integration
|
||||||
|
|
||||||
|
Launch plugin agents for complex tasks:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Deep code review
|
||||||
|
argument-hint: [file-path]
|
||||||
|
---
|
||||||
|
|
||||||
|
Initiate comprehensive review of @$1 using the code-reviewer agent.
|
||||||
|
|
||||||
|
The agent will analyze:
|
||||||
|
- Code structure
|
||||||
|
- Security issues
|
||||||
|
- Performance
|
||||||
|
- Best practices
|
||||||
|
|
||||||
|
Agent uses plugin resources:
|
||||||
|
- ${CLAUDE_PLUGIN_ROOT}/config/rules.json
|
||||||
|
- ${CLAUDE_PLUGIN_ROOT}/checklists/review.md
|
||||||
|
```
|
||||||
|
|
||||||
|
**Key points:**
|
||||||
|
- Agent must exist in `plugin/agents/` directory
|
||||||
|
- Claude uses Task tool to launch agent
|
||||||
|
- Document agent capabilities
|
||||||
|
- Reference plugin resources agent uses
|
||||||
|
|
||||||
|
### Skill Integration
|
||||||
|
|
||||||
|
Leverage plugin skills for specialized knowledge:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Document API with standards
|
||||||
|
argument-hint: [api-file]
|
||||||
|
---
|
||||||
|
|
||||||
|
Document API in @$1 following plugin standards.
|
||||||
|
|
||||||
|
Use the api-docs-standards skill to ensure:
|
||||||
|
- Complete endpoint documentation
|
||||||
|
- Consistent formatting
|
||||||
|
- Example quality
|
||||||
|
- Error documentation
|
||||||
|
|
||||||
|
Generate production-ready API docs.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Key points:**
|
||||||
|
- Skill must exist in `plugin/skills/` directory
|
||||||
|
- Mention skill name to trigger invocation
|
||||||
|
- Document skill purpose
|
||||||
|
- Explain what skill provides
|
||||||
|
|
||||||
|
### Hook Coordination
|
||||||
|
|
||||||
|
Design commands that work with plugin hooks:
|
||||||
|
- Commands can prepare state for hooks to process
|
||||||
|
- Hooks execute automatically on tool events
|
||||||
|
- Commands should document expected hook behavior
|
||||||
|
- Guide Claude on interpreting hook output
|
||||||
|
|
||||||
|
See `references/plugin-features-reference.md` for examples of commands that coordinate with hooks
|
||||||
|
|
||||||
|
### Multi-Component Workflows
|
||||||
|
|
||||||
|
Combine agents, skills, and scripts:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Comprehensive review workflow
|
||||||
|
argument-hint: [file]
|
||||||
|
allowed-tools: Bash(node:*), Read
|
||||||
|
---
|
||||||
|
|
||||||
|
Target: @$1
|
||||||
|
|
||||||
|
Phase 1 - Static Analysis:
|
||||||
|
!`node ${CLAUDE_PLUGIN_ROOT}/scripts/lint.js $1`
|
||||||
|
|
||||||
|
Phase 2 - Deep Review:
|
||||||
|
Launch code-reviewer agent for detailed analysis.
|
||||||
|
|
||||||
|
Phase 3 - Standards Check:
|
||||||
|
Use coding-standards skill for validation.
|
||||||
|
|
||||||
|
Phase 4 - Report:
|
||||||
|
Template: @${CLAUDE_PLUGIN_ROOT}/templates/review.md
|
||||||
|
|
||||||
|
Compile findings into report following template.
|
||||||
|
```
|
||||||
|
|
||||||
|
**When to use:**
|
||||||
|
- Complex multi-step workflows
|
||||||
|
- Leverage multiple plugin capabilities
|
||||||
|
- Require specialized analysis
|
||||||
|
- Need structured outputs
|
||||||
|
|
||||||
|
## Validation Patterns
|
||||||
|
|
||||||
|
Commands should validate inputs and resources before processing.
|
||||||
|
|
||||||
|
### Argument Validation
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Deploy with validation
|
||||||
|
argument-hint: [environment]
|
||||||
|
---
|
||||||
|
|
||||||
|
Validate environment: !`echo "$1" | grep -E "^(dev|staging|prod)$" || echo "INVALID"`
|
||||||
|
|
||||||
|
If $1 is valid environment:
|
||||||
|
Deploy to $1
|
||||||
|
Otherwise:
|
||||||
|
Explain valid environments: dev, staging, prod
|
||||||
|
Show usage: /deploy [environment]
|
||||||
|
```
|
||||||
|
|
||||||
|
### File Existence Checks
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Process configuration
|
||||||
|
argument-hint: [config-file]
|
||||||
|
---
|
||||||
|
|
||||||
|
Check file exists: !`test -f $1 && echo "EXISTS" || echo "MISSING"`
|
||||||
|
|
||||||
|
If file exists:
|
||||||
|
Process configuration: @$1
|
||||||
|
Otherwise:
|
||||||
|
Explain where to place config file
|
||||||
|
Show expected format
|
||||||
|
Provide example configuration
|
||||||
|
```
|
||||||
|
|
||||||
|
### Plugin Resource Validation
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Run plugin analyzer
|
||||||
|
allowed-tools: Bash(test:*)
|
||||||
|
---
|
||||||
|
|
||||||
|
Validate plugin setup:
|
||||||
|
- Script: !`test -x ${CLAUDE_PLUGIN_ROOT}/bin/analyze && echo "✓" || echo "✗"`
|
||||||
|
- Config: !`test -f ${CLAUDE_PLUGIN_ROOT}/config.json && echo "✓" || echo "✗"`
|
||||||
|
|
||||||
|
If all checks pass, run analysis.
|
||||||
|
Otherwise, report missing components.
|
||||||
|
```
|
||||||
|
|
||||||
|
### Error Handling
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Build with error handling
|
||||||
|
allowed-tools: Bash(*)
|
||||||
|
---
|
||||||
|
|
||||||
|
Execute build: !`bash ${CLAUDE_PLUGIN_ROOT}/scripts/build.sh 2>&1 || echo "BUILD_FAILED"`
|
||||||
|
|
||||||
|
If build succeeded:
|
||||||
|
Report success and output location
|
||||||
|
If build failed:
|
||||||
|
Analyze error output
|
||||||
|
Suggest likely causes
|
||||||
|
Provide troubleshooting steps
|
||||||
|
```
|
||||||
|
|
||||||
|
**Best practices:**
|
||||||
|
- Validate early in command
|
||||||
|
- Provide helpful error messages
|
||||||
|
- Suggest corrective actions
|
||||||
|
- Handle edge cases gracefully
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
For detailed frontmatter field specifications, see `references/frontmatter-reference.md`.
|
||||||
|
For plugin-specific features and patterns, see `references/plugin-features-reference.md`.
|
||||||
|
For command pattern examples, see `examples/` directory.
|
||||||
557
skills/command-development/examples/plugin-commands.md
Normal file
557
skills/command-development/examples/plugin-commands.md
Normal file
@@ -0,0 +1,557 @@
|
|||||||
|
# Plugin Command Examples
|
||||||
|
|
||||||
|
Practical examples of commands designed for Claude Code plugins, demonstrating plugin-specific patterns and features.
|
||||||
|
|
||||||
|
## Table of Contents
|
||||||
|
|
||||||
|
1. [Simple Plugin Command](#1-simple-plugin-command)
|
||||||
|
2. [Script-Based Analysis](#2-script-based-analysis)
|
||||||
|
3. [Template-Based Generation](#3-template-based-generation)
|
||||||
|
4. [Multi-Script Workflow](#4-multi-script-workflow)
|
||||||
|
5. [Configuration-Driven Deployment](#5-configuration-driven-deployment)
|
||||||
|
6. [Agent Integration](#6-agent-integration)
|
||||||
|
7. [Skill Integration](#7-skill-integration)
|
||||||
|
8. [Multi-Component Workflow](#8-multi-component-workflow)
|
||||||
|
9. [Validated Input Command](#9-validated-input-command)
|
||||||
|
10. [Environment-Aware Command](#10-environment-aware-command)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 1. Simple Plugin Command
|
||||||
|
|
||||||
|
**Use case:** Basic command that uses plugin script
|
||||||
|
|
||||||
|
**File:** `commands/analyze.md`
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Analyze code quality using plugin tools
|
||||||
|
argument-hint: [file-path]
|
||||||
|
allowed-tools: Bash(node:*), Read
|
||||||
|
---
|
||||||
|
|
||||||
|
Analyze @$1 using plugin's quality checker:
|
||||||
|
|
||||||
|
!`node ${CLAUDE_PLUGIN_ROOT}/scripts/quality-check.js $1`
|
||||||
|
|
||||||
|
Review the analysis output and provide:
|
||||||
|
1. Summary of findings
|
||||||
|
2. Priority issues to address
|
||||||
|
3. Suggested improvements
|
||||||
|
4. Code quality score interpretation
|
||||||
|
```
|
||||||
|
|
||||||
|
**Key features:**
|
||||||
|
- Uses `${CLAUDE_PLUGIN_ROOT}` for portable path
|
||||||
|
- Combines file reference with script execution
|
||||||
|
- Simple single-purpose command
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 2. Script-Based Analysis
|
||||||
|
|
||||||
|
**Use case:** Run comprehensive analysis using multiple plugin scripts
|
||||||
|
|
||||||
|
**File:** `commands/full-audit.md`
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Complete code audit using plugin suite
|
||||||
|
argument-hint: [directory]
|
||||||
|
allowed-tools: Bash(*)
|
||||||
|
model: sonnet
|
||||||
|
---
|
||||||
|
|
||||||
|
Running complete audit on $1:
|
||||||
|
|
||||||
|
**Security scan:**
|
||||||
|
!`bash ${CLAUDE_PLUGIN_ROOT}/scripts/security-scan.sh $1`
|
||||||
|
|
||||||
|
**Performance analysis:**
|
||||||
|
!`bash ${CLAUDE_PLUGIN_ROOT}/scripts/perf-analyze.sh $1`
|
||||||
|
|
||||||
|
**Best practices check:**
|
||||||
|
!`bash ${CLAUDE_PLUGIN_ROOT}/scripts/best-practices.sh $1`
|
||||||
|
|
||||||
|
Analyze all results and create comprehensive report including:
|
||||||
|
- Critical issues requiring immediate attention
|
||||||
|
- Performance optimization opportunities
|
||||||
|
- Security vulnerabilities and fixes
|
||||||
|
- Overall health score and recommendations
|
||||||
|
```
|
||||||
|
|
||||||
|
**Key features:**
|
||||||
|
- Multiple script executions
|
||||||
|
- Organized output sections
|
||||||
|
- Comprehensive workflow
|
||||||
|
- Clear reporting structure
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 3. Template-Based Generation
|
||||||
|
|
||||||
|
**Use case:** Generate documentation following plugin template
|
||||||
|
|
||||||
|
**File:** `commands/gen-api-docs.md`
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Generate API documentation from template
|
||||||
|
argument-hint: [api-file]
|
||||||
|
---
|
||||||
|
|
||||||
|
Template structure: @${CLAUDE_PLUGIN_ROOT}/templates/api-documentation.md
|
||||||
|
|
||||||
|
API implementation: @$1
|
||||||
|
|
||||||
|
Generate complete API documentation following the template format above.
|
||||||
|
|
||||||
|
Ensure documentation includes:
|
||||||
|
- Endpoint descriptions with HTTP methods
|
||||||
|
- Request/response schemas
|
||||||
|
- Authentication requirements
|
||||||
|
- Error codes and handling
|
||||||
|
- Usage examples with curl commands
|
||||||
|
- Rate limiting information
|
||||||
|
|
||||||
|
Format output as markdown suitable for README or docs site.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Key features:**
|
||||||
|
- Uses plugin template
|
||||||
|
- Combines template with source file
|
||||||
|
- Standardized output format
|
||||||
|
- Clear documentation structure
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 4. Multi-Script Workflow
|
||||||
|
|
||||||
|
**Use case:** Orchestrate build, test, and deploy workflow
|
||||||
|
|
||||||
|
**File:** `commands/release.md`
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Execute complete release workflow
|
||||||
|
argument-hint: [version]
|
||||||
|
allowed-tools: Bash(*), Read
|
||||||
|
---
|
||||||
|
|
||||||
|
Executing release workflow for version $1:
|
||||||
|
|
||||||
|
**Step 1 - Pre-release validation:**
|
||||||
|
!`bash ${CLAUDE_PLUGIN_ROOT}/scripts/pre-release-check.sh $1`
|
||||||
|
|
||||||
|
**Step 2 - Build artifacts:**
|
||||||
|
!`bash ${CLAUDE_PLUGIN_ROOT}/scripts/build-release.sh $1`
|
||||||
|
|
||||||
|
**Step 3 - Run test suite:**
|
||||||
|
!`bash ${CLAUDE_PLUGIN_ROOT}/scripts/run-tests.sh`
|
||||||
|
|
||||||
|
**Step 4 - Package release:**
|
||||||
|
!`bash ${CLAUDE_PLUGIN_ROOT}/scripts/package.sh $1`
|
||||||
|
|
||||||
|
Review all step outputs and report:
|
||||||
|
1. Any failures or warnings
|
||||||
|
2. Build artifacts location
|
||||||
|
3. Test results summary
|
||||||
|
4. Next steps for deployment
|
||||||
|
5. Rollback plan if needed
|
||||||
|
```
|
||||||
|
|
||||||
|
**Key features:**
|
||||||
|
- Multi-step workflow
|
||||||
|
- Sequential script execution
|
||||||
|
- Clear step numbering
|
||||||
|
- Comprehensive reporting
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 5. Configuration-Driven Deployment
|
||||||
|
|
||||||
|
**Use case:** Deploy using environment-specific plugin configuration
|
||||||
|
|
||||||
|
**File:** `commands/deploy.md`
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Deploy application to environment
|
||||||
|
argument-hint: [environment]
|
||||||
|
allowed-tools: Read, Bash(*)
|
||||||
|
---
|
||||||
|
|
||||||
|
Deployment configuration for $1: @${CLAUDE_PLUGIN_ROOT}/config/$1-deploy.json
|
||||||
|
|
||||||
|
Current git state: !`git rev-parse --short HEAD`
|
||||||
|
|
||||||
|
Build info: !`cat package.json | grep -E '(name|version)'`
|
||||||
|
|
||||||
|
Execute deployment to $1 environment using configuration above.
|
||||||
|
|
||||||
|
Deployment checklist:
|
||||||
|
1. Validate configuration settings
|
||||||
|
2. Build application for $1
|
||||||
|
3. Run pre-deployment tests
|
||||||
|
4. Deploy to target environment
|
||||||
|
5. Run smoke tests
|
||||||
|
6. Verify deployment success
|
||||||
|
7. Update deployment log
|
||||||
|
|
||||||
|
Report deployment status and any issues encountered.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Key features:**
|
||||||
|
- Environment-specific configuration
|
||||||
|
- Dynamic config file loading
|
||||||
|
- Pre-deployment validation
|
||||||
|
- Structured checklist
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 6. Agent Integration
|
||||||
|
|
||||||
|
**Use case:** Command that launches plugin agent for complex task
|
||||||
|
|
||||||
|
**File:** `commands/deep-review.md`
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Deep code review using plugin agent
|
||||||
|
argument-hint: [file-or-directory]
|
||||||
|
---
|
||||||
|
|
||||||
|
Initiate comprehensive code review of @$1 using the code-reviewer agent.
|
||||||
|
|
||||||
|
The agent will perform:
|
||||||
|
1. **Static analysis** - Check for code smells and anti-patterns
|
||||||
|
2. **Security audit** - Identify potential vulnerabilities
|
||||||
|
3. **Performance review** - Find optimization opportunities
|
||||||
|
4. **Best practices** - Ensure code follows standards
|
||||||
|
5. **Documentation check** - Verify adequate documentation
|
||||||
|
|
||||||
|
The agent has access to:
|
||||||
|
- Plugin's linting rules: ${CLAUDE_PLUGIN_ROOT}/config/lint-rules.json
|
||||||
|
- Security checklist: ${CLAUDE_PLUGIN_ROOT}/checklists/security.md
|
||||||
|
- Performance guidelines: ${CLAUDE_PLUGIN_ROOT}/docs/performance.md
|
||||||
|
|
||||||
|
Note: This uses the Task tool to launch the plugin's code-reviewer agent for thorough analysis.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Key features:**
|
||||||
|
- Delegates to plugin agent
|
||||||
|
- Documents agent capabilities
|
||||||
|
- References plugin resources
|
||||||
|
- Clear scope definition
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 7. Skill Integration
|
||||||
|
|
||||||
|
**Use case:** Command that leverages plugin skill for specialized knowledge
|
||||||
|
|
||||||
|
**File:** `commands/document-api.md`
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Document API following plugin standards
|
||||||
|
argument-hint: [api-file]
|
||||||
|
---
|
||||||
|
|
||||||
|
API source code: @$1
|
||||||
|
|
||||||
|
Generate API documentation following the plugin's API documentation standards.
|
||||||
|
|
||||||
|
Use the api-documentation-standards skill to ensure:
|
||||||
|
- **OpenAPI compliance** - Follow OpenAPI 3.0 specification
|
||||||
|
- **Consistent formatting** - Use plugin's documentation style
|
||||||
|
- **Complete coverage** - Document all endpoints and schemas
|
||||||
|
- **Example quality** - Provide realistic usage examples
|
||||||
|
- **Error documentation** - Cover all error scenarios
|
||||||
|
|
||||||
|
The skill provides:
|
||||||
|
- Standard documentation templates
|
||||||
|
- API documentation best practices
|
||||||
|
- Common patterns for this codebase
|
||||||
|
- Quality validation criteria
|
||||||
|
|
||||||
|
Generate production-ready API documentation.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Key features:**
|
||||||
|
- Invokes plugin skill by name
|
||||||
|
- Documents skill purpose
|
||||||
|
- Clear expectations
|
||||||
|
- Leverages skill knowledge
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 8. Multi-Component Workflow
|
||||||
|
|
||||||
|
**Use case:** Complex workflow using agents, skills, and scripts
|
||||||
|
|
||||||
|
**File:** `commands/complete-review.md`
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Comprehensive review using all plugin components
|
||||||
|
argument-hint: [file-path]
|
||||||
|
allowed-tools: Bash(node:*), Read
|
||||||
|
---
|
||||||
|
|
||||||
|
Target file: @$1
|
||||||
|
|
||||||
|
Execute comprehensive review workflow:
|
||||||
|
|
||||||
|
**Phase 1: Automated Analysis**
|
||||||
|
Run plugin analyzer: !`node ${CLAUDE_PLUGIN_ROOT}/scripts/analyze.js $1`
|
||||||
|
|
||||||
|
**Phase 2: Deep Review (Agent)**
|
||||||
|
Launch the code-quality-reviewer agent for detailed analysis.
|
||||||
|
Agent will examine:
|
||||||
|
- Code structure and organization
|
||||||
|
- Error handling patterns
|
||||||
|
- Testing coverage
|
||||||
|
- Documentation quality
|
||||||
|
|
||||||
|
**Phase 3: Standards Check (Skill)**
|
||||||
|
Use the coding-standards skill to validate:
|
||||||
|
- Naming conventions
|
||||||
|
- Code formatting
|
||||||
|
- Best practices adherence
|
||||||
|
- Framework-specific patterns
|
||||||
|
|
||||||
|
**Phase 4: Report Generation**
|
||||||
|
Template: @${CLAUDE_PLUGIN_ROOT}/templates/review-report.md
|
||||||
|
|
||||||
|
Compile all findings into comprehensive report following template.
|
||||||
|
|
||||||
|
**Phase 5: Recommendations**
|
||||||
|
Generate prioritized action items:
|
||||||
|
1. Critical issues (must fix)
|
||||||
|
2. Important improvements (should fix)
|
||||||
|
3. Nice-to-have enhancements (could fix)
|
||||||
|
|
||||||
|
Include specific file locations and suggested changes for each item.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Key features:**
|
||||||
|
- Multi-phase workflow
|
||||||
|
- Combines scripts, agents, skills
|
||||||
|
- Template-based reporting
|
||||||
|
- Prioritized outputs
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 9. Validated Input Command
|
||||||
|
|
||||||
|
**Use case:** Command with input validation and error handling
|
||||||
|
|
||||||
|
**File:** `commands/build-env.md`
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Build for specific environment with validation
|
||||||
|
argument-hint: [environment]
|
||||||
|
allowed-tools: Bash(*)
|
||||||
|
---
|
||||||
|
|
||||||
|
Validate environment argument: !`echo "$1" | grep -E "^(dev|staging|prod)$" && echo "VALID" || echo "INVALID"`
|
||||||
|
|
||||||
|
Check build script exists: !`test -x ${CLAUDE_PLUGIN_ROOT}/scripts/build.sh && echo "EXISTS" || echo "MISSING"`
|
||||||
|
|
||||||
|
Verify configuration available: !`test -f ${CLAUDE_PLUGIN_ROOT}/config/$1.json && echo "FOUND" || echo "NOT_FOUND"`
|
||||||
|
|
||||||
|
If all validations pass:
|
||||||
|
|
||||||
|
**Configuration:** @${CLAUDE_PLUGIN_ROOT}/config/$1.json
|
||||||
|
|
||||||
|
**Execute build:** !`bash ${CLAUDE_PLUGIN_ROOT}/scripts/build.sh $1 2>&1`
|
||||||
|
|
||||||
|
**Validation results:** !`bash ${CLAUDE_PLUGIN_ROOT}/scripts/validate-build.sh $1 2>&1`
|
||||||
|
|
||||||
|
Report build status and any issues.
|
||||||
|
|
||||||
|
If validations fail:
|
||||||
|
- Explain which validation failed
|
||||||
|
- Provide expected values/locations
|
||||||
|
- Suggest corrective actions
|
||||||
|
- Document troubleshooting steps
|
||||||
|
```
|
||||||
|
|
||||||
|
**Key features:**
|
||||||
|
- Input validation
|
||||||
|
- Resource existence checks
|
||||||
|
- Error handling
|
||||||
|
- Helpful error messages
|
||||||
|
- Graceful failure handling
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 10. Environment-Aware Command
|
||||||
|
|
||||||
|
**Use case:** Command that adapts behavior based on environment
|
||||||
|
|
||||||
|
**File:** `commands/run-checks.md`
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Run environment-appropriate checks
|
||||||
|
argument-hint: [environment]
|
||||||
|
allowed-tools: Bash(*), Read
|
||||||
|
---
|
||||||
|
|
||||||
|
Environment: $1
|
||||||
|
|
||||||
|
Load environment configuration: @${CLAUDE_PLUGIN_ROOT}/config/$1-checks.json
|
||||||
|
|
||||||
|
Determine check level: !`echo "$1" | grep -E "^prod$" && echo "FULL" || echo "BASIC"`
|
||||||
|
|
||||||
|
**For production environment:**
|
||||||
|
- Full test suite: !`bash ${CLAUDE_PLUGIN_ROOT}/scripts/test-full.sh`
|
||||||
|
- Security scan: !`bash ${CLAUDE_PLUGIN_ROOT}/scripts/security-scan.sh`
|
||||||
|
- Performance audit: !`bash ${CLAUDE_PLUGIN_ROOT}/scripts/perf-check.sh`
|
||||||
|
- Compliance check: !`bash ${CLAUDE_PLUGIN_ROOT}/scripts/compliance.sh`
|
||||||
|
|
||||||
|
**For non-production environments:**
|
||||||
|
- Basic tests: !`bash ${CLAUDE_PLUGIN_ROOT}/scripts/test-basic.sh`
|
||||||
|
- Quick lint: !`bash ${CLAUDE_PLUGIN_ROOT}/scripts/lint.sh`
|
||||||
|
|
||||||
|
Analyze results based on environment requirements:
|
||||||
|
|
||||||
|
**Production:** All checks must pass with zero critical issues
|
||||||
|
**Staging:** No critical issues, warnings acceptable
|
||||||
|
**Development:** Focus on blocking issues only
|
||||||
|
|
||||||
|
Report status and recommend proceed/block decision.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Key features:**
|
||||||
|
- Environment-aware logic
|
||||||
|
- Conditional execution
|
||||||
|
- Different validation levels
|
||||||
|
- Appropriate reporting per environment
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Common Patterns Summary
|
||||||
|
|
||||||
|
### Pattern: Plugin Script Execution
|
||||||
|
```markdown
|
||||||
|
!`node ${CLAUDE_PLUGIN_ROOT}/scripts/script-name.js $1`
|
||||||
|
```
|
||||||
|
Use for: Running plugin-provided Node.js scripts
|
||||||
|
|
||||||
|
### Pattern: Plugin Configuration Loading
|
||||||
|
```markdown
|
||||||
|
@${CLAUDE_PLUGIN_ROOT}/config/config-name.json
|
||||||
|
```
|
||||||
|
Use for: Loading plugin configuration files
|
||||||
|
|
||||||
|
### Pattern: Plugin Template Usage
|
||||||
|
```markdown
|
||||||
|
@${CLAUDE_PLUGIN_ROOT}/templates/template-name.md
|
||||||
|
```
|
||||||
|
Use for: Using plugin templates for generation
|
||||||
|
|
||||||
|
### Pattern: Agent Invocation
|
||||||
|
```markdown
|
||||||
|
Launch the [agent-name] agent for [task description].
|
||||||
|
```
|
||||||
|
Use for: Delegating complex tasks to plugin agents
|
||||||
|
|
||||||
|
### Pattern: Skill Reference
|
||||||
|
```markdown
|
||||||
|
Use the [skill-name] skill to ensure [requirements].
|
||||||
|
```
|
||||||
|
Use for: Leveraging plugin skills for specialized knowledge
|
||||||
|
|
||||||
|
### Pattern: Input Validation
|
||||||
|
```markdown
|
||||||
|
Validate input: !`echo "$1" | grep -E "^pattern$" && echo "OK" || echo "ERROR"`
|
||||||
|
```
|
||||||
|
Use for: Validating command arguments
|
||||||
|
|
||||||
|
### Pattern: Resource Validation
|
||||||
|
```markdown
|
||||||
|
Check exists: !`test -f ${CLAUDE_PLUGIN_ROOT}/path/file && echo "YES" || echo "NO"`
|
||||||
|
```
|
||||||
|
Use for: Verifying required plugin files exist
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Development Tips
|
||||||
|
|
||||||
|
### Testing Plugin Commands
|
||||||
|
|
||||||
|
1. **Test with plugin installed:**
|
||||||
|
```bash
|
||||||
|
cd /path/to/plugin
|
||||||
|
claude /command-name args
|
||||||
|
```
|
||||||
|
|
||||||
|
2. **Verify ${CLAUDE_PLUGIN_ROOT} expansion:**
|
||||||
|
```bash
|
||||||
|
# Add debug output to command
|
||||||
|
!`echo "Plugin root: ${CLAUDE_PLUGIN_ROOT}"`
|
||||||
|
```
|
||||||
|
|
||||||
|
3. **Test across different working directories:**
|
||||||
|
```bash
|
||||||
|
cd /tmp && claude /command-name
|
||||||
|
cd /other/project && claude /command-name
|
||||||
|
```
|
||||||
|
|
||||||
|
4. **Validate resource availability:**
|
||||||
|
```bash
|
||||||
|
# Check all plugin resources exist
|
||||||
|
!`ls -la ${CLAUDE_PLUGIN_ROOT}/scripts/`
|
||||||
|
!`ls -la ${CLAUDE_PLUGIN_ROOT}/config/`
|
||||||
|
```
|
||||||
|
|
||||||
|
### Common Mistakes to Avoid
|
||||||
|
|
||||||
|
1. **Using relative paths instead of ${CLAUDE_PLUGIN_ROOT}:**
|
||||||
|
```markdown
|
||||||
|
# Wrong
|
||||||
|
!`node ./scripts/analyze.js`
|
||||||
|
|
||||||
|
# Correct
|
||||||
|
!`node ${CLAUDE_PLUGIN_ROOT}/scripts/analyze.js`
|
||||||
|
```
|
||||||
|
|
||||||
|
2. **Forgetting to allow required tools:**
|
||||||
|
```markdown
|
||||||
|
# Missing allowed-tools
|
||||||
|
!`bash script.sh` # Will fail without Bash permission
|
||||||
|
|
||||||
|
# Correct
|
||||||
|
---
|
||||||
|
allowed-tools: Bash(*)
|
||||||
|
---
|
||||||
|
!`bash ${CLAUDE_PLUGIN_ROOT}/scripts/script.sh`
|
||||||
|
```
|
||||||
|
|
||||||
|
3. **Not validating inputs:**
|
||||||
|
```markdown
|
||||||
|
# Risky - no validation
|
||||||
|
Deploy to $1 environment
|
||||||
|
|
||||||
|
# Better - with validation
|
||||||
|
Validate: !`echo "$1" | grep -E "^(dev|staging|prod)$" || echo "INVALID"`
|
||||||
|
Deploy to $1 environment (if valid)
|
||||||
|
```
|
||||||
|
|
||||||
|
4. **Hardcoding plugin paths:**
|
||||||
|
```markdown
|
||||||
|
# Wrong - breaks on different installations
|
||||||
|
@/home/user/.claude/plugins/my-plugin/config.json
|
||||||
|
|
||||||
|
# Correct - works everywhere
|
||||||
|
@${CLAUDE_PLUGIN_ROOT}/config.json
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
For detailed plugin-specific features, see `references/plugin-features-reference.md`.
|
||||||
|
For general command development, see main `SKILL.md`.
|
||||||
504
skills/command-development/examples/simple-commands.md
Normal file
504
skills/command-development/examples/simple-commands.md
Normal file
@@ -0,0 +1,504 @@
|
|||||||
|
# Simple Command Examples
|
||||||
|
|
||||||
|
Basic slash command patterns for common use cases.
|
||||||
|
|
||||||
|
**Important:** All examples below are written as instructions FOR Claude (agent consumption), not messages TO users. Commands tell Claude what to do, not tell users what will happen.
|
||||||
|
|
||||||
|
## Example 1: Code Review Command
|
||||||
|
|
||||||
|
**File:** `.claude/commands/review.md`
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Review code for quality and issues
|
||||||
|
allowed-tools: Read, Bash(git:*)
|
||||||
|
---
|
||||||
|
|
||||||
|
Review the code in this repository for:
|
||||||
|
|
||||||
|
1. **Code Quality:**
|
||||||
|
- Readability and maintainability
|
||||||
|
- Consistent style and formatting
|
||||||
|
- Appropriate abstraction levels
|
||||||
|
|
||||||
|
2. **Potential Issues:**
|
||||||
|
- Logic errors or bugs
|
||||||
|
- Edge cases not handled
|
||||||
|
- Performance concerns
|
||||||
|
|
||||||
|
3. **Best Practices:**
|
||||||
|
- Design patterns used correctly
|
||||||
|
- Error handling present
|
||||||
|
- Documentation adequate
|
||||||
|
|
||||||
|
Provide specific feedback with file and line references.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Usage:**
|
||||||
|
```
|
||||||
|
> /review
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Example 2: Security Review Command
|
||||||
|
|
||||||
|
**File:** `.claude/commands/security-review.md`
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Review code for security vulnerabilities
|
||||||
|
allowed-tools: Read, Grep
|
||||||
|
model: sonnet
|
||||||
|
---
|
||||||
|
|
||||||
|
Perform comprehensive security review checking for:
|
||||||
|
|
||||||
|
**Common Vulnerabilities:**
|
||||||
|
- SQL injection risks
|
||||||
|
- Cross-site scripting (XSS)
|
||||||
|
- Authentication/authorization issues
|
||||||
|
- Insecure data handling
|
||||||
|
- Hardcoded secrets or credentials
|
||||||
|
|
||||||
|
**Security Best Practices:**
|
||||||
|
- Input validation present
|
||||||
|
- Output encoding correct
|
||||||
|
- Secure defaults used
|
||||||
|
- Error messages safe
|
||||||
|
- Logging appropriate (no sensitive data)
|
||||||
|
|
||||||
|
For each issue found:
|
||||||
|
- File and line number
|
||||||
|
- Severity (Critical/High/Medium/Low)
|
||||||
|
- Description of vulnerability
|
||||||
|
- Recommended fix
|
||||||
|
|
||||||
|
Prioritize issues by severity.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Usage:**
|
||||||
|
```
|
||||||
|
> /security-review
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Example 3: Test Command with File Argument
|
||||||
|
|
||||||
|
**File:** `.claude/commands/test-file.md`
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Run tests for specific file
|
||||||
|
argument-hint: [test-file]
|
||||||
|
allowed-tools: Bash(npm:*), Bash(jest:*)
|
||||||
|
---
|
||||||
|
|
||||||
|
Run tests for $1:
|
||||||
|
|
||||||
|
Test execution: !`npm test $1`
|
||||||
|
|
||||||
|
Analyze results:
|
||||||
|
- Tests passed/failed
|
||||||
|
- Code coverage
|
||||||
|
- Performance issues
|
||||||
|
- Flaky tests
|
||||||
|
|
||||||
|
If failures found, suggest fixes based on error messages.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Usage:**
|
||||||
|
```
|
||||||
|
> /test-file src/utils/helpers.test.ts
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Example 4: Documentation Generator
|
||||||
|
|
||||||
|
**File:** `.claude/commands/document.md`
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Generate documentation for file
|
||||||
|
argument-hint: [source-file]
|
||||||
|
---
|
||||||
|
|
||||||
|
Generate comprehensive documentation for @$1
|
||||||
|
|
||||||
|
Include:
|
||||||
|
|
||||||
|
**Overview:**
|
||||||
|
- Purpose and responsibility
|
||||||
|
- Main functionality
|
||||||
|
- Dependencies
|
||||||
|
|
||||||
|
**API Documentation:**
|
||||||
|
- Function/method signatures
|
||||||
|
- Parameter descriptions with types
|
||||||
|
- Return values with types
|
||||||
|
- Exceptions/errors thrown
|
||||||
|
|
||||||
|
**Usage Examples:**
|
||||||
|
- Basic usage
|
||||||
|
- Common patterns
|
||||||
|
- Edge cases
|
||||||
|
|
||||||
|
**Implementation Notes:**
|
||||||
|
- Algorithm complexity
|
||||||
|
- Performance considerations
|
||||||
|
- Known limitations
|
||||||
|
|
||||||
|
Format as Markdown suitable for project documentation.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Usage:**
|
||||||
|
```
|
||||||
|
> /document src/api/users.ts
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Example 5: Git Status Summary
|
||||||
|
|
||||||
|
**File:** `.claude/commands/git-status.md`
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Summarize Git repository status
|
||||||
|
allowed-tools: Bash(git:*)
|
||||||
|
---
|
||||||
|
|
||||||
|
Repository Status Summary:
|
||||||
|
|
||||||
|
**Current Branch:** !`git branch --show-current`
|
||||||
|
|
||||||
|
**Status:** !`git status --short`
|
||||||
|
|
||||||
|
**Recent Commits:** !`git log --oneline -5`
|
||||||
|
|
||||||
|
**Remote Status:** !`git fetch && git status -sb`
|
||||||
|
|
||||||
|
Provide:
|
||||||
|
- Summary of changes
|
||||||
|
- Suggested next actions
|
||||||
|
- Any warnings or issues
|
||||||
|
```
|
||||||
|
|
||||||
|
**Usage:**
|
||||||
|
```
|
||||||
|
> /git-status
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Example 6: Deployment Command
|
||||||
|
|
||||||
|
**File:** `.claude/commands/deploy.md`
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Deploy to specified environment
|
||||||
|
argument-hint: [environment] [version]
|
||||||
|
allowed-tools: Bash(kubectl:*), Read
|
||||||
|
---
|
||||||
|
|
||||||
|
Deploy to $1 environment using version $2
|
||||||
|
|
||||||
|
**Pre-deployment Checks:**
|
||||||
|
1. Verify $1 configuration exists
|
||||||
|
2. Check version $2 is valid
|
||||||
|
3. Verify cluster accessibility: !`kubectl cluster-info`
|
||||||
|
|
||||||
|
**Deployment Steps:**
|
||||||
|
1. Update deployment manifest with version $2
|
||||||
|
2. Apply configuration to $1
|
||||||
|
3. Monitor rollout status
|
||||||
|
4. Verify pod health
|
||||||
|
5. Run smoke tests
|
||||||
|
|
||||||
|
**Rollback Plan:**
|
||||||
|
Document current version for rollback if issues occur.
|
||||||
|
|
||||||
|
Proceed with deployment? (yes/no)
|
||||||
|
```
|
||||||
|
|
||||||
|
**Usage:**
|
||||||
|
```
|
||||||
|
> /deploy staging v1.2.3
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Example 7: Comparison Command
|
||||||
|
|
||||||
|
**File:** `.claude/commands/compare-files.md`
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Compare two files
|
||||||
|
argument-hint: [file1] [file2]
|
||||||
|
---
|
||||||
|
|
||||||
|
Compare @$1 with @$2
|
||||||
|
|
||||||
|
**Analysis:**
|
||||||
|
|
||||||
|
1. **Differences:**
|
||||||
|
- Lines added
|
||||||
|
- Lines removed
|
||||||
|
- Lines modified
|
||||||
|
|
||||||
|
2. **Functional Changes:**
|
||||||
|
- Breaking changes
|
||||||
|
- New features
|
||||||
|
- Bug fixes
|
||||||
|
- Refactoring
|
||||||
|
|
||||||
|
3. **Impact:**
|
||||||
|
- Affected components
|
||||||
|
- Required updates elsewhere
|
||||||
|
- Migration requirements
|
||||||
|
|
||||||
|
4. **Recommendations:**
|
||||||
|
- Code review focus areas
|
||||||
|
- Testing requirements
|
||||||
|
- Documentation updates needed
|
||||||
|
|
||||||
|
Present as structured comparison report.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Usage:**
|
||||||
|
```
|
||||||
|
> /compare-files src/old-api.ts src/new-api.ts
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Example 8: Quick Fix Command
|
||||||
|
|
||||||
|
**File:** `.claude/commands/quick-fix.md`
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Quick fix for common issues
|
||||||
|
argument-hint: [issue-description]
|
||||||
|
model: haiku
|
||||||
|
---
|
||||||
|
|
||||||
|
Quickly fix: $ARGUMENTS
|
||||||
|
|
||||||
|
**Approach:**
|
||||||
|
1. Identify the issue
|
||||||
|
2. Find relevant code
|
||||||
|
3. Propose fix
|
||||||
|
4. Explain solution
|
||||||
|
|
||||||
|
Focus on:
|
||||||
|
- Simple, direct solution
|
||||||
|
- Minimal changes
|
||||||
|
- Following existing patterns
|
||||||
|
- No breaking changes
|
||||||
|
|
||||||
|
Provide code changes with file paths and line numbers.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Usage:**
|
||||||
|
```
|
||||||
|
> /quick-fix button not responding to clicks
|
||||||
|
> /quick-fix typo in error message
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Example 9: Research Command
|
||||||
|
|
||||||
|
**File:** `.claude/commands/research.md`
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Research best practices for topic
|
||||||
|
argument-hint: [topic]
|
||||||
|
model: sonnet
|
||||||
|
---
|
||||||
|
|
||||||
|
Research best practices for: $ARGUMENTS
|
||||||
|
|
||||||
|
**Coverage:**
|
||||||
|
|
||||||
|
1. **Current State:**
|
||||||
|
- How we currently handle this
|
||||||
|
- Existing implementations
|
||||||
|
|
||||||
|
2. **Industry Standards:**
|
||||||
|
- Common patterns
|
||||||
|
- Recommended approaches
|
||||||
|
- Tools and libraries
|
||||||
|
|
||||||
|
3. **Comparison:**
|
||||||
|
- Our approach vs standards
|
||||||
|
- Gaps or improvements needed
|
||||||
|
- Migration considerations
|
||||||
|
|
||||||
|
4. **Recommendations:**
|
||||||
|
- Concrete action items
|
||||||
|
- Priority and effort estimates
|
||||||
|
- Resources for implementation
|
||||||
|
|
||||||
|
Provide actionable guidance based on research.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Usage:**
|
||||||
|
```
|
||||||
|
> /research error handling in async operations
|
||||||
|
> /research API authentication patterns
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Example 10: Explain Code Command
|
||||||
|
|
||||||
|
**File:** `.claude/commands/explain.md`
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Explain how code works
|
||||||
|
argument-hint: [file-or-function]
|
||||||
|
---
|
||||||
|
|
||||||
|
Explain @$1 in detail
|
||||||
|
|
||||||
|
**Explanation Structure:**
|
||||||
|
|
||||||
|
1. **Overview:**
|
||||||
|
- What it does
|
||||||
|
- Why it exists
|
||||||
|
- How it fits in system
|
||||||
|
|
||||||
|
2. **Step-by-Step:**
|
||||||
|
- Line-by-line walkthrough
|
||||||
|
- Key algorithms or logic
|
||||||
|
- Important details
|
||||||
|
|
||||||
|
3. **Inputs and Outputs:**
|
||||||
|
- Parameters and types
|
||||||
|
- Return values
|
||||||
|
- Side effects
|
||||||
|
|
||||||
|
4. **Edge Cases:**
|
||||||
|
- Error handling
|
||||||
|
- Special cases
|
||||||
|
- Limitations
|
||||||
|
|
||||||
|
5. **Usage Examples:**
|
||||||
|
- How to call it
|
||||||
|
- Common patterns
|
||||||
|
- Integration points
|
||||||
|
|
||||||
|
Explain at level appropriate for junior engineer.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Usage:**
|
||||||
|
```
|
||||||
|
> /explain src/utils/cache.ts
|
||||||
|
> /explain AuthService.login
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Key Patterns
|
||||||
|
|
||||||
|
### Pattern 1: Read-Only Analysis
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
allowed-tools: Read, Grep
|
||||||
|
---
|
||||||
|
|
||||||
|
Analyze but don't modify...
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use for:** Code review, documentation, analysis
|
||||||
|
|
||||||
|
### Pattern 2: Git Operations
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
allowed-tools: Bash(git:*)
|
||||||
|
---
|
||||||
|
|
||||||
|
!`git status`
|
||||||
|
Analyze and suggest...
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use for:** Repository status, commit analysis
|
||||||
|
|
||||||
|
### Pattern 3: Single Argument
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
argument-hint: [target]
|
||||||
|
---
|
||||||
|
|
||||||
|
Process $1...
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use for:** File operations, targeted actions
|
||||||
|
|
||||||
|
### Pattern 4: Multiple Arguments
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
argument-hint: [source] [target] [options]
|
||||||
|
---
|
||||||
|
|
||||||
|
Process $1 to $2 with $3...
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use for:** Workflows, deployments, comparisons
|
||||||
|
|
||||||
|
### Pattern 5: Fast Execution
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
model: haiku
|
||||||
|
---
|
||||||
|
|
||||||
|
Quick simple task...
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use for:** Simple, repetitive commands
|
||||||
|
|
||||||
|
### Pattern 6: File Comparison
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
Compare @$1 with @$2...
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use for:** Diff analysis, migration planning
|
||||||
|
|
||||||
|
### Pattern 7: Context Gathering
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
allowed-tools: Bash(git:*), Read
|
||||||
|
---
|
||||||
|
|
||||||
|
Context: !`git status`
|
||||||
|
Files: @file1 @file2
|
||||||
|
|
||||||
|
Analyze...
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use for:** Informed decision making
|
||||||
|
|
||||||
|
## Tips for Writing Simple Commands
|
||||||
|
|
||||||
|
1. **Start basic:** Single responsibility, clear purpose
|
||||||
|
2. **Add complexity gradually:** Start without frontmatter
|
||||||
|
3. **Test incrementally:** Verify each feature works
|
||||||
|
4. **Use descriptive names:** Command name should indicate purpose
|
||||||
|
5. **Document arguments:** Always use argument-hint
|
||||||
|
6. **Provide examples:** Show usage in comments
|
||||||
|
7. **Handle errors:** Consider missing arguments or files
|
||||||
722
skills/command-development/references/advanced-workflows.md
Normal file
722
skills/command-development/references/advanced-workflows.md
Normal file
@@ -0,0 +1,722 @@
|
|||||||
|
# Advanced Workflow Patterns
|
||||||
|
|
||||||
|
Multi-step command sequences and composition patterns for complex workflows.
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
|
||||||
|
Advanced workflows combine multiple commands, coordinate state across invocations, and create sophisticated automation sequences. These patterns enable building complex functionality from simple command building blocks.
|
||||||
|
|
||||||
|
## Multi-Step Command Patterns
|
||||||
|
|
||||||
|
### Sequential Workflow Command
|
||||||
|
|
||||||
|
Commands that guide users through multi-step processes:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Complete PR review workflow
|
||||||
|
argument-hint: [pr-number]
|
||||||
|
allowed-tools: Bash(gh:*), Read, Grep
|
||||||
|
---
|
||||||
|
|
||||||
|
# PR Review Workflow for #$1
|
||||||
|
|
||||||
|
## Step 1: Fetch PR Details
|
||||||
|
!`gh pr view $1 --json title,body,author,files`
|
||||||
|
|
||||||
|
## Step 2: Review Files
|
||||||
|
Files changed: !`gh pr diff $1 --name-only`
|
||||||
|
|
||||||
|
For each file:
|
||||||
|
- Check code quality
|
||||||
|
- Verify tests exist
|
||||||
|
- Review documentation
|
||||||
|
|
||||||
|
## Step 3: Run Checks
|
||||||
|
Test status: !`gh pr checks $1`
|
||||||
|
|
||||||
|
Verify:
|
||||||
|
- All tests passing
|
||||||
|
- No merge conflicts
|
||||||
|
- CI/CD successful
|
||||||
|
|
||||||
|
## Step 4: Provide Feedback
|
||||||
|
|
||||||
|
Summarize:
|
||||||
|
- Issues found (critical/minor)
|
||||||
|
- Suggestions for improvement
|
||||||
|
- Approval recommendation
|
||||||
|
|
||||||
|
Would you like to:
|
||||||
|
1. Approve PR
|
||||||
|
2. Request changes
|
||||||
|
3. Leave comments only
|
||||||
|
|
||||||
|
Reply with your choice and I'll help complete the action.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Key features:**
|
||||||
|
- Numbered steps for clarity
|
||||||
|
- Bash execution for context
|
||||||
|
- Decision points for user input
|
||||||
|
- Next action suggestions
|
||||||
|
|
||||||
|
### State-Carrying Workflow
|
||||||
|
|
||||||
|
Commands that maintain state between invocations:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Initialize deployment workflow
|
||||||
|
allowed-tools: Write, Bash(git:*)
|
||||||
|
---
|
||||||
|
|
||||||
|
# Initialize Deployment
|
||||||
|
|
||||||
|
Creating deployment tracking file...
|
||||||
|
|
||||||
|
Current branch: !`git branch --show-current`
|
||||||
|
Latest commit: !`git log -1 --format=%H`
|
||||||
|
|
||||||
|
Deployment state saved to `.claude/deployment-state.local.md`:
|
||||||
|
|
||||||
|
\`\`\`markdown
|
||||||
|
---
|
||||||
|
initialized: true
|
||||||
|
branch: $(git branch --show-current)
|
||||||
|
commit: $(git log -1 --format=%H)
|
||||||
|
timestamp: $(date -u +%Y-%m-%dT%H:%M:%SZ)
|
||||||
|
status: initialized
|
||||||
|
---
|
||||||
|
|
||||||
|
# Deployment Tracking
|
||||||
|
|
||||||
|
Branch: $(git branch --show-current)
|
||||||
|
Started: $(date)
|
||||||
|
|
||||||
|
Next steps:
|
||||||
|
1. Run tests: /deploy-test
|
||||||
|
2. Build: /deploy-build
|
||||||
|
3. Deploy: /deploy-execute
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
State saved. Run `/deploy-test` to continue.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Next command** (`/deploy-test`):
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Run deployment tests
|
||||||
|
allowed-tools: Read, Bash(npm:*)
|
||||||
|
---
|
||||||
|
|
||||||
|
Reading deployment state from `.claude/deployment-state.local.md`...
|
||||||
|
|
||||||
|
Running tests: !`npm test`
|
||||||
|
|
||||||
|
Updating state to 'tested'...
|
||||||
|
|
||||||
|
Tests complete. Run `/deploy-build` to continue.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Pattern benefits:**
|
||||||
|
- Persistent state across commands
|
||||||
|
- Clear workflow progression
|
||||||
|
- Safety checkpoints
|
||||||
|
- Resume capability
|
||||||
|
|
||||||
|
### Conditional Workflow Branching
|
||||||
|
|
||||||
|
Commands that adapt based on conditions:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Smart deployment workflow
|
||||||
|
argument-hint: [environment]
|
||||||
|
allowed-tools: Bash(git:*), Bash(npm:*), Read
|
||||||
|
---
|
||||||
|
|
||||||
|
# Deploy to $1
|
||||||
|
|
||||||
|
## Pre-flight Checks
|
||||||
|
|
||||||
|
Branch: !`git branch --show-current`
|
||||||
|
Status: !`git status --short`
|
||||||
|
|
||||||
|
**Checking conditions:**
|
||||||
|
|
||||||
|
1. Branch status:
|
||||||
|
- If main/master: Require approval
|
||||||
|
- If feature branch: Warning about target
|
||||||
|
- If hotfix: Fast-track process
|
||||||
|
|
||||||
|
2. Tests:
|
||||||
|
!`npm test`
|
||||||
|
- If tests fail: STOP - fix tests first
|
||||||
|
- If tests pass: Continue
|
||||||
|
|
||||||
|
3. Environment:
|
||||||
|
- If $1 = 'production': Extra validation
|
||||||
|
- If $1 = 'staging': Standard process
|
||||||
|
- If $1 = 'dev': Minimal checks
|
||||||
|
|
||||||
|
**Workflow decision:**
|
||||||
|
Based on above, proceeding with: [determined workflow]
|
||||||
|
|
||||||
|
[Conditional steps based on environment and status]
|
||||||
|
|
||||||
|
Ready to deploy? (yes/no)
|
||||||
|
```
|
||||||
|
|
||||||
|
## Command Composition Patterns
|
||||||
|
|
||||||
|
### Command Chaining
|
||||||
|
|
||||||
|
Commands designed to work together:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Prepare for code review
|
||||||
|
---
|
||||||
|
|
||||||
|
# Prepare Code Review
|
||||||
|
|
||||||
|
Running preparation sequence:
|
||||||
|
|
||||||
|
1. Format code: /format-code
|
||||||
|
2. Run linter: /lint-code
|
||||||
|
3. Run tests: /test-all
|
||||||
|
4. Generate coverage: /coverage-report
|
||||||
|
5. Create review summary: /review-summary
|
||||||
|
|
||||||
|
This is a meta-command. After completing each step above,
|
||||||
|
I'll compile results and prepare comprehensive review materials.
|
||||||
|
|
||||||
|
Starting sequence...
|
||||||
|
```
|
||||||
|
|
||||||
|
**Individual commands** are simple:
|
||||||
|
- `/format-code` - Just formats
|
||||||
|
- `/lint-code` - Just lints
|
||||||
|
- `/test-all` - Just tests
|
||||||
|
|
||||||
|
**Composition command** orchestrates them.
|
||||||
|
|
||||||
|
### Pipeline Pattern
|
||||||
|
|
||||||
|
Commands that process output from previous commands:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Analyze test failures
|
||||||
|
---
|
||||||
|
|
||||||
|
# Analyze Test Failures
|
||||||
|
|
||||||
|
## Step 1: Get test results
|
||||||
|
(Run /test-all first if not done)
|
||||||
|
|
||||||
|
Reading test output...
|
||||||
|
|
||||||
|
## Step 2: Categorize failures
|
||||||
|
- Flaky tests (random failures)
|
||||||
|
- Consistent failures
|
||||||
|
- New failures vs existing
|
||||||
|
|
||||||
|
## Step 3: Prioritize
|
||||||
|
Rank by:
|
||||||
|
- Impact (critical path vs edge case)
|
||||||
|
- Frequency (always fails vs sometimes)
|
||||||
|
- Effort (quick fix vs major work)
|
||||||
|
|
||||||
|
## Step 4: Generate fix plan
|
||||||
|
For each failure:
|
||||||
|
- Root cause hypothesis
|
||||||
|
- Suggested fix approach
|
||||||
|
- Estimated effort
|
||||||
|
|
||||||
|
Would you like me to:
|
||||||
|
1. Fix highest priority failure
|
||||||
|
2. Generate detailed fix plans for all
|
||||||
|
3. Create GitHub issues for each
|
||||||
|
```
|
||||||
|
|
||||||
|
### Parallel Execution Pattern
|
||||||
|
|
||||||
|
Commands that coordinate multiple simultaneous operations:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Run comprehensive validation
|
||||||
|
allowed-tools: Bash(*), Read
|
||||||
|
---
|
||||||
|
|
||||||
|
# Comprehensive Validation
|
||||||
|
|
||||||
|
Running validations in parallel...
|
||||||
|
|
||||||
|
Starting:
|
||||||
|
- Code quality checks
|
||||||
|
- Security scanning
|
||||||
|
- Dependency audit
|
||||||
|
- Performance profiling
|
||||||
|
|
||||||
|
This will take 2-3 minutes. I'll monitor all processes
|
||||||
|
and report when complete.
|
||||||
|
|
||||||
|
[Poll each process and report progress]
|
||||||
|
|
||||||
|
All validations complete. Summary:
|
||||||
|
- Quality: PASS (0 issues)
|
||||||
|
- Security: WARN (2 minor issues)
|
||||||
|
- Dependencies: PASS
|
||||||
|
- Performance: PASS (baseline met)
|
||||||
|
|
||||||
|
Details:
|
||||||
|
[Collated results from all checks]
|
||||||
|
```
|
||||||
|
|
||||||
|
## Workflow State Management
|
||||||
|
|
||||||
|
### Using .local.md Files
|
||||||
|
|
||||||
|
Store workflow state in plugin-specific files:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
.claude/plugin-name-workflow.local.md:
|
||||||
|
|
||||||
|
---
|
||||||
|
workflow: deployment
|
||||||
|
stage: testing
|
||||||
|
started: 2025-01-15T10:30:00Z
|
||||||
|
environment: staging
|
||||||
|
branch: feature/new-api
|
||||||
|
commit: abc123def
|
||||||
|
tests_passed: false
|
||||||
|
build_complete: false
|
||||||
|
---
|
||||||
|
|
||||||
|
# Deployment Workflow State
|
||||||
|
|
||||||
|
Current stage: Testing
|
||||||
|
Started: 2025-01-15 10:30 UTC
|
||||||
|
|
||||||
|
Completed steps:
|
||||||
|
- ✅ Validation
|
||||||
|
- ✅ Branch check
|
||||||
|
- ⏳ Testing (in progress)
|
||||||
|
|
||||||
|
Pending steps:
|
||||||
|
- Build
|
||||||
|
- Deploy
|
||||||
|
- Smoke tests
|
||||||
|
```
|
||||||
|
|
||||||
|
**Reading state in commands:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Continue deployment workflow
|
||||||
|
allowed-tools: Read, Write
|
||||||
|
---
|
||||||
|
|
||||||
|
Reading workflow state from .claude/plugin-name-workflow.local.md...
|
||||||
|
|
||||||
|
Current stage: @.claude/plugin-name-workflow.local.md
|
||||||
|
|
||||||
|
[Parse YAML frontmatter to determine next step]
|
||||||
|
|
||||||
|
Next action based on state: [determined action]
|
||||||
|
```
|
||||||
|
|
||||||
|
### Workflow Recovery
|
||||||
|
|
||||||
|
Handle interrupted workflows:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Resume deployment workflow
|
||||||
|
allowed-tools: Read
|
||||||
|
---
|
||||||
|
|
||||||
|
# Resume Deployment
|
||||||
|
|
||||||
|
Checking for interrupted workflow...
|
||||||
|
|
||||||
|
State file: @.claude/plugin-name-workflow.local.md
|
||||||
|
|
||||||
|
**Workflow found:**
|
||||||
|
- Started: [timestamp]
|
||||||
|
- Environment: [env]
|
||||||
|
- Last completed: [step]
|
||||||
|
|
||||||
|
**Recovery options:**
|
||||||
|
1. Resume from last step
|
||||||
|
2. Restart from beginning
|
||||||
|
3. Abort and clean up
|
||||||
|
|
||||||
|
Which would you like? (1/2/3)
|
||||||
|
```
|
||||||
|
|
||||||
|
## Workflow Coordination Patterns
|
||||||
|
|
||||||
|
### Cross-Command Communication
|
||||||
|
|
||||||
|
Commands that signal each other:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Mark feature complete
|
||||||
|
allowed-tools: Write
|
||||||
|
---
|
||||||
|
|
||||||
|
# Mark Feature Complete
|
||||||
|
|
||||||
|
Writing completion marker...
|
||||||
|
|
||||||
|
Creating: .claude/feature-complete.flag
|
||||||
|
|
||||||
|
This signals other commands that feature is ready for:
|
||||||
|
- Integration testing (/integration-test will auto-detect)
|
||||||
|
- Documentation generation (/docs-generate will include)
|
||||||
|
- Release notes (/release-notes will add)
|
||||||
|
|
||||||
|
Feature marked complete.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Other commands check for flag:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Generate release notes
|
||||||
|
allowed-tools: Read, Bash(git:*)
|
||||||
|
---
|
||||||
|
|
||||||
|
Checking for completed features...
|
||||||
|
|
||||||
|
if [ -f .claude/feature-complete.flag ]; then
|
||||||
|
Feature ready for release notes
|
||||||
|
fi
|
||||||
|
|
||||||
|
[Include in release notes]
|
||||||
|
```
|
||||||
|
|
||||||
|
### Workflow Locking
|
||||||
|
|
||||||
|
Prevent concurrent workflow execution:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Start deployment
|
||||||
|
allowed-tools: Read, Write, Bash
|
||||||
|
---
|
||||||
|
|
||||||
|
# Start Deployment
|
||||||
|
|
||||||
|
Checking for active deployments...
|
||||||
|
|
||||||
|
if [ -f .claude/deployment.lock ]; then
|
||||||
|
ERROR: Deployment already in progress
|
||||||
|
Started: [timestamp from lock file]
|
||||||
|
|
||||||
|
Cannot start concurrent deployment.
|
||||||
|
Wait for completion or run /deployment-abort
|
||||||
|
|
||||||
|
Exit.
|
||||||
|
fi
|
||||||
|
|
||||||
|
Creating deployment lock...
|
||||||
|
|
||||||
|
Deployment started. Lock created.
|
||||||
|
[Proceed with deployment]
|
||||||
|
```
|
||||||
|
|
||||||
|
**Lock cleanup:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Complete deployment
|
||||||
|
allowed-tools: Write, Bash
|
||||||
|
---
|
||||||
|
|
||||||
|
Deployment complete.
|
||||||
|
|
||||||
|
Removing deployment lock...
|
||||||
|
rm .claude/deployment.lock
|
||||||
|
|
||||||
|
Ready for next deployment.
|
||||||
|
```
|
||||||
|
|
||||||
|
## Advanced Argument Handling
|
||||||
|
|
||||||
|
### Optional Arguments with Defaults
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Deploy with optional version
|
||||||
|
argument-hint: [environment] [version]
|
||||||
|
---
|
||||||
|
|
||||||
|
Environment: ${1:-staging}
|
||||||
|
Version: ${2:-latest}
|
||||||
|
|
||||||
|
Deploying ${2:-latest} to ${1:-staging}...
|
||||||
|
|
||||||
|
Note: Using defaults for missing arguments:
|
||||||
|
- Environment defaults to 'staging'
|
||||||
|
- Version defaults to 'latest'
|
||||||
|
```
|
||||||
|
|
||||||
|
### Argument Validation
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Deploy to validated environment
|
||||||
|
argument-hint: [environment]
|
||||||
|
---
|
||||||
|
|
||||||
|
Environment: $1
|
||||||
|
|
||||||
|
Validating environment...
|
||||||
|
|
||||||
|
valid_envs="dev staging production"
|
||||||
|
if ! echo "$valid_envs" | grep -w "$1" > /dev/null; then
|
||||||
|
ERROR: Invalid environment '$1'
|
||||||
|
Valid options: dev, staging, production
|
||||||
|
Exit.
|
||||||
|
fi
|
||||||
|
|
||||||
|
Environment validated. Proceeding...
|
||||||
|
```
|
||||||
|
|
||||||
|
### Argument Transformation
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Deploy with shorthand
|
||||||
|
argument-hint: [env-shorthand]
|
||||||
|
---
|
||||||
|
|
||||||
|
Input: $1
|
||||||
|
|
||||||
|
Expanding shorthand:
|
||||||
|
- d/dev → development
|
||||||
|
- s/stg → staging
|
||||||
|
- p/prod → production
|
||||||
|
|
||||||
|
case "$1" in
|
||||||
|
d|dev) ENV="development";;
|
||||||
|
s|stg) ENV="staging";;
|
||||||
|
p|prod) ENV="production";;
|
||||||
|
*) ENV="$1";;
|
||||||
|
esac
|
||||||
|
|
||||||
|
Deploying to: $ENV
|
||||||
|
```
|
||||||
|
|
||||||
|
## Error Handling in Workflows
|
||||||
|
|
||||||
|
### Graceful Failure
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Resilient deployment workflow
|
||||||
|
---
|
||||||
|
|
||||||
|
# Deployment Workflow
|
||||||
|
|
||||||
|
Running steps with error handling...
|
||||||
|
|
||||||
|
## Step 1: Tests
|
||||||
|
!`npm test`
|
||||||
|
|
||||||
|
if [ $? -ne 0 ]; then
|
||||||
|
ERROR: Tests failed
|
||||||
|
|
||||||
|
Options:
|
||||||
|
1. Fix tests and retry
|
||||||
|
2. Skip tests (NOT recommended)
|
||||||
|
3. Abort deployment
|
||||||
|
|
||||||
|
What would you like to do?
|
||||||
|
|
||||||
|
[Wait for user input before continuing]
|
||||||
|
fi
|
||||||
|
|
||||||
|
## Step 2: Build
|
||||||
|
[Continue only if Step 1 succeeded]
|
||||||
|
```
|
||||||
|
|
||||||
|
### Rollback on Failure
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Deployment with rollback
|
||||||
|
---
|
||||||
|
|
||||||
|
# Deploy with Rollback
|
||||||
|
|
||||||
|
Saving current state for rollback...
|
||||||
|
Previous version: !`current-version.sh`
|
||||||
|
|
||||||
|
Deploying new version...
|
||||||
|
|
||||||
|
!`deploy.sh`
|
||||||
|
|
||||||
|
if [ $? -ne 0 ]; then
|
||||||
|
DEPLOYMENT FAILED
|
||||||
|
|
||||||
|
Initiating automatic rollback...
|
||||||
|
!`rollback.sh`
|
||||||
|
|
||||||
|
Rolled back to previous version.
|
||||||
|
Check logs for failure details.
|
||||||
|
fi
|
||||||
|
|
||||||
|
Deployment complete.
|
||||||
|
```
|
||||||
|
|
||||||
|
### Checkpoint Recovery
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Workflow with checkpoints
|
||||||
|
---
|
||||||
|
|
||||||
|
# Multi-Stage Deployment
|
||||||
|
|
||||||
|
## Checkpoint 1: Validation
|
||||||
|
!`validate.sh`
|
||||||
|
echo "checkpoint:validation" >> .claude/deployment-checkpoints.log
|
||||||
|
|
||||||
|
## Checkpoint 2: Build
|
||||||
|
!`build.sh`
|
||||||
|
echo "checkpoint:build" >> .claude/deployment-checkpoints.log
|
||||||
|
|
||||||
|
## Checkpoint 3: Deploy
|
||||||
|
!`deploy.sh`
|
||||||
|
echo "checkpoint:deploy" >> .claude/deployment-checkpoints.log
|
||||||
|
|
||||||
|
If any step fails, resume with:
|
||||||
|
/deployment-resume [last-successful-checkpoint]
|
||||||
|
```
|
||||||
|
|
||||||
|
## Best Practices
|
||||||
|
|
||||||
|
### Workflow Design
|
||||||
|
|
||||||
|
1. **Clear progression**: Number steps, show current position
|
||||||
|
2. **Explicit state**: Don't rely on implicit state
|
||||||
|
3. **User control**: Provide decision points
|
||||||
|
4. **Error recovery**: Handle failures gracefully
|
||||||
|
5. **Progress indication**: Show what's done, what's pending
|
||||||
|
|
||||||
|
### Command Composition
|
||||||
|
|
||||||
|
1. **Single responsibility**: Each command does one thing well
|
||||||
|
2. **Composable design**: Commands work together easily
|
||||||
|
3. **Standard interfaces**: Consistent input/output formats
|
||||||
|
4. **Loose coupling**: Commands don't depend on each other's internals
|
||||||
|
|
||||||
|
### State Management
|
||||||
|
|
||||||
|
1. **Persistent state**: Use .local.md files
|
||||||
|
2. **Atomic updates**: Write complete state files atomically
|
||||||
|
3. **State validation**: Check state file format/completeness
|
||||||
|
4. **Cleanup**: Remove stale state files
|
||||||
|
5. **Documentation**: Document state file formats
|
||||||
|
|
||||||
|
### Error Handling
|
||||||
|
|
||||||
|
1. **Fail fast**: Detect errors early
|
||||||
|
2. **Clear messages**: Explain what went wrong
|
||||||
|
3. **Recovery options**: Provide clear next steps
|
||||||
|
4. **State preservation**: Keep state for recovery
|
||||||
|
5. **Rollback capability**: Support undoing changes
|
||||||
|
|
||||||
|
## Example: Complete Deployment Workflow
|
||||||
|
|
||||||
|
### Initialize Command
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Initialize deployment
|
||||||
|
argument-hint: [environment]
|
||||||
|
allowed-tools: Write, Bash(git:*)
|
||||||
|
---
|
||||||
|
|
||||||
|
# Initialize Deployment to $1
|
||||||
|
|
||||||
|
Creating workflow state...
|
||||||
|
|
||||||
|
\`\`\`yaml
|
||||||
|
---
|
||||||
|
workflow: deployment
|
||||||
|
environment: $1
|
||||||
|
branch: !`git branch --show-current`
|
||||||
|
commit: !`git rev-parse HEAD`
|
||||||
|
stage: initialized
|
||||||
|
timestamp: !`date -u +%Y-%m-%dT%H:%M:%SZ`
|
||||||
|
---
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
Written to .claude/deployment-state.local.md
|
||||||
|
|
||||||
|
Next: Run /deployment-validate
|
||||||
|
```
|
||||||
|
|
||||||
|
### Validation Command
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Validate deployment
|
||||||
|
allowed-tools: Read, Bash
|
||||||
|
---
|
||||||
|
|
||||||
|
Reading state: @.claude/deployment-state.local.md
|
||||||
|
|
||||||
|
Running validation...
|
||||||
|
- Branch check: PASS
|
||||||
|
- Tests: PASS
|
||||||
|
- Build: PASS
|
||||||
|
|
||||||
|
Updating state to 'validated'...
|
||||||
|
|
||||||
|
Next: Run /deployment-execute
|
||||||
|
```
|
||||||
|
|
||||||
|
### Execution Command
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Execute deployment
|
||||||
|
allowed-tools: Read, Bash, Write
|
||||||
|
---
|
||||||
|
|
||||||
|
Reading state: @.claude/deployment-state.local.md
|
||||||
|
|
||||||
|
Executing deployment to [environment]...
|
||||||
|
|
||||||
|
!`deploy.sh [environment]`
|
||||||
|
|
||||||
|
Deployment complete.
|
||||||
|
Updating state to 'completed'...
|
||||||
|
|
||||||
|
Cleanup: /deployment-cleanup
|
||||||
|
```
|
||||||
|
|
||||||
|
### Cleanup Command
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Clean up deployment
|
||||||
|
allowed-tools: Bash
|
||||||
|
---
|
||||||
|
|
||||||
|
Removing deployment state...
|
||||||
|
rm .claude/deployment-state.local.md
|
||||||
|
|
||||||
|
Deployment workflow complete.
|
||||||
|
```
|
||||||
|
|
||||||
|
This complete workflow demonstrates state management, sequential execution, error handling, and clean separation of concerns across multiple commands.
|
||||||
739
skills/command-development/references/documentation-patterns.md
Normal file
739
skills/command-development/references/documentation-patterns.md
Normal file
@@ -0,0 +1,739 @@
|
|||||||
|
# Command Documentation Patterns
|
||||||
|
|
||||||
|
Strategies for creating self-documenting, maintainable commands with excellent user experience.
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
|
||||||
|
Well-documented commands are easier to use, maintain, and distribute. Documentation should be embedded in the command itself, making it immediately accessible to users and maintainers.
|
||||||
|
|
||||||
|
## Self-Documenting Command Structure
|
||||||
|
|
||||||
|
### Complete Command Template
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Clear, actionable description under 60 chars
|
||||||
|
argument-hint: [arg1] [arg2] [optional-arg]
|
||||||
|
allowed-tools: Read, Bash(git:*)
|
||||||
|
model: sonnet
|
||||||
|
---
|
||||||
|
|
||||||
|
<!--
|
||||||
|
COMMAND: command-name
|
||||||
|
VERSION: 1.0.0
|
||||||
|
AUTHOR: Team Name
|
||||||
|
LAST UPDATED: 2025-01-15
|
||||||
|
|
||||||
|
PURPOSE:
|
||||||
|
Detailed explanation of what this command does and why it exists.
|
||||||
|
|
||||||
|
USAGE:
|
||||||
|
/command-name arg1 arg2
|
||||||
|
|
||||||
|
ARGUMENTS:
|
||||||
|
arg1: Description of first argument (required)
|
||||||
|
arg2: Description of second argument (optional, defaults to X)
|
||||||
|
|
||||||
|
EXAMPLES:
|
||||||
|
/command-name feature-branch main
|
||||||
|
→ Compares feature-branch with main
|
||||||
|
|
||||||
|
/command-name my-branch
|
||||||
|
→ Compares my-branch with current branch
|
||||||
|
|
||||||
|
REQUIREMENTS:
|
||||||
|
- Git repository
|
||||||
|
- Branch must exist
|
||||||
|
- Permissions to read repository
|
||||||
|
|
||||||
|
RELATED COMMANDS:
|
||||||
|
/other-command - Related functionality
|
||||||
|
/another-command - Alternative approach
|
||||||
|
|
||||||
|
TROUBLESHOOTING:
|
||||||
|
- If branch not found: Check branch name spelling
|
||||||
|
- If permission denied: Check repository access
|
||||||
|
|
||||||
|
CHANGELOG:
|
||||||
|
v1.0.0 (2025-01-15): Initial release
|
||||||
|
v0.9.0 (2025-01-10): Beta version
|
||||||
|
-->
|
||||||
|
|
||||||
|
# Command Implementation
|
||||||
|
|
||||||
|
[Command prompt content here...]
|
||||||
|
|
||||||
|
[Explain what will happen...]
|
||||||
|
|
||||||
|
[Guide user through steps...]
|
||||||
|
|
||||||
|
[Provide clear output...]
|
||||||
|
```
|
||||||
|
|
||||||
|
### Documentation Comment Sections
|
||||||
|
|
||||||
|
**PURPOSE**: Why the command exists
|
||||||
|
- Problem it solves
|
||||||
|
- Use cases
|
||||||
|
- When to use vs when not to use
|
||||||
|
|
||||||
|
**USAGE**: Basic syntax
|
||||||
|
- Command invocation pattern
|
||||||
|
- Required vs optional arguments
|
||||||
|
- Default values
|
||||||
|
|
||||||
|
**ARGUMENTS**: Detailed argument documentation
|
||||||
|
- Each argument described
|
||||||
|
- Type information
|
||||||
|
- Valid values/ranges
|
||||||
|
- Defaults
|
||||||
|
|
||||||
|
**EXAMPLES**: Concrete usage examples
|
||||||
|
- Common use cases
|
||||||
|
- Edge cases
|
||||||
|
- Expected outputs
|
||||||
|
|
||||||
|
**REQUIREMENTS**: Prerequisites
|
||||||
|
- Dependencies
|
||||||
|
- Permissions
|
||||||
|
- Environmental setup
|
||||||
|
|
||||||
|
**RELATED COMMANDS**: Connections
|
||||||
|
- Similar commands
|
||||||
|
- Complementary commands
|
||||||
|
- Alternative approaches
|
||||||
|
|
||||||
|
**TROUBLESHOOTING**: Common issues
|
||||||
|
- Known problems
|
||||||
|
- Solutions
|
||||||
|
- Workarounds
|
||||||
|
|
||||||
|
**CHANGELOG**: Version history
|
||||||
|
- What changed when
|
||||||
|
- Breaking changes highlighted
|
||||||
|
- Migration guidance
|
||||||
|
|
||||||
|
## In-Line Documentation Patterns
|
||||||
|
|
||||||
|
### Commented Sections
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Complex multi-step command
|
||||||
|
---
|
||||||
|
|
||||||
|
<!-- SECTION 1: VALIDATION -->
|
||||||
|
<!-- This section checks prerequisites before proceeding -->
|
||||||
|
|
||||||
|
Checking prerequisites...
|
||||||
|
- Git repository: !`git rev-parse --git-dir 2>/dev/null`
|
||||||
|
- Branch exists: [validation logic]
|
||||||
|
|
||||||
|
<!-- SECTION 2: ANALYSIS -->
|
||||||
|
<!-- Analyzes the differences between branches -->
|
||||||
|
|
||||||
|
Analyzing differences between $1 and $2...
|
||||||
|
[Analysis logic...]
|
||||||
|
|
||||||
|
<!-- SECTION 3: RECOMMENDATIONS -->
|
||||||
|
<!-- Provides actionable recommendations -->
|
||||||
|
|
||||||
|
Based on analysis, recommend:
|
||||||
|
[Recommendations...]
|
||||||
|
|
||||||
|
<!-- END: Next steps for user -->
|
||||||
|
```
|
||||||
|
|
||||||
|
### Inline Explanations
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Deployment command with inline docs
|
||||||
|
---
|
||||||
|
|
||||||
|
# Deploy to $1
|
||||||
|
|
||||||
|
## Pre-flight Checks
|
||||||
|
|
||||||
|
<!-- We check branch status to prevent deploying from wrong branch -->
|
||||||
|
Current branch: !`git branch --show-current`
|
||||||
|
|
||||||
|
<!-- Production deploys must come from main/master -->
|
||||||
|
if [ "$1" = "production" ] && [ "$(git branch --show-current)" != "main" ]; then
|
||||||
|
⚠️ WARNING: Not on main branch for production deploy
|
||||||
|
This is unusual. Confirm this is intentional.
|
||||||
|
fi
|
||||||
|
|
||||||
|
<!-- Test status ensures we don't deploy broken code -->
|
||||||
|
Running tests: !`npm test`
|
||||||
|
|
||||||
|
✓ All checks passed
|
||||||
|
|
||||||
|
## Deployment
|
||||||
|
|
||||||
|
<!-- Actual deployment happens here -->
|
||||||
|
<!-- Uses blue-green strategy for zero-downtime -->
|
||||||
|
Deploying to $1 environment...
|
||||||
|
[Deployment steps...]
|
||||||
|
|
||||||
|
<!-- Post-deployment verification -->
|
||||||
|
Verifying deployment health...
|
||||||
|
[Health checks...]
|
||||||
|
|
||||||
|
Deployment complete!
|
||||||
|
|
||||||
|
## Next Steps
|
||||||
|
|
||||||
|
<!-- Guide user on what to do after deployment -->
|
||||||
|
1. Monitor logs: /logs $1
|
||||||
|
2. Run smoke tests: /smoke-test $1
|
||||||
|
3. Notify team: /notify-deployment $1
|
||||||
|
```
|
||||||
|
|
||||||
|
### Decision Point Documentation
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Interactive deployment command
|
||||||
|
---
|
||||||
|
|
||||||
|
# Interactive Deployment
|
||||||
|
|
||||||
|
## Configuration Review
|
||||||
|
|
||||||
|
Target: $1
|
||||||
|
Current version: !`cat version.txt`
|
||||||
|
New version: $2
|
||||||
|
|
||||||
|
<!-- DECISION POINT: User confirms configuration -->
|
||||||
|
<!-- This pause allows user to verify everything is correct -->
|
||||||
|
<!-- We can't automatically proceed because deployment is risky -->
|
||||||
|
|
||||||
|
Review the above configuration.
|
||||||
|
|
||||||
|
**Continue with deployment?**
|
||||||
|
- Reply "yes" to proceed
|
||||||
|
- Reply "no" to cancel
|
||||||
|
- Reply "edit" to modify configuration
|
||||||
|
|
||||||
|
[Await user input before continuing...]
|
||||||
|
|
||||||
|
<!-- After user confirms, we proceed with deployment -->
|
||||||
|
<!-- All subsequent steps are automated -->
|
||||||
|
|
||||||
|
Proceeding with deployment...
|
||||||
|
```
|
||||||
|
|
||||||
|
## Help Text Patterns
|
||||||
|
|
||||||
|
### Built-in Help Command
|
||||||
|
|
||||||
|
Create a help subcommand for complex commands:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Main command with help
|
||||||
|
argument-hint: [subcommand] [args]
|
||||||
|
---
|
||||||
|
|
||||||
|
# Command Processor
|
||||||
|
|
||||||
|
if [ "$1" = "help" ] || [ "$1" = "--help" ] || [ "$1" = "-h" ]; then
|
||||||
|
**Command Help**
|
||||||
|
|
||||||
|
USAGE:
|
||||||
|
/command [subcommand] [args]
|
||||||
|
|
||||||
|
SUBCOMMANDS:
|
||||||
|
init [name] Initialize new configuration
|
||||||
|
deploy [env] Deploy to environment
|
||||||
|
status Show current status
|
||||||
|
rollback Rollback last deployment
|
||||||
|
help Show this help
|
||||||
|
|
||||||
|
EXAMPLES:
|
||||||
|
/command init my-project
|
||||||
|
/command deploy staging
|
||||||
|
/command status
|
||||||
|
/command rollback
|
||||||
|
|
||||||
|
For detailed help on a subcommand:
|
||||||
|
/command [subcommand] --help
|
||||||
|
|
||||||
|
Exit.
|
||||||
|
fi
|
||||||
|
|
||||||
|
[Regular command processing...]
|
||||||
|
```
|
||||||
|
|
||||||
|
### Contextual Help
|
||||||
|
|
||||||
|
Provide help based on context:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Context-aware command
|
||||||
|
argument-hint: [operation] [target]
|
||||||
|
---
|
||||||
|
|
||||||
|
# Context-Aware Operation
|
||||||
|
|
||||||
|
if [ -z "$1" ]; then
|
||||||
|
**No operation specified**
|
||||||
|
|
||||||
|
Available operations:
|
||||||
|
- analyze: Analyze target for issues
|
||||||
|
- fix: Apply automatic fixes
|
||||||
|
- report: Generate detailed report
|
||||||
|
|
||||||
|
Usage: /command [operation] [target]
|
||||||
|
|
||||||
|
Examples:
|
||||||
|
/command analyze src/
|
||||||
|
/command fix src/app.js
|
||||||
|
/command report
|
||||||
|
|
||||||
|
Run /command help for more details.
|
||||||
|
|
||||||
|
Exit.
|
||||||
|
fi
|
||||||
|
|
||||||
|
[Command continues if operation provided...]
|
||||||
|
```
|
||||||
|
|
||||||
|
## Error Message Documentation
|
||||||
|
|
||||||
|
### Helpful Error Messages
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Command with good error messages
|
||||||
|
---
|
||||||
|
|
||||||
|
# Validation Command
|
||||||
|
|
||||||
|
if [ -z "$1" ]; then
|
||||||
|
❌ ERROR: Missing required argument
|
||||||
|
|
||||||
|
The 'file-path' argument is required.
|
||||||
|
|
||||||
|
USAGE:
|
||||||
|
/validate [file-path]
|
||||||
|
|
||||||
|
EXAMPLE:
|
||||||
|
/validate src/app.js
|
||||||
|
|
||||||
|
Try again with a file path.
|
||||||
|
|
||||||
|
Exit.
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ ! -f "$1" ]; then
|
||||||
|
❌ ERROR: File not found: $1
|
||||||
|
|
||||||
|
The specified file does not exist or is not accessible.
|
||||||
|
|
||||||
|
COMMON CAUSES:
|
||||||
|
1. Typo in file path
|
||||||
|
2. File was deleted or moved
|
||||||
|
3. Insufficient permissions
|
||||||
|
|
||||||
|
SUGGESTIONS:
|
||||||
|
- Check spelling: $1
|
||||||
|
- Verify file exists: ls -la $(dirname "$1")
|
||||||
|
- Check permissions: ls -l "$1"
|
||||||
|
|
||||||
|
Exit.
|
||||||
|
fi
|
||||||
|
|
||||||
|
[Command continues if validation passes...]
|
||||||
|
```
|
||||||
|
|
||||||
|
### Error Recovery Guidance
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Command with recovery guidance
|
||||||
|
---
|
||||||
|
|
||||||
|
# Operation Command
|
||||||
|
|
||||||
|
Running operation...
|
||||||
|
|
||||||
|
!`risky-operation.sh`
|
||||||
|
|
||||||
|
if [ $? -ne 0 ]; then
|
||||||
|
❌ OPERATION FAILED
|
||||||
|
|
||||||
|
The operation encountered an error and could not complete.
|
||||||
|
|
||||||
|
WHAT HAPPENED:
|
||||||
|
The risky-operation.sh script returned a non-zero exit code.
|
||||||
|
|
||||||
|
WHAT THIS MEANS:
|
||||||
|
- Changes may be partially applied
|
||||||
|
- System may be in inconsistent state
|
||||||
|
- Manual intervention may be needed
|
||||||
|
|
||||||
|
RECOVERY STEPS:
|
||||||
|
1. Check operation logs: cat /tmp/operation.log
|
||||||
|
2. Verify system state: /check-state
|
||||||
|
3. If needed, rollback: /rollback-operation
|
||||||
|
4. Fix underlying issue
|
||||||
|
5. Retry operation: /retry-operation
|
||||||
|
|
||||||
|
NEED HELP?
|
||||||
|
- Check troubleshooting guide: /help troubleshooting
|
||||||
|
- Contact support with error code: ERR_OP_FAILED_001
|
||||||
|
|
||||||
|
Exit.
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
## Usage Example Documentation
|
||||||
|
|
||||||
|
### Embedded Examples
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Command with embedded examples
|
||||||
|
---
|
||||||
|
|
||||||
|
# Feature Command
|
||||||
|
|
||||||
|
This command performs feature analysis with multiple options.
|
||||||
|
|
||||||
|
## Basic Usage
|
||||||
|
|
||||||
|
\`\`\`
|
||||||
|
/feature analyze src/
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
Analyzes all files in src/ directory for feature usage.
|
||||||
|
|
||||||
|
## Advanced Usage
|
||||||
|
|
||||||
|
\`\`\`
|
||||||
|
/feature analyze src/ --detailed
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
Provides detailed analysis including:
|
||||||
|
- Feature breakdown by file
|
||||||
|
- Usage patterns
|
||||||
|
- Optimization suggestions
|
||||||
|
|
||||||
|
## Use Cases
|
||||||
|
|
||||||
|
**Use Case 1: Quick overview**
|
||||||
|
\`\`\`
|
||||||
|
/feature analyze .
|
||||||
|
\`\`\`
|
||||||
|
Get high-level feature summary of entire project.
|
||||||
|
|
||||||
|
**Use Case 2: Specific directory**
|
||||||
|
\`\`\`
|
||||||
|
/feature analyze src/components
|
||||||
|
\`\`\`
|
||||||
|
Focus analysis on components directory only.
|
||||||
|
|
||||||
|
**Use Case 3: Comparison**
|
||||||
|
\`\`\`
|
||||||
|
/feature analyze src/ --compare baseline.json
|
||||||
|
\`\`\`
|
||||||
|
Compare current features against baseline.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
Now processing your request...
|
||||||
|
|
||||||
|
[Command implementation...]
|
||||||
|
```
|
||||||
|
|
||||||
|
### Example-Driven Documentation
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Example-heavy command
|
||||||
|
---
|
||||||
|
|
||||||
|
# Transformation Command
|
||||||
|
|
||||||
|
## What This Does
|
||||||
|
|
||||||
|
Transforms data from one format to another.
|
||||||
|
|
||||||
|
## Examples First
|
||||||
|
|
||||||
|
### Example 1: JSON to YAML
|
||||||
|
**Input:** `data.json`
|
||||||
|
\`\`\`json
|
||||||
|
{"name": "test", "value": 42}
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
**Command:** `/transform data.json yaml`
|
||||||
|
|
||||||
|
**Output:** `data.yaml`
|
||||||
|
\`\`\`yaml
|
||||||
|
name: test
|
||||||
|
value: 42
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
### Example 2: CSV to JSON
|
||||||
|
**Input:** `data.csv`
|
||||||
|
\`\`\`csv
|
||||||
|
name,value
|
||||||
|
test,42
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
**Command:** `/transform data.csv json`
|
||||||
|
|
||||||
|
**Output:** `data.json`
|
||||||
|
\`\`\`json
|
||||||
|
[{"name": "test", "value": "42"}]
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
### Example 3: With Options
|
||||||
|
**Command:** `/transform data.json yaml --pretty --sort-keys`
|
||||||
|
|
||||||
|
**Result:** Formatted YAML with sorted keys
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Your Transformation
|
||||||
|
|
||||||
|
File: $1
|
||||||
|
Format: $2
|
||||||
|
|
||||||
|
[Perform transformation...]
|
||||||
|
```
|
||||||
|
|
||||||
|
## Maintenance Documentation
|
||||||
|
|
||||||
|
### Version and Changelog
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
<!--
|
||||||
|
VERSION: 2.1.0
|
||||||
|
LAST UPDATED: 2025-01-15
|
||||||
|
AUTHOR: DevOps Team
|
||||||
|
|
||||||
|
CHANGELOG:
|
||||||
|
v2.1.0 (2025-01-15):
|
||||||
|
- Added support for YAML configuration
|
||||||
|
- Improved error messages
|
||||||
|
- Fixed bug with special characters in arguments
|
||||||
|
|
||||||
|
v2.0.0 (2025-01-01):
|
||||||
|
- BREAKING: Changed argument order
|
||||||
|
- BREAKING: Removed deprecated --old-flag
|
||||||
|
- Added new validation checks
|
||||||
|
- Migration guide: /migration-v2
|
||||||
|
|
||||||
|
v1.5.0 (2024-12-15):
|
||||||
|
- Added --verbose flag
|
||||||
|
- Improved performance by 50%
|
||||||
|
|
||||||
|
v1.0.0 (2024-12-01):
|
||||||
|
- Initial stable release
|
||||||
|
|
||||||
|
MIGRATION NOTES:
|
||||||
|
From v1.x to v2.0:
|
||||||
|
Old: /command arg1 arg2 --old-flag
|
||||||
|
New: /command arg2 arg1
|
||||||
|
|
||||||
|
The --old-flag is removed. Use --new-flag instead.
|
||||||
|
|
||||||
|
DEPRECATION WARNINGS:
|
||||||
|
- The --legacy-mode flag is deprecated as of v2.1.0
|
||||||
|
- Will be removed in v3.0.0 (estimated 2025-06-01)
|
||||||
|
- Use --modern-mode instead
|
||||||
|
|
||||||
|
KNOWN ISSUES:
|
||||||
|
- #123: Slow performance with large files (workaround: use --stream flag)
|
||||||
|
- #456: Special characters in Windows (fix planned for v2.2.0)
|
||||||
|
-->
|
||||||
|
```
|
||||||
|
|
||||||
|
### Maintenance Notes
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
<!--
|
||||||
|
MAINTENANCE NOTES:
|
||||||
|
|
||||||
|
CODE STRUCTURE:
|
||||||
|
- Lines 1-50: Argument parsing and validation
|
||||||
|
- Lines 51-100: Main processing logic
|
||||||
|
- Lines 101-150: Output formatting
|
||||||
|
- Lines 151-200: Error handling
|
||||||
|
|
||||||
|
DEPENDENCIES:
|
||||||
|
- Requires git 2.x or later
|
||||||
|
- Uses jq for JSON processing
|
||||||
|
- Needs bash 4.0+ for associative arrays
|
||||||
|
|
||||||
|
PERFORMANCE:
|
||||||
|
- Fast path for small inputs (< 1MB)
|
||||||
|
- Streams large files to avoid memory issues
|
||||||
|
- Caches results in /tmp for 1 hour
|
||||||
|
|
||||||
|
SECURITY CONSIDERATIONS:
|
||||||
|
- Validates all inputs to prevent injection
|
||||||
|
- Uses allowed-tools to limit Bash access
|
||||||
|
- No credentials in command file
|
||||||
|
|
||||||
|
TESTING:
|
||||||
|
- Unit tests: tests/command-test.sh
|
||||||
|
- Integration tests: tests/integration/
|
||||||
|
- Manual test checklist: tests/manual-checklist.md
|
||||||
|
|
||||||
|
FUTURE IMPROVEMENTS:
|
||||||
|
- TODO: Add support for TOML format
|
||||||
|
- TODO: Implement parallel processing
|
||||||
|
- TODO: Add progress bar for large files
|
||||||
|
|
||||||
|
RELATED FILES:
|
||||||
|
- lib/parser.sh: Shared parsing logic
|
||||||
|
- lib/formatter.sh: Output formatting
|
||||||
|
- config/defaults.yml: Default configuration
|
||||||
|
-->
|
||||||
|
```
|
||||||
|
|
||||||
|
## README Documentation
|
||||||
|
|
||||||
|
Commands should have companion README files:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
# Command Name
|
||||||
|
|
||||||
|
Brief description of what the command does.
|
||||||
|
|
||||||
|
## Installation
|
||||||
|
|
||||||
|
This command is part of the [plugin-name] plugin.
|
||||||
|
|
||||||
|
Install with:
|
||||||
|
\`\`\`
|
||||||
|
/plugin install plugin-name
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
## Usage
|
||||||
|
|
||||||
|
Basic usage:
|
||||||
|
\`\`\`
|
||||||
|
/command-name [arg1] [arg2]
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
## Arguments
|
||||||
|
|
||||||
|
- `arg1`: Description (required)
|
||||||
|
- `arg2`: Description (optional, defaults to X)
|
||||||
|
|
||||||
|
## Examples
|
||||||
|
|
||||||
|
### Example 1: Basic Usage
|
||||||
|
\`\`\`
|
||||||
|
/command-name value1 value2
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
Description of what happens.
|
||||||
|
|
||||||
|
### Example 2: Advanced Usage
|
||||||
|
\`\`\`
|
||||||
|
/command-name value1 --option
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
Description of advanced feature.
|
||||||
|
|
||||||
|
## Configuration
|
||||||
|
|
||||||
|
Optional configuration file: `.claude/command-name.local.md`
|
||||||
|
|
||||||
|
\`\`\`markdown
|
||||||
|
---
|
||||||
|
default_arg: value
|
||||||
|
enable_feature: true
|
||||||
|
---
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
## Requirements
|
||||||
|
|
||||||
|
- Git 2.x or later
|
||||||
|
- jq (for JSON processing)
|
||||||
|
- Node.js 14+ (optional, for advanced features)
|
||||||
|
|
||||||
|
## Troubleshooting
|
||||||
|
|
||||||
|
### Issue: Command not found
|
||||||
|
|
||||||
|
**Solution:** Ensure plugin is installed and enabled.
|
||||||
|
|
||||||
|
### Issue: Permission denied
|
||||||
|
|
||||||
|
**Solution:** Check file permissions and allowed-tools setting.
|
||||||
|
|
||||||
|
## Contributing
|
||||||
|
|
||||||
|
Contributions welcome! See [CONTRIBUTING.md](CONTRIBUTING.md).
|
||||||
|
|
||||||
|
## License
|
||||||
|
|
||||||
|
MIT License - See [LICENSE](LICENSE).
|
||||||
|
|
||||||
|
## Support
|
||||||
|
|
||||||
|
- Issues: https://github.com/user/plugin/issues
|
||||||
|
- Docs: https://docs.example.com
|
||||||
|
- Email: support@example.com
|
||||||
|
```
|
||||||
|
|
||||||
|
## Best Practices
|
||||||
|
|
||||||
|
### Documentation Principles
|
||||||
|
|
||||||
|
1. **Write for your future self**: Assume you'll forget details
|
||||||
|
2. **Examples before explanations**: Show, then tell
|
||||||
|
3. **Progressive disclosure**: Basic info first, details available
|
||||||
|
4. **Keep it current**: Update docs when code changes
|
||||||
|
5. **Test your docs**: Verify examples actually work
|
||||||
|
|
||||||
|
### Documentation Locations
|
||||||
|
|
||||||
|
1. **In command file**: Core usage, examples, inline explanations
|
||||||
|
2. **README**: Installation, configuration, troubleshooting
|
||||||
|
3. **Separate docs**: Detailed guides, tutorials, API reference
|
||||||
|
4. **Comments**: Implementation details for maintainers
|
||||||
|
|
||||||
|
### Documentation Style
|
||||||
|
|
||||||
|
1. **Clear and concise**: No unnecessary words
|
||||||
|
2. **Active voice**: "Run the command" not "The command can be run"
|
||||||
|
3. **Consistent terminology**: Use same terms throughout
|
||||||
|
4. **Formatted well**: Use headings, lists, code blocks
|
||||||
|
5. **Accessible**: Assume reader is beginner
|
||||||
|
|
||||||
|
### Documentation Maintenance
|
||||||
|
|
||||||
|
1. **Version everything**: Track what changed when
|
||||||
|
2. **Deprecate gracefully**: Warn before removing features
|
||||||
|
3. **Migration guides**: Help users upgrade
|
||||||
|
4. **Archive old docs**: Keep old versions accessible
|
||||||
|
5. **Review regularly**: Ensure docs match reality
|
||||||
|
|
||||||
|
## Documentation Checklist
|
||||||
|
|
||||||
|
Before releasing a command:
|
||||||
|
|
||||||
|
- [ ] Description in frontmatter is clear
|
||||||
|
- [ ] argument-hint documents all arguments
|
||||||
|
- [ ] Usage examples in comments
|
||||||
|
- [ ] Common use cases shown
|
||||||
|
- [ ] Error messages are helpful
|
||||||
|
- [ ] Requirements documented
|
||||||
|
- [ ] Related commands listed
|
||||||
|
- [ ] Changelog maintained
|
||||||
|
- [ ] Version number updated
|
||||||
|
- [ ] README created/updated
|
||||||
|
- [ ] Examples actually work
|
||||||
|
- [ ] Troubleshooting section complete
|
||||||
|
|
||||||
|
With good documentation, commands become self-service, reducing support burden and improving user experience.
|
||||||
463
skills/command-development/references/frontmatter-reference.md
Normal file
463
skills/command-development/references/frontmatter-reference.md
Normal file
@@ -0,0 +1,463 @@
|
|||||||
|
# Command Frontmatter Reference
|
||||||
|
|
||||||
|
Complete reference for YAML frontmatter fields in slash commands.
|
||||||
|
|
||||||
|
## Frontmatter Overview
|
||||||
|
|
||||||
|
YAML frontmatter is optional metadata at the start of command files:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Brief description
|
||||||
|
allowed-tools: Read, Write
|
||||||
|
model: sonnet
|
||||||
|
argument-hint: [arg1] [arg2]
|
||||||
|
---
|
||||||
|
|
||||||
|
Command prompt content here...
|
||||||
|
```
|
||||||
|
|
||||||
|
All fields are optional. Commands work without any frontmatter.
|
||||||
|
|
||||||
|
## Field Specifications
|
||||||
|
|
||||||
|
### description
|
||||||
|
|
||||||
|
**Type:** String
|
||||||
|
**Required:** No
|
||||||
|
**Default:** First line of command prompt
|
||||||
|
**Max Length:** ~60 characters recommended for `/help` display
|
||||||
|
|
||||||
|
**Purpose:** Describes what the command does, shown in `/help` output
|
||||||
|
|
||||||
|
**Examples:**
|
||||||
|
```yaml
|
||||||
|
description: Review code for security issues
|
||||||
|
```
|
||||||
|
```yaml
|
||||||
|
description: Deploy to staging environment
|
||||||
|
```
|
||||||
|
```yaml
|
||||||
|
description: Generate API documentation
|
||||||
|
```
|
||||||
|
|
||||||
|
**Best practices:**
|
||||||
|
- Keep under 60 characters for clean display
|
||||||
|
- Start with verb (Review, Deploy, Generate)
|
||||||
|
- Be specific about what command does
|
||||||
|
- Avoid redundant "command" or "slash command"
|
||||||
|
|
||||||
|
**Good:**
|
||||||
|
- ✅ "Review PR for code quality and security"
|
||||||
|
- ✅ "Deploy application to specified environment"
|
||||||
|
- ✅ "Generate comprehensive API documentation"
|
||||||
|
|
||||||
|
**Bad:**
|
||||||
|
- ❌ "This command reviews PRs" (unnecessary "This command")
|
||||||
|
- ❌ "Review" (too vague)
|
||||||
|
- ❌ "A command that reviews pull requests for code quality, security issues, and best practices" (too long)
|
||||||
|
|
||||||
|
### allowed-tools
|
||||||
|
|
||||||
|
**Type:** String or Array of strings
|
||||||
|
**Required:** No
|
||||||
|
**Default:** Inherits from conversation permissions
|
||||||
|
|
||||||
|
**Purpose:** Restrict or specify which tools command can use
|
||||||
|
|
||||||
|
**Formats:**
|
||||||
|
|
||||||
|
**Single tool:**
|
||||||
|
```yaml
|
||||||
|
allowed-tools: Read
|
||||||
|
```
|
||||||
|
|
||||||
|
**Multiple tools (comma-separated):**
|
||||||
|
```yaml
|
||||||
|
allowed-tools: Read, Write, Edit
|
||||||
|
```
|
||||||
|
|
||||||
|
**Multiple tools (array):**
|
||||||
|
```yaml
|
||||||
|
allowed-tools:
|
||||||
|
- Read
|
||||||
|
- Write
|
||||||
|
- Bash(git:*)
|
||||||
|
```
|
||||||
|
|
||||||
|
**Tool Patterns:**
|
||||||
|
|
||||||
|
**Specific tools:**
|
||||||
|
```yaml
|
||||||
|
allowed-tools: Read, Grep, Edit
|
||||||
|
```
|
||||||
|
|
||||||
|
**Bash with command filter:**
|
||||||
|
```yaml
|
||||||
|
allowed-tools: Bash(git:*) # Only git commands
|
||||||
|
allowed-tools: Bash(npm:*) # Only npm commands
|
||||||
|
allowed-tools: Bash(docker:*) # Only docker commands
|
||||||
|
```
|
||||||
|
|
||||||
|
**All tools (not recommended):**
|
||||||
|
```yaml
|
||||||
|
allowed-tools: "*"
|
||||||
|
```
|
||||||
|
|
||||||
|
**When to use:**
|
||||||
|
|
||||||
|
1. **Security:** Restrict command to safe operations
|
||||||
|
```yaml
|
||||||
|
allowed-tools: Read, Grep # Read-only command
|
||||||
|
```
|
||||||
|
|
||||||
|
2. **Clarity:** Document required tools
|
||||||
|
```yaml
|
||||||
|
allowed-tools: Bash(git:*), Read
|
||||||
|
```
|
||||||
|
|
||||||
|
3. **Bash execution:** Enable bash command output
|
||||||
|
```yaml
|
||||||
|
allowed-tools: Bash(git status:*), Bash(git diff:*)
|
||||||
|
```
|
||||||
|
|
||||||
|
**Best practices:**
|
||||||
|
- Be as restrictive as possible
|
||||||
|
- Use command filters for Bash (e.g., `git:*` not `*`)
|
||||||
|
- Only specify when different from conversation permissions
|
||||||
|
- Document why specific tools are needed
|
||||||
|
|
||||||
|
### model
|
||||||
|
|
||||||
|
**Type:** String
|
||||||
|
**Required:** No
|
||||||
|
**Default:** Inherits from conversation
|
||||||
|
**Values:** `sonnet`, `opus`, `haiku`
|
||||||
|
|
||||||
|
**Purpose:** Specify which Claude model executes the command
|
||||||
|
|
||||||
|
**Examples:**
|
||||||
|
```yaml
|
||||||
|
model: haiku # Fast, efficient for simple tasks
|
||||||
|
```
|
||||||
|
```yaml
|
||||||
|
model: sonnet # Balanced performance (default)
|
||||||
|
```
|
||||||
|
```yaml
|
||||||
|
model: opus # Maximum capability for complex tasks
|
||||||
|
```
|
||||||
|
|
||||||
|
**When to use:**
|
||||||
|
|
||||||
|
**Use `haiku` for:**
|
||||||
|
- Simple, formulaic commands
|
||||||
|
- Fast execution needed
|
||||||
|
- Low complexity tasks
|
||||||
|
- Frequent invocations
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
---
|
||||||
|
description: Format code file
|
||||||
|
model: haiku
|
||||||
|
---
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use `sonnet` for:**
|
||||||
|
- Standard commands (default)
|
||||||
|
- Balanced speed/quality
|
||||||
|
- Most common use cases
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
---
|
||||||
|
description: Review code changes
|
||||||
|
model: sonnet
|
||||||
|
---
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use `opus` for:**
|
||||||
|
- Complex analysis
|
||||||
|
- Architectural decisions
|
||||||
|
- Deep code understanding
|
||||||
|
- Critical tasks
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
---
|
||||||
|
description: Analyze system architecture
|
||||||
|
model: opus
|
||||||
|
---
|
||||||
|
```
|
||||||
|
|
||||||
|
**Best practices:**
|
||||||
|
- Omit unless specific need
|
||||||
|
- Use `haiku` for speed when possible
|
||||||
|
- Reserve `opus` for genuinely complex tasks
|
||||||
|
- Test with different models to find right balance
|
||||||
|
|
||||||
|
### argument-hint
|
||||||
|
|
||||||
|
**Type:** String
|
||||||
|
**Required:** No
|
||||||
|
**Default:** None
|
||||||
|
|
||||||
|
**Purpose:** Document expected arguments for users and autocomplete
|
||||||
|
|
||||||
|
**Format:**
|
||||||
|
```yaml
|
||||||
|
argument-hint: [arg1] [arg2] [optional-arg]
|
||||||
|
```
|
||||||
|
|
||||||
|
**Examples:**
|
||||||
|
|
||||||
|
**Single argument:**
|
||||||
|
```yaml
|
||||||
|
argument-hint: [pr-number]
|
||||||
|
```
|
||||||
|
|
||||||
|
**Multiple required arguments:**
|
||||||
|
```yaml
|
||||||
|
argument-hint: [environment] [version]
|
||||||
|
```
|
||||||
|
|
||||||
|
**Optional arguments:**
|
||||||
|
```yaml
|
||||||
|
argument-hint: [file-path] [options]
|
||||||
|
```
|
||||||
|
|
||||||
|
**Descriptive names:**
|
||||||
|
```yaml
|
||||||
|
argument-hint: [source-branch] [target-branch] [commit-message]
|
||||||
|
```
|
||||||
|
|
||||||
|
**Best practices:**
|
||||||
|
- Use square brackets `[]` for each argument
|
||||||
|
- Use descriptive names (not `arg1`, `arg2`)
|
||||||
|
- Indicate optional vs required in description
|
||||||
|
- Match order to positional arguments in command
|
||||||
|
- Keep concise but clear
|
||||||
|
|
||||||
|
**Examples by pattern:**
|
||||||
|
|
||||||
|
**Simple command:**
|
||||||
|
```yaml
|
||||||
|
---
|
||||||
|
description: Fix issue by number
|
||||||
|
argument-hint: [issue-number]
|
||||||
|
---
|
||||||
|
|
||||||
|
Fix issue #$1...
|
||||||
|
```
|
||||||
|
|
||||||
|
**Multi-argument:**
|
||||||
|
```yaml
|
||||||
|
---
|
||||||
|
description: Deploy to environment
|
||||||
|
argument-hint: [app-name] [environment] [version]
|
||||||
|
---
|
||||||
|
|
||||||
|
Deploy $1 to $2 using version $3...
|
||||||
|
```
|
||||||
|
|
||||||
|
**With options:**
|
||||||
|
```yaml
|
||||||
|
---
|
||||||
|
description: Run tests with options
|
||||||
|
argument-hint: [test-pattern] [options]
|
||||||
|
---
|
||||||
|
|
||||||
|
Run tests matching $1 with options: $2
|
||||||
|
```
|
||||||
|
|
||||||
|
### disable-model-invocation
|
||||||
|
|
||||||
|
**Type:** Boolean
|
||||||
|
**Required:** No
|
||||||
|
**Default:** false
|
||||||
|
|
||||||
|
**Purpose:** Prevent SlashCommand tool from programmatically invoking command
|
||||||
|
|
||||||
|
**Examples:**
|
||||||
|
```yaml
|
||||||
|
disable-model-invocation: true
|
||||||
|
```
|
||||||
|
|
||||||
|
**When to use:**
|
||||||
|
|
||||||
|
1. **Manual-only commands:** Commands requiring user judgment
|
||||||
|
```yaml
|
||||||
|
---
|
||||||
|
description: Approve deployment to production
|
||||||
|
disable-model-invocation: true
|
||||||
|
---
|
||||||
|
```
|
||||||
|
|
||||||
|
2. **Destructive operations:** Commands with irreversible effects
|
||||||
|
```yaml
|
||||||
|
---
|
||||||
|
description: Delete all test data
|
||||||
|
disable-model-invocation: true
|
||||||
|
---
|
||||||
|
```
|
||||||
|
|
||||||
|
3. **Interactive workflows:** Commands needing user input
|
||||||
|
```yaml
|
||||||
|
---
|
||||||
|
description: Walk through setup wizard
|
||||||
|
disable-model-invocation: true
|
||||||
|
---
|
||||||
|
```
|
||||||
|
|
||||||
|
**Default behavior (false):**
|
||||||
|
- Command available to SlashCommand tool
|
||||||
|
- Claude can invoke programmatically
|
||||||
|
- Still available for manual invocation
|
||||||
|
|
||||||
|
**When true:**
|
||||||
|
- Command only invokable by user typing `/command`
|
||||||
|
- Not available to SlashCommand tool
|
||||||
|
- Safer for sensitive operations
|
||||||
|
|
||||||
|
**Best practices:**
|
||||||
|
- Use sparingly (limits Claude's autonomy)
|
||||||
|
- Document why in command comments
|
||||||
|
- Consider if command should exist if always manual
|
||||||
|
|
||||||
|
## Complete Examples
|
||||||
|
|
||||||
|
### Minimal Command
|
||||||
|
|
||||||
|
No frontmatter needed:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
Review this code for common issues and suggest improvements.
|
||||||
|
```
|
||||||
|
|
||||||
|
### Simple Command
|
||||||
|
|
||||||
|
Just description:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Review code for issues
|
||||||
|
---
|
||||||
|
|
||||||
|
Review this code for common issues and suggest improvements.
|
||||||
|
```
|
||||||
|
|
||||||
|
### Standard Command
|
||||||
|
|
||||||
|
Description and tools:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Review Git changes
|
||||||
|
allowed-tools: Bash(git:*), Read
|
||||||
|
---
|
||||||
|
|
||||||
|
Current changes: !`git diff --name-only`
|
||||||
|
|
||||||
|
Review each changed file for:
|
||||||
|
- Code quality
|
||||||
|
- Potential bugs
|
||||||
|
- Best practices
|
||||||
|
```
|
||||||
|
|
||||||
|
### Complex Command
|
||||||
|
|
||||||
|
All common fields:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Deploy application to environment
|
||||||
|
argument-hint: [app-name] [environment] [version]
|
||||||
|
allowed-tools: Bash(kubectl:*), Bash(helm:*), Read
|
||||||
|
model: sonnet
|
||||||
|
---
|
||||||
|
|
||||||
|
Deploy $1 to $2 environment using version $3
|
||||||
|
|
||||||
|
Pre-deployment checks:
|
||||||
|
- Verify $2 configuration
|
||||||
|
- Check cluster status: !`kubectl cluster-info`
|
||||||
|
- Validate version $3 exists
|
||||||
|
|
||||||
|
Proceed with deployment following deployment runbook.
|
||||||
|
```
|
||||||
|
|
||||||
|
### Manual-Only Command
|
||||||
|
|
||||||
|
Restricted invocation:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Approve production deployment
|
||||||
|
argument-hint: [deployment-id]
|
||||||
|
disable-model-invocation: true
|
||||||
|
allowed-tools: Bash(gh:*)
|
||||||
|
---
|
||||||
|
|
||||||
|
<!--
|
||||||
|
MANUAL APPROVAL REQUIRED
|
||||||
|
This command requires human judgment and cannot be automated.
|
||||||
|
-->
|
||||||
|
|
||||||
|
Review deployment $1 for production approval:
|
||||||
|
|
||||||
|
Deployment details: !`gh api /deployments/$1`
|
||||||
|
|
||||||
|
Verify:
|
||||||
|
- All tests passed
|
||||||
|
- Security scan clean
|
||||||
|
- Stakeholder approval
|
||||||
|
- Rollback plan ready
|
||||||
|
|
||||||
|
Type "APPROVED" to confirm deployment.
|
||||||
|
```
|
||||||
|
|
||||||
|
## Validation
|
||||||
|
|
||||||
|
### Common Errors
|
||||||
|
|
||||||
|
**Invalid YAML syntax:**
|
||||||
|
```yaml
|
||||||
|
---
|
||||||
|
description: Missing quote
|
||||||
|
allowed-tools: Read, Write
|
||||||
|
model: sonnet
|
||||||
|
--- # ❌ Missing closing quote above
|
||||||
|
```
|
||||||
|
|
||||||
|
**Fix:** Validate YAML syntax
|
||||||
|
|
||||||
|
**Incorrect tool specification:**
|
||||||
|
```yaml
|
||||||
|
allowed-tools: Bash # ❌ Missing command filter
|
||||||
|
```
|
||||||
|
|
||||||
|
**Fix:** Use `Bash(git:*)` format
|
||||||
|
|
||||||
|
**Invalid model name:**
|
||||||
|
```yaml
|
||||||
|
model: gpt4 # ❌ Not a valid Claude model
|
||||||
|
```
|
||||||
|
|
||||||
|
**Fix:** Use `sonnet`, `opus`, or `haiku`
|
||||||
|
|
||||||
|
### Validation Checklist
|
||||||
|
|
||||||
|
Before committing command:
|
||||||
|
- [ ] YAML syntax valid (no errors)
|
||||||
|
- [ ] Description under 60 characters
|
||||||
|
- [ ] allowed-tools uses proper format
|
||||||
|
- [ ] model is valid value if specified
|
||||||
|
- [ ] argument-hint matches positional arguments
|
||||||
|
- [ ] disable-model-invocation used appropriately
|
||||||
|
|
||||||
|
## Best Practices Summary
|
||||||
|
|
||||||
|
1. **Start minimal:** Add frontmatter only when needed
|
||||||
|
2. **Document arguments:** Always use argument-hint with arguments
|
||||||
|
3. **Restrict tools:** Use most restrictive allowed-tools that works
|
||||||
|
4. **Choose right model:** Use haiku for speed, opus for complexity
|
||||||
|
5. **Manual-only sparingly:** Only use disable-model-invocation when necessary
|
||||||
|
6. **Clear descriptions:** Make commands discoverable in `/help`
|
||||||
|
7. **Test thoroughly:** Verify frontmatter works as expected
|
||||||
920
skills/command-development/references/interactive-commands.md
Normal file
920
skills/command-development/references/interactive-commands.md
Normal file
@@ -0,0 +1,920 @@
|
|||||||
|
# Interactive Command Patterns
|
||||||
|
|
||||||
|
Comprehensive guide to creating commands that gather user feedback and make decisions through the AskUserQuestion tool.
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
|
||||||
|
Some commands need user input that doesn't work well with simple arguments. For example:
|
||||||
|
- Choosing between multiple complex options with trade-offs
|
||||||
|
- Selecting multiple items from a list
|
||||||
|
- Making decisions that require explanation
|
||||||
|
- Gathering preferences or configuration interactively
|
||||||
|
|
||||||
|
For these cases, use the **AskUserQuestion tool** within command execution rather than relying on command arguments.
|
||||||
|
|
||||||
|
## When to Use AskUserQuestion
|
||||||
|
|
||||||
|
### Use AskUserQuestion When:
|
||||||
|
|
||||||
|
1. **Multiple choice decisions** with explanations needed
|
||||||
|
2. **Complex options** that require context to choose
|
||||||
|
3. **Multi-select scenarios** (choosing multiple items)
|
||||||
|
4. **Preference gathering** for configuration
|
||||||
|
5. **Interactive workflows** that adapt based on answers
|
||||||
|
|
||||||
|
### Use Command Arguments When:
|
||||||
|
|
||||||
|
1. **Simple values** (file paths, numbers, names)
|
||||||
|
2. **Known inputs** user already has
|
||||||
|
3. **Scriptable workflows** that should be automatable
|
||||||
|
4. **Fast invocations** where prompting would slow down
|
||||||
|
|
||||||
|
## AskUserQuestion Basics
|
||||||
|
|
||||||
|
### Tool Parameters
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
{
|
||||||
|
questions: [
|
||||||
|
{
|
||||||
|
question: "Which authentication method should we use?",
|
||||||
|
header: "Auth method", // Short label (max 12 chars)
|
||||||
|
multiSelect: false, // true for multiple selection
|
||||||
|
options: [
|
||||||
|
{
|
||||||
|
label: "OAuth 2.0",
|
||||||
|
description: "Industry standard, supports multiple providers"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: "JWT",
|
||||||
|
description: "Stateless, good for APIs"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: "Session",
|
||||||
|
description: "Traditional, server-side state"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Key points:**
|
||||||
|
- Users can always choose "Other" to provide custom input (automatic)
|
||||||
|
- `multiSelect: true` allows selecting multiple options
|
||||||
|
- Options should be 2-4 choices (not more)
|
||||||
|
- Can ask 1-4 questions per tool call
|
||||||
|
|
||||||
|
## Command Pattern for User Interaction
|
||||||
|
|
||||||
|
### Basic Interactive Command
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Interactive setup command
|
||||||
|
allowed-tools: AskUserQuestion, Write
|
||||||
|
---
|
||||||
|
|
||||||
|
# Interactive Plugin Setup
|
||||||
|
|
||||||
|
This command will guide you through configuring the plugin with a series of questions.
|
||||||
|
|
||||||
|
## Step 1: Gather Configuration
|
||||||
|
|
||||||
|
Use the AskUserQuestion tool to ask:
|
||||||
|
|
||||||
|
**Question 1 - Deployment target:**
|
||||||
|
- header: "Deploy to"
|
||||||
|
- question: "Which deployment platform will you use?"
|
||||||
|
- options:
|
||||||
|
- AWS (Amazon Web Services with ECS/EKS)
|
||||||
|
- GCP (Google Cloud with GKE)
|
||||||
|
- Azure (Microsoft Azure with AKS)
|
||||||
|
- Local (Docker on local machine)
|
||||||
|
|
||||||
|
**Question 2 - Environment strategy:**
|
||||||
|
- header: "Environments"
|
||||||
|
- question: "How many environments do you need?"
|
||||||
|
- options:
|
||||||
|
- Single (Just production)
|
||||||
|
- Standard (Dev, Staging, Production)
|
||||||
|
- Complete (Dev, QA, Staging, Production)
|
||||||
|
|
||||||
|
**Question 3 - Features to enable:**
|
||||||
|
- header: "Features"
|
||||||
|
- question: "Which features do you want to enable?"
|
||||||
|
- multiSelect: true
|
||||||
|
- options:
|
||||||
|
- Auto-scaling (Automatic resource scaling)
|
||||||
|
- Monitoring (Health checks and metrics)
|
||||||
|
- CI/CD (Automated deployment pipeline)
|
||||||
|
- Backups (Automated database backups)
|
||||||
|
|
||||||
|
## Step 2: Process Answers
|
||||||
|
|
||||||
|
Based on the answers received from AskUserQuestion:
|
||||||
|
|
||||||
|
1. Parse the deployment target choice
|
||||||
|
2. Set up environment-specific configuration
|
||||||
|
3. Enable selected features
|
||||||
|
4. Generate configuration files
|
||||||
|
|
||||||
|
## Step 3: Generate Configuration
|
||||||
|
|
||||||
|
Create `.claude/plugin-name.local.md` with:
|
||||||
|
|
||||||
|
\`\`\`yaml
|
||||||
|
---
|
||||||
|
deployment_target: [answer from Q1]
|
||||||
|
environments: [answer from Q2]
|
||||||
|
features:
|
||||||
|
auto_scaling: [true if selected in Q3]
|
||||||
|
monitoring: [true if selected in Q3]
|
||||||
|
ci_cd: [true if selected in Q3]
|
||||||
|
backups: [true if selected in Q3]
|
||||||
|
---
|
||||||
|
|
||||||
|
# Plugin Configuration
|
||||||
|
|
||||||
|
Generated: [timestamp]
|
||||||
|
Target: [deployment_target]
|
||||||
|
Environments: [environments]
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
## Step 4: Confirm and Next Steps
|
||||||
|
|
||||||
|
Confirm configuration created and guide user on next steps.
|
||||||
|
```
|
||||||
|
|
||||||
|
### Multi-Stage Interactive Workflow
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Multi-stage interactive workflow
|
||||||
|
allowed-tools: AskUserQuestion, Read, Write, Bash
|
||||||
|
---
|
||||||
|
|
||||||
|
# Multi-Stage Deployment Setup
|
||||||
|
|
||||||
|
This command walks through deployment setup in stages, adapting based on your answers.
|
||||||
|
|
||||||
|
## Stage 1: Basic Configuration
|
||||||
|
|
||||||
|
Use AskUserQuestion to ask about deployment basics.
|
||||||
|
|
||||||
|
Based on answers, determine which additional questions to ask.
|
||||||
|
|
||||||
|
## Stage 2: Advanced Options (Conditional)
|
||||||
|
|
||||||
|
If user selected "Advanced" deployment in Stage 1:
|
||||||
|
|
||||||
|
Use AskUserQuestion to ask about:
|
||||||
|
- Load balancing strategy
|
||||||
|
- Caching configuration
|
||||||
|
- Security hardening options
|
||||||
|
|
||||||
|
If user selected "Simple" deployment:
|
||||||
|
- Skip advanced questions
|
||||||
|
- Use sensible defaults
|
||||||
|
|
||||||
|
## Stage 3: Confirmation
|
||||||
|
|
||||||
|
Show summary of all selections.
|
||||||
|
|
||||||
|
Use AskUserQuestion for final confirmation:
|
||||||
|
- header: "Confirm"
|
||||||
|
- question: "Does this configuration look correct?"
|
||||||
|
- options:
|
||||||
|
- Yes (Proceed with setup)
|
||||||
|
- No (Start over)
|
||||||
|
- Modify (Let me adjust specific settings)
|
||||||
|
|
||||||
|
If "Modify", ask which specific setting to change.
|
||||||
|
|
||||||
|
## Stage 4: Execute Setup
|
||||||
|
|
||||||
|
Based on confirmed configuration, execute setup steps.
|
||||||
|
```
|
||||||
|
|
||||||
|
## Interactive Question Design
|
||||||
|
|
||||||
|
### Question Structure
|
||||||
|
|
||||||
|
**Good questions:**
|
||||||
|
```markdown
|
||||||
|
Question: "Which database should we use for this project?"
|
||||||
|
Header: "Database"
|
||||||
|
Options:
|
||||||
|
- PostgreSQL (Relational, ACID compliant, best for complex queries)
|
||||||
|
- MongoDB (Document store, flexible schema, best for rapid iteration)
|
||||||
|
- Redis (In-memory, fast, best for caching and sessions)
|
||||||
|
```
|
||||||
|
|
||||||
|
**Poor questions:**
|
||||||
|
```markdown
|
||||||
|
Question: "Database?" // Too vague
|
||||||
|
Header: "DB" // Unclear abbreviation
|
||||||
|
Options:
|
||||||
|
- Option 1 // Not descriptive
|
||||||
|
- Option 2
|
||||||
|
```
|
||||||
|
|
||||||
|
### Option Design Best Practices
|
||||||
|
|
||||||
|
**Clear labels:**
|
||||||
|
- Use 1-5 words
|
||||||
|
- Specific and descriptive
|
||||||
|
- No jargon without context
|
||||||
|
|
||||||
|
**Helpful descriptions:**
|
||||||
|
- Explain what the option means
|
||||||
|
- Mention key benefits or trade-offs
|
||||||
|
- Help user make informed decision
|
||||||
|
- Keep to 1-2 sentences
|
||||||
|
|
||||||
|
**Appropriate number:**
|
||||||
|
- 2-4 options per question
|
||||||
|
- Don't overwhelm with too many choices
|
||||||
|
- Group related options
|
||||||
|
- "Other" automatically provided
|
||||||
|
|
||||||
|
### Multi-Select Questions
|
||||||
|
|
||||||
|
**When to use multiSelect:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
Use AskUserQuestion for enabling features:
|
||||||
|
|
||||||
|
Question: "Which features do you want to enable?"
|
||||||
|
Header: "Features"
|
||||||
|
multiSelect: true // Allow selecting multiple
|
||||||
|
Options:
|
||||||
|
- Logging (Detailed operation logs)
|
||||||
|
- Metrics (Performance monitoring)
|
||||||
|
- Alerts (Error notifications)
|
||||||
|
- Backups (Automatic backups)
|
||||||
|
```
|
||||||
|
|
||||||
|
User can select any combination: none, some, or all.
|
||||||
|
|
||||||
|
**When NOT to use multiSelect:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
Question: "Which authentication method?"
|
||||||
|
multiSelect: false // Only one auth method makes sense
|
||||||
|
```
|
||||||
|
|
||||||
|
Mutually exclusive choices should not use multiSelect.
|
||||||
|
|
||||||
|
## Command Patterns with AskUserQuestion
|
||||||
|
|
||||||
|
### Pattern 1: Simple Yes/No Decision
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Command with confirmation
|
||||||
|
allowed-tools: AskUserQuestion, Bash
|
||||||
|
---
|
||||||
|
|
||||||
|
# Destructive Operation
|
||||||
|
|
||||||
|
This operation will delete all cached data.
|
||||||
|
|
||||||
|
Use AskUserQuestion to confirm:
|
||||||
|
|
||||||
|
Question: "This will delete all cached data. Are you sure?"
|
||||||
|
Header: "Confirm"
|
||||||
|
Options:
|
||||||
|
- Yes (Proceed with deletion)
|
||||||
|
- No (Cancel operation)
|
||||||
|
|
||||||
|
If user selects "Yes":
|
||||||
|
Execute deletion
|
||||||
|
Report completion
|
||||||
|
|
||||||
|
If user selects "No":
|
||||||
|
Cancel operation
|
||||||
|
Exit without changes
|
||||||
|
```
|
||||||
|
|
||||||
|
### Pattern 2: Multiple Configuration Questions
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Multi-question configuration
|
||||||
|
allowed-tools: AskUserQuestion, Write
|
||||||
|
---
|
||||||
|
|
||||||
|
# Project Configuration Setup
|
||||||
|
|
||||||
|
Gather configuration through multiple questions.
|
||||||
|
|
||||||
|
Use AskUserQuestion with multiple questions in one call:
|
||||||
|
|
||||||
|
**Question 1:**
|
||||||
|
- question: "Which programming language?"
|
||||||
|
- header: "Language"
|
||||||
|
- options: Python, TypeScript, Go, Rust
|
||||||
|
|
||||||
|
**Question 2:**
|
||||||
|
- question: "Which test framework?"
|
||||||
|
- header: "Testing"
|
||||||
|
- options: Jest, PyTest, Go Test, Cargo Test
|
||||||
|
(Adapt based on language from Q1)
|
||||||
|
|
||||||
|
**Question 3:**
|
||||||
|
- question: "Which CI/CD platform?"
|
||||||
|
- header: "CI/CD"
|
||||||
|
- options: GitHub Actions, GitLab CI, CircleCI
|
||||||
|
|
||||||
|
**Question 4:**
|
||||||
|
- question: "Which features do you need?"
|
||||||
|
- header: "Features"
|
||||||
|
- multiSelect: true
|
||||||
|
- options: Linting, Type checking, Code coverage, Security scanning
|
||||||
|
|
||||||
|
Process all answers together to generate cohesive configuration.
|
||||||
|
```
|
||||||
|
|
||||||
|
### Pattern 3: Conditional Question Flow
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Conditional interactive workflow
|
||||||
|
allowed-tools: AskUserQuestion, Read, Write
|
||||||
|
---
|
||||||
|
|
||||||
|
# Adaptive Configuration
|
||||||
|
|
||||||
|
## Question 1: Deployment Complexity
|
||||||
|
|
||||||
|
Use AskUserQuestion:
|
||||||
|
|
||||||
|
Question: "How complex is your deployment?"
|
||||||
|
Header: "Complexity"
|
||||||
|
Options:
|
||||||
|
- Simple (Single server, straightforward)
|
||||||
|
- Standard (Multiple servers, load balancing)
|
||||||
|
- Complex (Microservices, orchestration)
|
||||||
|
|
||||||
|
## Conditional Questions Based on Answer
|
||||||
|
|
||||||
|
If answer is "Simple":
|
||||||
|
- No additional questions
|
||||||
|
- Use minimal configuration
|
||||||
|
|
||||||
|
If answer is "Standard":
|
||||||
|
- Ask about load balancing strategy
|
||||||
|
- Ask about scaling policy
|
||||||
|
|
||||||
|
If answer is "Complex":
|
||||||
|
- Ask about orchestration platform (Kubernetes, Docker Swarm)
|
||||||
|
- Ask about service mesh (Istio, Linkerd, None)
|
||||||
|
- Ask about monitoring (Prometheus, Datadog, CloudWatch)
|
||||||
|
- Ask about logging aggregation
|
||||||
|
|
||||||
|
## Process Conditional Answers
|
||||||
|
|
||||||
|
Generate configuration appropriate for selected complexity level.
|
||||||
|
```
|
||||||
|
|
||||||
|
### Pattern 4: Iterative Collection
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Collect multiple items iteratively
|
||||||
|
allowed-tools: AskUserQuestion, Write
|
||||||
|
---
|
||||||
|
|
||||||
|
# Collect Team Members
|
||||||
|
|
||||||
|
We'll collect team member information for the project.
|
||||||
|
|
||||||
|
## Question: How many team members?
|
||||||
|
|
||||||
|
Use AskUserQuestion:
|
||||||
|
|
||||||
|
Question: "How many team members should we set up?"
|
||||||
|
Header: "Team size"
|
||||||
|
Options:
|
||||||
|
- 2 people
|
||||||
|
- 3 people
|
||||||
|
- 4 people
|
||||||
|
- 6 people
|
||||||
|
|
||||||
|
## Iterate Through Team Members
|
||||||
|
|
||||||
|
For each team member (1 to N based on answer):
|
||||||
|
|
||||||
|
Use AskUserQuestion for member details:
|
||||||
|
|
||||||
|
Question: "What role for team member [number]?"
|
||||||
|
Header: "Role"
|
||||||
|
Options:
|
||||||
|
- Frontend Developer
|
||||||
|
- Backend Developer
|
||||||
|
- DevOps Engineer
|
||||||
|
- QA Engineer
|
||||||
|
- Designer
|
||||||
|
|
||||||
|
Store each member's information.
|
||||||
|
|
||||||
|
## Generate Team Configuration
|
||||||
|
|
||||||
|
After collecting all N members, create team configuration file with all members and their roles.
|
||||||
|
```
|
||||||
|
|
||||||
|
### Pattern 5: Dependency Selection
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Select dependencies with multi-select
|
||||||
|
allowed-tools: AskUserQuestion
|
||||||
|
---
|
||||||
|
|
||||||
|
# Configure Project Dependencies
|
||||||
|
|
||||||
|
## Question: Required Libraries
|
||||||
|
|
||||||
|
Use AskUserQuestion with multiSelect:
|
||||||
|
|
||||||
|
Question: "Which libraries does your project need?"
|
||||||
|
Header: "Dependencies"
|
||||||
|
multiSelect: true
|
||||||
|
Options:
|
||||||
|
- React (UI framework)
|
||||||
|
- Express (Web server)
|
||||||
|
- TypeORM (Database ORM)
|
||||||
|
- Jest (Testing framework)
|
||||||
|
- Axios (HTTP client)
|
||||||
|
|
||||||
|
User can select any combination.
|
||||||
|
|
||||||
|
## Process Selections
|
||||||
|
|
||||||
|
For each selected library:
|
||||||
|
- Add to package.json dependencies
|
||||||
|
- Generate sample configuration
|
||||||
|
- Create usage examples
|
||||||
|
- Update documentation
|
||||||
|
```
|
||||||
|
|
||||||
|
## Best Practices for Interactive Commands
|
||||||
|
|
||||||
|
### Question Design
|
||||||
|
|
||||||
|
1. **Clear and specific**: Question should be unambiguous
|
||||||
|
2. **Concise header**: Max 12 characters for clean display
|
||||||
|
3. **Helpful options**: Labels are clear, descriptions explain trade-offs
|
||||||
|
4. **Appropriate count**: 2-4 options per question, 1-4 questions per call
|
||||||
|
5. **Logical order**: Questions flow naturally
|
||||||
|
|
||||||
|
### Error Handling
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
# Handle AskUserQuestion Responses
|
||||||
|
|
||||||
|
After calling AskUserQuestion, verify answers received:
|
||||||
|
|
||||||
|
If answers are empty or invalid:
|
||||||
|
Something went wrong gathering responses.
|
||||||
|
|
||||||
|
Please try again or provide configuration manually:
|
||||||
|
[Show alternative approach]
|
||||||
|
|
||||||
|
Exit.
|
||||||
|
|
||||||
|
If answers look correct:
|
||||||
|
Process as expected
|
||||||
|
```
|
||||||
|
|
||||||
|
### Progressive Disclosure
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
# Start Simple, Get Detailed as Needed
|
||||||
|
|
||||||
|
## Question 1: Setup Type
|
||||||
|
|
||||||
|
Use AskUserQuestion:
|
||||||
|
|
||||||
|
Question: "How would you like to set up?"
|
||||||
|
Header: "Setup type"
|
||||||
|
Options:
|
||||||
|
- Quick (Use recommended defaults)
|
||||||
|
- Custom (Configure all options)
|
||||||
|
- Guided (Step-by-step with explanations)
|
||||||
|
|
||||||
|
If "Quick":
|
||||||
|
Apply defaults, minimal questions
|
||||||
|
|
||||||
|
If "Custom":
|
||||||
|
Ask all available configuration questions
|
||||||
|
|
||||||
|
If "Guided":
|
||||||
|
Ask questions with extra explanation
|
||||||
|
Provide recommendations along the way
|
||||||
|
```
|
||||||
|
|
||||||
|
### Multi-Select Guidelines
|
||||||
|
|
||||||
|
**Good multi-select use:**
|
||||||
|
```markdown
|
||||||
|
Question: "Which features do you want to enable?"
|
||||||
|
multiSelect: true
|
||||||
|
Options:
|
||||||
|
- Logging
|
||||||
|
- Metrics
|
||||||
|
- Alerts
|
||||||
|
- Backups
|
||||||
|
|
||||||
|
Reason: User might want any combination
|
||||||
|
```
|
||||||
|
|
||||||
|
**Bad multi-select use:**
|
||||||
|
```markdown
|
||||||
|
Question: "Which database engine?"
|
||||||
|
multiSelect: true // ❌ Should be single-select
|
||||||
|
|
||||||
|
Reason: Can only use one database engine
|
||||||
|
```
|
||||||
|
|
||||||
|
## Advanced Patterns
|
||||||
|
|
||||||
|
### Validation Loop
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Interactive with validation
|
||||||
|
allowed-tools: AskUserQuestion, Bash
|
||||||
|
---
|
||||||
|
|
||||||
|
# Setup with Validation
|
||||||
|
|
||||||
|
## Gather Configuration
|
||||||
|
|
||||||
|
Use AskUserQuestion to collect settings.
|
||||||
|
|
||||||
|
## Validate Configuration
|
||||||
|
|
||||||
|
Check if configuration is valid:
|
||||||
|
- Required dependencies available?
|
||||||
|
- Settings compatible with each other?
|
||||||
|
- No conflicts detected?
|
||||||
|
|
||||||
|
If validation fails:
|
||||||
|
Show validation errors
|
||||||
|
|
||||||
|
Use AskUserQuestion to ask:
|
||||||
|
|
||||||
|
Question: "Configuration has issues. What would you like to do?"
|
||||||
|
Header: "Next step"
|
||||||
|
Options:
|
||||||
|
- Fix (Adjust settings to resolve issues)
|
||||||
|
- Override (Proceed despite warnings)
|
||||||
|
- Cancel (Abort setup)
|
||||||
|
|
||||||
|
Based on answer, retry or proceed or exit.
|
||||||
|
```
|
||||||
|
|
||||||
|
### Build Configuration Incrementally
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Incremental configuration builder
|
||||||
|
allowed-tools: AskUserQuestion, Write, Read
|
||||||
|
---
|
||||||
|
|
||||||
|
# Incremental Setup
|
||||||
|
|
||||||
|
## Phase 1: Core Settings
|
||||||
|
|
||||||
|
Use AskUserQuestion for core settings.
|
||||||
|
|
||||||
|
Save to `.claude/config-partial.yml`
|
||||||
|
|
||||||
|
## Phase 2: Review Core Settings
|
||||||
|
|
||||||
|
Show user the core settings:
|
||||||
|
|
||||||
|
Based on these core settings, you need to configure:
|
||||||
|
- [Setting A] (because you chose [X])
|
||||||
|
- [Setting B] (because you chose [Y])
|
||||||
|
|
||||||
|
Ready to continue?
|
||||||
|
|
||||||
|
## Phase 3: Detailed Settings
|
||||||
|
|
||||||
|
Use AskUserQuestion for settings based on Phase 1 answers.
|
||||||
|
|
||||||
|
Merge with core settings.
|
||||||
|
|
||||||
|
## Phase 4: Final Review
|
||||||
|
|
||||||
|
Present complete configuration.
|
||||||
|
|
||||||
|
Use AskUserQuestion for confirmation:
|
||||||
|
|
||||||
|
Question: "Is this configuration correct?"
|
||||||
|
Options:
|
||||||
|
- Yes (Save and apply)
|
||||||
|
- No (Start over)
|
||||||
|
- Modify (Edit specific settings)
|
||||||
|
```
|
||||||
|
|
||||||
|
### Dynamic Options Based on Context
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Context-aware questions
|
||||||
|
allowed-tools: AskUserQuestion, Bash, Read
|
||||||
|
---
|
||||||
|
|
||||||
|
# Context-Aware Setup
|
||||||
|
|
||||||
|
## Detect Current State
|
||||||
|
|
||||||
|
Check existing configuration:
|
||||||
|
- Current language: !`detect-language.sh`
|
||||||
|
- Existing frameworks: !`detect-frameworks.sh`
|
||||||
|
- Available tools: !`check-tools.sh`
|
||||||
|
|
||||||
|
## Ask Context-Appropriate Questions
|
||||||
|
|
||||||
|
Based on detected language, ask relevant questions.
|
||||||
|
|
||||||
|
If language is TypeScript:
|
||||||
|
|
||||||
|
Use AskUserQuestion:
|
||||||
|
|
||||||
|
Question: "Which TypeScript features should we enable?"
|
||||||
|
Options:
|
||||||
|
- Strict Mode (Maximum type safety)
|
||||||
|
- Decorators (Experimental decorator support)
|
||||||
|
- Path Mapping (Module path aliases)
|
||||||
|
|
||||||
|
If language is Python:
|
||||||
|
|
||||||
|
Use AskUserQuestion:
|
||||||
|
|
||||||
|
Question: "Which Python tools should we configure?"
|
||||||
|
Options:
|
||||||
|
- Type Hints (mypy for type checking)
|
||||||
|
- Black (Code formatting)
|
||||||
|
- Pylint (Linting and style)
|
||||||
|
|
||||||
|
Questions adapt to project context.
|
||||||
|
```
|
||||||
|
|
||||||
|
## Real-World Example: Multi-Agent Swarm Launch
|
||||||
|
|
||||||
|
**From multi-agent-swarm plugin:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Launch multi-agent swarm
|
||||||
|
allowed-tools: AskUserQuestion, Read, Write, Bash
|
||||||
|
---
|
||||||
|
|
||||||
|
# Launch Multi-Agent Swarm
|
||||||
|
|
||||||
|
## Interactive Mode (No Task List Provided)
|
||||||
|
|
||||||
|
If user didn't provide task list file, help create one interactively.
|
||||||
|
|
||||||
|
### Question 1: Agent Count
|
||||||
|
|
||||||
|
Use AskUserQuestion:
|
||||||
|
|
||||||
|
Question: "How many agents should we launch?"
|
||||||
|
Header: "Agent count"
|
||||||
|
Options:
|
||||||
|
- 2 agents (Best for simple projects)
|
||||||
|
- 3 agents (Good for medium projects)
|
||||||
|
- 4 agents (Standard team size)
|
||||||
|
- 6 agents (Large projects)
|
||||||
|
- 8 agents (Complex multi-component projects)
|
||||||
|
|
||||||
|
### Question 2: Task Definition Approach
|
||||||
|
|
||||||
|
Use AskUserQuestion:
|
||||||
|
|
||||||
|
Question: "How would you like to define tasks?"
|
||||||
|
Header: "Task setup"
|
||||||
|
Options:
|
||||||
|
- File (I have a task list file ready)
|
||||||
|
- Guided (Help me create tasks interactively)
|
||||||
|
- Custom (Other approach)
|
||||||
|
|
||||||
|
If "File":
|
||||||
|
Ask for file path
|
||||||
|
Validate file exists and has correct format
|
||||||
|
|
||||||
|
If "Guided":
|
||||||
|
Enter iterative task creation mode (see below)
|
||||||
|
|
||||||
|
### Question 3: Coordination Mode
|
||||||
|
|
||||||
|
Use AskUserQuestion:
|
||||||
|
|
||||||
|
Question: "How should agents coordinate?"
|
||||||
|
Header: "Coordination"
|
||||||
|
Options:
|
||||||
|
- Team Leader (One agent coordinates others)
|
||||||
|
- Collaborative (Agents coordinate as peers)
|
||||||
|
- Autonomous (Independent work, minimal coordination)
|
||||||
|
|
||||||
|
### Iterative Task Creation (If "Guided" Selected)
|
||||||
|
|
||||||
|
For each agent (1 to N from Question 1):
|
||||||
|
|
||||||
|
**Question A: Agent Name**
|
||||||
|
Question: "What should we call agent [number]?"
|
||||||
|
Header: "Agent name"
|
||||||
|
Options:
|
||||||
|
- auth-agent
|
||||||
|
- api-agent
|
||||||
|
- ui-agent
|
||||||
|
- db-agent
|
||||||
|
(Provide relevant suggestions based on common patterns)
|
||||||
|
|
||||||
|
**Question B: Task Type**
|
||||||
|
Question: "What task for [agent-name]?"
|
||||||
|
Header: "Task type"
|
||||||
|
Options:
|
||||||
|
- Authentication (User auth, JWT, OAuth)
|
||||||
|
- API Endpoints (REST/GraphQL APIs)
|
||||||
|
- UI Components (Frontend components)
|
||||||
|
- Database (Schema, migrations, queries)
|
||||||
|
- Testing (Test suites and coverage)
|
||||||
|
- Documentation (Docs, README, guides)
|
||||||
|
|
||||||
|
**Question C: Dependencies**
|
||||||
|
Question: "What does [agent-name] depend on?"
|
||||||
|
Header: "Dependencies"
|
||||||
|
multiSelect: true
|
||||||
|
Options:
|
||||||
|
- [List of previously defined agents]
|
||||||
|
- No dependencies
|
||||||
|
|
||||||
|
**Question D: Base Branch**
|
||||||
|
Question: "Which base branch for PR?"
|
||||||
|
Header: "PR base"
|
||||||
|
Options:
|
||||||
|
- main
|
||||||
|
- staging
|
||||||
|
- develop
|
||||||
|
|
||||||
|
Store all task information for each agent.
|
||||||
|
|
||||||
|
### Generate Task List File
|
||||||
|
|
||||||
|
After collecting all agent task details:
|
||||||
|
|
||||||
|
1. Ask for project name
|
||||||
|
2. Generate task list in proper format
|
||||||
|
3. Save to `.daisy/swarm/tasks.md`
|
||||||
|
4. Show user the file path
|
||||||
|
5. Proceed with launch using generated task list
|
||||||
|
```
|
||||||
|
|
||||||
|
## Best Practices
|
||||||
|
|
||||||
|
### Question Writing
|
||||||
|
|
||||||
|
1. **Be specific**: "Which database?" not "Choose option?"
|
||||||
|
2. **Explain trade-offs**: Describe pros/cons in option descriptions
|
||||||
|
3. **Provide context**: Question text should stand alone
|
||||||
|
4. **Guide decisions**: Help user make informed choice
|
||||||
|
5. **Keep concise**: Header max 12 chars, descriptions 1-2 sentences
|
||||||
|
|
||||||
|
### Option Design
|
||||||
|
|
||||||
|
1. **Meaningful labels**: Specific, clear names
|
||||||
|
2. **Informative descriptions**: Explain what each option does
|
||||||
|
3. **Show trade-offs**: Help user understand implications
|
||||||
|
4. **Consistent detail**: All options equally explained
|
||||||
|
5. **2-4 options**: Not too few, not too many
|
||||||
|
|
||||||
|
### Flow Design
|
||||||
|
|
||||||
|
1. **Logical order**: Questions flow naturally
|
||||||
|
2. **Build on previous**: Later questions use earlier answers
|
||||||
|
3. **Minimize questions**: Ask only what's needed
|
||||||
|
4. **Group related**: Ask related questions together
|
||||||
|
5. **Show progress**: Indicate where in flow
|
||||||
|
|
||||||
|
### User Experience
|
||||||
|
|
||||||
|
1. **Set expectations**: Tell user what to expect
|
||||||
|
2. **Explain why**: Help user understand purpose
|
||||||
|
3. **Provide defaults**: Suggest recommended options
|
||||||
|
4. **Allow escape**: Let user cancel or restart
|
||||||
|
5. **Confirm actions**: Summarize before executing
|
||||||
|
|
||||||
|
## Common Patterns
|
||||||
|
|
||||||
|
### Pattern: Feature Selection
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
Use AskUserQuestion:
|
||||||
|
|
||||||
|
Question: "Which features do you need?"
|
||||||
|
Header: "Features"
|
||||||
|
multiSelect: true
|
||||||
|
Options:
|
||||||
|
- Authentication
|
||||||
|
- Authorization
|
||||||
|
- Rate Limiting
|
||||||
|
- Caching
|
||||||
|
```
|
||||||
|
|
||||||
|
### Pattern: Environment Configuration
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
Use AskUserQuestion:
|
||||||
|
|
||||||
|
Question: "Which environment is this?"
|
||||||
|
Header: "Environment"
|
||||||
|
Options:
|
||||||
|
- Development (Local development)
|
||||||
|
- Staging (Pre-production testing)
|
||||||
|
- Production (Live environment)
|
||||||
|
```
|
||||||
|
|
||||||
|
### Pattern: Priority Selection
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
Use AskUserQuestion:
|
||||||
|
|
||||||
|
Question: "What's the priority for this task?"
|
||||||
|
Header: "Priority"
|
||||||
|
Options:
|
||||||
|
- Critical (Must be done immediately)
|
||||||
|
- High (Important, do soon)
|
||||||
|
- Medium (Standard priority)
|
||||||
|
- Low (Nice to have)
|
||||||
|
```
|
||||||
|
|
||||||
|
### Pattern: Scope Selection
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
Use AskUserQuestion:
|
||||||
|
|
||||||
|
Question: "What scope should we analyze?"
|
||||||
|
Header: "Scope"
|
||||||
|
Options:
|
||||||
|
- Current file (Just this file)
|
||||||
|
- Current directory (All files in directory)
|
||||||
|
- Entire project (Full codebase scan)
|
||||||
|
```
|
||||||
|
|
||||||
|
## Combining Arguments and Questions
|
||||||
|
|
||||||
|
### Use Both Appropriately
|
||||||
|
|
||||||
|
**Arguments for known values:**
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
argument-hint: [project-name]
|
||||||
|
allowed-tools: AskUserQuestion, Write
|
||||||
|
---
|
||||||
|
|
||||||
|
Setup for project: $1
|
||||||
|
|
||||||
|
Now gather additional configuration...
|
||||||
|
|
||||||
|
Use AskUserQuestion for options that require explanation.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Questions for complex choices:**
|
||||||
|
```markdown
|
||||||
|
Project name from argument: $1
|
||||||
|
|
||||||
|
Now use AskUserQuestion to choose:
|
||||||
|
- Architecture pattern
|
||||||
|
- Technology stack
|
||||||
|
- Deployment strategy
|
||||||
|
|
||||||
|
These require explanation, so questions work better than arguments.
|
||||||
|
```
|
||||||
|
|
||||||
|
## Troubleshooting
|
||||||
|
|
||||||
|
**Questions not appearing:**
|
||||||
|
- Verify AskUserQuestion in allowed-tools
|
||||||
|
- Check question format is correct
|
||||||
|
- Ensure options array has 2-4 items
|
||||||
|
|
||||||
|
**User can't make selection:**
|
||||||
|
- Check option labels are clear
|
||||||
|
- Verify descriptions are helpful
|
||||||
|
- Consider if too many options
|
||||||
|
- Ensure multiSelect setting is correct
|
||||||
|
|
||||||
|
**Flow feels confusing:**
|
||||||
|
- Reduce number of questions
|
||||||
|
- Group related questions
|
||||||
|
- Add explanation between stages
|
||||||
|
- Show progress through workflow
|
||||||
|
|
||||||
|
With AskUserQuestion, commands become interactive wizards that guide users through complex decisions while maintaining the clarity that simple arguments provide for straightforward inputs.
|
||||||
@@ -0,0 +1,904 @@
|
|||||||
|
# Marketplace Considerations for Commands
|
||||||
|
|
||||||
|
Guidelines for creating commands designed for distribution and marketplace success.
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
|
||||||
|
Commands distributed through marketplaces need additional consideration beyond personal use commands. They must work across environments, handle diverse use cases, and provide excellent user experience for unknown users.
|
||||||
|
|
||||||
|
## Design for Distribution
|
||||||
|
|
||||||
|
### Universal Compatibility
|
||||||
|
|
||||||
|
**Cross-platform considerations:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Cross-platform command
|
||||||
|
allowed-tools: Bash(*)
|
||||||
|
---
|
||||||
|
|
||||||
|
# Platform-Aware Command
|
||||||
|
|
||||||
|
Detecting platform...
|
||||||
|
|
||||||
|
case "$(uname)" in
|
||||||
|
Darwin*) PLATFORM="macOS" ;;
|
||||||
|
Linux*) PLATFORM="Linux" ;;
|
||||||
|
MINGW*|MSYS*|CYGWIN*) PLATFORM="Windows" ;;
|
||||||
|
*) PLATFORM="Unknown" ;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
Platform: $PLATFORM
|
||||||
|
|
||||||
|
<!-- Adjust behavior based on platform -->
|
||||||
|
if [ "$PLATFORM" = "Windows" ]; then
|
||||||
|
# Windows-specific handling
|
||||||
|
PATH_SEP="\\"
|
||||||
|
NULL_DEVICE="NUL"
|
||||||
|
else
|
||||||
|
# Unix-like handling
|
||||||
|
PATH_SEP="/"
|
||||||
|
NULL_DEVICE="/dev/null"
|
||||||
|
fi
|
||||||
|
|
||||||
|
[Platform-appropriate implementation...]
|
||||||
|
```
|
||||||
|
|
||||||
|
**Avoid platform-specific commands:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
<!-- BAD: macOS-specific -->
|
||||||
|
!`pbcopy < file.txt`
|
||||||
|
|
||||||
|
<!-- GOOD: Platform detection -->
|
||||||
|
if command -v pbcopy > /dev/null; then
|
||||||
|
pbcopy < file.txt
|
||||||
|
elif command -v xclip > /dev/null; then
|
||||||
|
xclip -selection clipboard < file.txt
|
||||||
|
elif command -v clip.exe > /dev/null; then
|
||||||
|
cat file.txt | clip.exe
|
||||||
|
else
|
||||||
|
echo "Clipboard not available on this platform"
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
### Minimal Dependencies
|
||||||
|
|
||||||
|
**Check for required tools:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Dependency-aware command
|
||||||
|
allowed-tools: Bash(*)
|
||||||
|
---
|
||||||
|
|
||||||
|
# Check Dependencies
|
||||||
|
|
||||||
|
Required tools:
|
||||||
|
- git
|
||||||
|
- jq
|
||||||
|
- node
|
||||||
|
|
||||||
|
Checking availability...
|
||||||
|
|
||||||
|
MISSING_DEPS=""
|
||||||
|
|
||||||
|
for tool in git jq node; do
|
||||||
|
if ! command -v $tool > /dev/null; then
|
||||||
|
MISSING_DEPS="$MISSING_DEPS $tool"
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
|
||||||
|
if [ -n "$MISSING_DEPS" ]; then
|
||||||
|
❌ ERROR: Missing required dependencies:$MISSING_DEPS
|
||||||
|
|
||||||
|
INSTALLATION:
|
||||||
|
- git: https://git-scm.com/downloads
|
||||||
|
- jq: https://stedolan.github.io/jq/download/
|
||||||
|
- node: https://nodejs.org/
|
||||||
|
|
||||||
|
Install missing tools and try again.
|
||||||
|
|
||||||
|
Exit.
|
||||||
|
fi
|
||||||
|
|
||||||
|
✓ All dependencies available
|
||||||
|
|
||||||
|
[Continue with command...]
|
||||||
|
```
|
||||||
|
|
||||||
|
**Document optional dependencies:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
<!--
|
||||||
|
DEPENDENCIES:
|
||||||
|
Required:
|
||||||
|
- git 2.0+: Version control
|
||||||
|
- jq 1.6+: JSON processing
|
||||||
|
|
||||||
|
Optional:
|
||||||
|
- gh: GitHub CLI (for PR operations)
|
||||||
|
- docker: Container operations (for containerized tests)
|
||||||
|
|
||||||
|
Feature availability depends on installed tools.
|
||||||
|
-->
|
||||||
|
```
|
||||||
|
|
||||||
|
### Graceful Degradation
|
||||||
|
|
||||||
|
**Handle missing features:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Feature-aware command
|
||||||
|
---
|
||||||
|
|
||||||
|
# Feature Detection
|
||||||
|
|
||||||
|
Detecting available features...
|
||||||
|
|
||||||
|
FEATURES=""
|
||||||
|
|
||||||
|
if command -v gh > /dev/null; then
|
||||||
|
FEATURES="$FEATURES github"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if command -v docker > /dev/null; then
|
||||||
|
FEATURES="$FEATURES docker"
|
||||||
|
fi
|
||||||
|
|
||||||
|
Available features: $FEATURES
|
||||||
|
|
||||||
|
if echo "$FEATURES" | grep -q "github"; then
|
||||||
|
# Full functionality with GitHub integration
|
||||||
|
echo "✓ GitHub integration available"
|
||||||
|
else
|
||||||
|
# Reduced functionality without GitHub
|
||||||
|
echo "⚠ Limited functionality: GitHub CLI not installed"
|
||||||
|
echo " Install 'gh' for full features"
|
||||||
|
fi
|
||||||
|
|
||||||
|
[Adapt behavior based on available features...]
|
||||||
|
```
|
||||||
|
|
||||||
|
## User Experience for Unknown Users
|
||||||
|
|
||||||
|
### Clear Onboarding
|
||||||
|
|
||||||
|
**First-run experience:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Command with onboarding
|
||||||
|
allowed-tools: Read, Write
|
||||||
|
---
|
||||||
|
|
||||||
|
# First Run Check
|
||||||
|
|
||||||
|
if [ ! -f ".claude/command-initialized" ]; then
|
||||||
|
**Welcome to Command Name!**
|
||||||
|
|
||||||
|
This appears to be your first time using this command.
|
||||||
|
|
||||||
|
WHAT THIS COMMAND DOES:
|
||||||
|
[Brief explanation of purpose and benefits]
|
||||||
|
|
||||||
|
QUICK START:
|
||||||
|
1. Basic usage: /command [arg]
|
||||||
|
2. For help: /command help
|
||||||
|
3. Examples: /command examples
|
||||||
|
|
||||||
|
SETUP:
|
||||||
|
No additional setup required. You're ready to go!
|
||||||
|
|
||||||
|
✓ Initialization complete
|
||||||
|
|
||||||
|
[Create initialization marker]
|
||||||
|
|
||||||
|
Ready to proceed with your request...
|
||||||
|
fi
|
||||||
|
|
||||||
|
[Normal command execution...]
|
||||||
|
```
|
||||||
|
|
||||||
|
**Progressive feature discovery:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Command with tips
|
||||||
|
---
|
||||||
|
|
||||||
|
# Command Execution
|
||||||
|
|
||||||
|
[Main functionality...]
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
💡 TIP: Did you know?
|
||||||
|
|
||||||
|
You can speed up this command with the --fast flag:
|
||||||
|
/command --fast [args]
|
||||||
|
|
||||||
|
For more tips: /command tips
|
||||||
|
```
|
||||||
|
|
||||||
|
### Comprehensive Error Handling
|
||||||
|
|
||||||
|
**Anticipate user mistakes:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Forgiving command
|
||||||
|
---
|
||||||
|
|
||||||
|
# User Input Handling
|
||||||
|
|
||||||
|
Argument: "$1"
|
||||||
|
|
||||||
|
<!-- Check for common typos -->
|
||||||
|
if [ "$1" = "hlep" ] || [ "$1" = "hepl" ]; then
|
||||||
|
Did you mean: help?
|
||||||
|
|
||||||
|
Showing help instead...
|
||||||
|
[Display help]
|
||||||
|
|
||||||
|
Exit.
|
||||||
|
fi
|
||||||
|
|
||||||
|
<!-- Suggest similar commands if not found -->
|
||||||
|
if [ "$1" != "valid-option1" ] && [ "$1" != "valid-option2" ]; then
|
||||||
|
❌ Unknown option: $1
|
||||||
|
|
||||||
|
Did you mean:
|
||||||
|
- valid-option1 (most similar)
|
||||||
|
- valid-option2
|
||||||
|
|
||||||
|
For all options: /command help
|
||||||
|
|
||||||
|
Exit.
|
||||||
|
fi
|
||||||
|
|
||||||
|
[Command continues...]
|
||||||
|
```
|
||||||
|
|
||||||
|
**Helpful diagnostics:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Diagnostic command
|
||||||
|
---
|
||||||
|
|
||||||
|
# Operation Failed
|
||||||
|
|
||||||
|
The operation could not complete.
|
||||||
|
|
||||||
|
**Diagnostic Information:**
|
||||||
|
|
||||||
|
Environment:
|
||||||
|
- Platform: $(uname)
|
||||||
|
- Shell: $SHELL
|
||||||
|
- Working directory: $(pwd)
|
||||||
|
- Command: /command $@
|
||||||
|
|
||||||
|
Checking common issues:
|
||||||
|
- Git repository: $(git rev-parse --git-dir 2>&1)
|
||||||
|
- Write permissions: $(test -w . && echo "OK" || echo "DENIED")
|
||||||
|
- Required files: $(test -f config.yml && echo "Found" || echo "Missing")
|
||||||
|
|
||||||
|
This information helps debug the issue.
|
||||||
|
|
||||||
|
For support, include the above diagnostics.
|
||||||
|
```
|
||||||
|
|
||||||
|
## Distribution Best Practices
|
||||||
|
|
||||||
|
### Namespace Awareness
|
||||||
|
|
||||||
|
**Avoid name collisions:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Namespaced command
|
||||||
|
---
|
||||||
|
|
||||||
|
<!--
|
||||||
|
COMMAND NAME: plugin-name-command
|
||||||
|
|
||||||
|
This command is namespaced with the plugin name to avoid
|
||||||
|
conflicts with commands from other plugins.
|
||||||
|
|
||||||
|
Alternative naming approaches:
|
||||||
|
- Use plugin prefix: /plugin-command
|
||||||
|
- Use category: /category-command
|
||||||
|
- Use verb-noun: /verb-noun
|
||||||
|
|
||||||
|
Chosen approach: plugin-name prefix
|
||||||
|
Reasoning: Clearest ownership, least likely to conflict
|
||||||
|
-->
|
||||||
|
|
||||||
|
# Plugin Name Command
|
||||||
|
|
||||||
|
[Implementation...]
|
||||||
|
```
|
||||||
|
|
||||||
|
**Document naming rationale:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
<!--
|
||||||
|
NAMING DECISION:
|
||||||
|
|
||||||
|
Command name: /deploy-app
|
||||||
|
|
||||||
|
Alternatives considered:
|
||||||
|
- /deploy: Too generic, likely conflicts
|
||||||
|
- /app-deploy: Less intuitive ordering
|
||||||
|
- /my-plugin-deploy: Too verbose
|
||||||
|
|
||||||
|
Final choice balances:
|
||||||
|
- Discoverability (clear purpose)
|
||||||
|
- Brevity (easy to type)
|
||||||
|
- Uniqueness (unlikely conflicts)
|
||||||
|
-->
|
||||||
|
```
|
||||||
|
|
||||||
|
### Configurability
|
||||||
|
|
||||||
|
**User preferences:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Configurable command
|
||||||
|
allowed-tools: Read
|
||||||
|
---
|
||||||
|
|
||||||
|
# Load User Configuration
|
||||||
|
|
||||||
|
Default configuration:
|
||||||
|
- verbose: false
|
||||||
|
- color: true
|
||||||
|
- max_results: 10
|
||||||
|
|
||||||
|
Checking for user config: .claude/plugin-name.local.md
|
||||||
|
|
||||||
|
if [ -f ".claude/plugin-name.local.md" ]; then
|
||||||
|
# Parse YAML frontmatter for settings
|
||||||
|
VERBOSE=$(grep "^verbose:" .claude/plugin-name.local.md | cut -d: -f2 | tr -d ' ')
|
||||||
|
COLOR=$(grep "^color:" .claude/plugin-name.local.md | cut -d: -f2 | tr -d ' ')
|
||||||
|
MAX_RESULTS=$(grep "^max_results:" .claude/plugin-name.local.md | cut -d: -f2 | tr -d ' ')
|
||||||
|
|
||||||
|
echo "✓ Using user configuration"
|
||||||
|
else
|
||||||
|
echo "Using default configuration"
|
||||||
|
echo "Create .claude/plugin-name.local.md to customize"
|
||||||
|
fi
|
||||||
|
|
||||||
|
[Use configuration in command...]
|
||||||
|
```
|
||||||
|
|
||||||
|
**Sensible defaults:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Command with smart defaults
|
||||||
|
---
|
||||||
|
|
||||||
|
# Smart Defaults
|
||||||
|
|
||||||
|
Configuration:
|
||||||
|
- Format: ${FORMAT:-json} # Defaults to json
|
||||||
|
- Output: ${OUTPUT:-stdout} # Defaults to stdout
|
||||||
|
- Verbose: ${VERBOSE:-false} # Defaults to false
|
||||||
|
|
||||||
|
These defaults work for 80% of use cases.
|
||||||
|
|
||||||
|
Override with arguments:
|
||||||
|
/command --format yaml --output file.txt --verbose
|
||||||
|
|
||||||
|
Or set in .claude/plugin-name.local.md:
|
||||||
|
\`\`\`yaml
|
||||||
|
---
|
||||||
|
format: yaml
|
||||||
|
output: custom.txt
|
||||||
|
verbose: true
|
||||||
|
---
|
||||||
|
\`\`\`
|
||||||
|
```
|
||||||
|
|
||||||
|
### Version Compatibility
|
||||||
|
|
||||||
|
**Version checking:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Version-aware command
|
||||||
|
---
|
||||||
|
|
||||||
|
<!--
|
||||||
|
COMMAND VERSION: 2.1.0
|
||||||
|
|
||||||
|
COMPATIBILITY:
|
||||||
|
- Requires plugin version: >= 2.0.0
|
||||||
|
- Breaking changes from v1.x documented in MIGRATION.md
|
||||||
|
|
||||||
|
VERSION HISTORY:
|
||||||
|
- v2.1.0: Added --new-feature flag
|
||||||
|
- v2.0.0: BREAKING: Changed argument order
|
||||||
|
- v1.0.0: Initial release
|
||||||
|
-->
|
||||||
|
|
||||||
|
# Version Check
|
||||||
|
|
||||||
|
Command version: 2.1.0
|
||||||
|
Plugin version: [detect from plugin.json]
|
||||||
|
|
||||||
|
if [ "$PLUGIN_VERSION" < "2.0.0" ]; then
|
||||||
|
❌ ERROR: Incompatible plugin version
|
||||||
|
|
||||||
|
This command requires plugin version >= 2.0.0
|
||||||
|
Current version: $PLUGIN_VERSION
|
||||||
|
|
||||||
|
Update plugin:
|
||||||
|
/plugin update plugin-name
|
||||||
|
|
||||||
|
Exit.
|
||||||
|
fi
|
||||||
|
|
||||||
|
✓ Version compatible
|
||||||
|
|
||||||
|
[Command continues...]
|
||||||
|
```
|
||||||
|
|
||||||
|
**Deprecation warnings:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Command with deprecation warnings
|
||||||
|
---
|
||||||
|
|
||||||
|
# Deprecation Check
|
||||||
|
|
||||||
|
if [ "$1" = "--old-flag" ]; then
|
||||||
|
⚠️ DEPRECATION WARNING
|
||||||
|
|
||||||
|
The --old-flag option is deprecated as of v2.0.0
|
||||||
|
It will be removed in v3.0.0 (est. June 2025)
|
||||||
|
|
||||||
|
Use instead: --new-flag
|
||||||
|
|
||||||
|
Example:
|
||||||
|
Old: /command --old-flag value
|
||||||
|
New: /command --new-flag value
|
||||||
|
|
||||||
|
See migration guide: /command migrate
|
||||||
|
|
||||||
|
Continuing with deprecated behavior for now...
|
||||||
|
fi
|
||||||
|
|
||||||
|
[Handle both old and new flags during deprecation period...]
|
||||||
|
```
|
||||||
|
|
||||||
|
## Marketplace Presentation
|
||||||
|
|
||||||
|
### Command Discovery
|
||||||
|
|
||||||
|
**Descriptive naming:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Review pull request with security and quality checks
|
||||||
|
---
|
||||||
|
|
||||||
|
<!-- GOOD: Descriptive name and description -->
|
||||||
|
```
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Do the thing
|
||||||
|
---
|
||||||
|
|
||||||
|
<!-- BAD: Vague description -->
|
||||||
|
```
|
||||||
|
|
||||||
|
**Searchable keywords:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
<!--
|
||||||
|
KEYWORDS: security, code-review, quality, validation, audit
|
||||||
|
|
||||||
|
These keywords help users discover this command when searching
|
||||||
|
for related functionality in the marketplace.
|
||||||
|
-->
|
||||||
|
```
|
||||||
|
|
||||||
|
### Showcase Examples
|
||||||
|
|
||||||
|
**Compelling demonstrations:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Advanced code analysis command
|
||||||
|
---
|
||||||
|
|
||||||
|
# Code Analysis Command
|
||||||
|
|
||||||
|
This command performs deep code analysis with actionable insights.
|
||||||
|
|
||||||
|
## Demo: Quick Security Audit
|
||||||
|
|
||||||
|
Try it now:
|
||||||
|
\`\`\`
|
||||||
|
/analyze-code src/ --security
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
**What you'll get:**
|
||||||
|
- Security vulnerability detection
|
||||||
|
- Code quality metrics
|
||||||
|
- Performance bottleneck identification
|
||||||
|
- Actionable recommendations
|
||||||
|
|
||||||
|
**Sample output:**
|
||||||
|
\`\`\`
|
||||||
|
Security Analysis Results
|
||||||
|
=========================
|
||||||
|
|
||||||
|
🔴 Critical (2):
|
||||||
|
- SQL injection risk in users.js:45
|
||||||
|
- XSS vulnerability in display.js:23
|
||||||
|
|
||||||
|
🟡 Warnings (5):
|
||||||
|
- Unvalidated input in api.js:67
|
||||||
|
...
|
||||||
|
|
||||||
|
Recommendations:
|
||||||
|
1. Fix critical issues immediately
|
||||||
|
2. Review warnings before next release
|
||||||
|
3. Run /analyze-code --fix for auto-fixes
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
Ready to analyze your code...
|
||||||
|
|
||||||
|
[Command implementation...]
|
||||||
|
```
|
||||||
|
|
||||||
|
### User Reviews and Feedback
|
||||||
|
|
||||||
|
**Feedback mechanism:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Command with feedback
|
||||||
|
---
|
||||||
|
|
||||||
|
# Command Complete
|
||||||
|
|
||||||
|
[Command results...]
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**How was your experience?**
|
||||||
|
|
||||||
|
This helps improve the command for everyone.
|
||||||
|
|
||||||
|
Rate this command:
|
||||||
|
- 👍 Helpful
|
||||||
|
- 👎 Not helpful
|
||||||
|
- 🐛 Found a bug
|
||||||
|
- 💡 Have a suggestion
|
||||||
|
|
||||||
|
Reply with an emoji or:
|
||||||
|
- /command feedback
|
||||||
|
|
||||||
|
Your feedback matters!
|
||||||
|
```
|
||||||
|
|
||||||
|
**Usage analytics preparation:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
<!--
|
||||||
|
ANALYTICS NOTES:
|
||||||
|
|
||||||
|
Track for improvement:
|
||||||
|
- Most common arguments
|
||||||
|
- Failure rates
|
||||||
|
- Average execution time
|
||||||
|
- User satisfaction scores
|
||||||
|
|
||||||
|
Privacy-preserving:
|
||||||
|
- No personally identifiable information
|
||||||
|
- Aggregate statistics only
|
||||||
|
- User opt-out respected
|
||||||
|
-->
|
||||||
|
```
|
||||||
|
|
||||||
|
## Quality Standards
|
||||||
|
|
||||||
|
### Professional Polish
|
||||||
|
|
||||||
|
**Consistent branding:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Branded command
|
||||||
|
---
|
||||||
|
|
||||||
|
# ✨ Command Name
|
||||||
|
|
||||||
|
Part of the [Plugin Name] suite
|
||||||
|
|
||||||
|
[Command functionality...]
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Need Help?**
|
||||||
|
- Documentation: https://docs.example.com
|
||||||
|
- Support: support@example.com
|
||||||
|
- Community: https://community.example.com
|
||||||
|
|
||||||
|
Powered by Plugin Name v2.1.0
|
||||||
|
```
|
||||||
|
|
||||||
|
**Attention to detail:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
<!-- Details that matter -->
|
||||||
|
|
||||||
|
✓ Use proper emoji/symbols consistently
|
||||||
|
✓ Align output columns neatly
|
||||||
|
✓ Format numbers with thousands separators
|
||||||
|
✓ Use color/formatting appropriately
|
||||||
|
✓ Provide progress indicators
|
||||||
|
✓ Show estimated time remaining
|
||||||
|
✓ Confirm successful operations
|
||||||
|
```
|
||||||
|
|
||||||
|
### Reliability
|
||||||
|
|
||||||
|
**Idempotency:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Idempotent command
|
||||||
|
---
|
||||||
|
|
||||||
|
# Safe Repeated Execution
|
||||||
|
|
||||||
|
Checking if operation already completed...
|
||||||
|
|
||||||
|
if [ -f ".claude/operation-completed.flag" ]; then
|
||||||
|
ℹ️ Operation already completed
|
||||||
|
|
||||||
|
Completed at: $(cat .claude/operation-completed.flag)
|
||||||
|
|
||||||
|
To re-run:
|
||||||
|
1. Remove flag: rm .claude/operation-completed.flag
|
||||||
|
2. Run command again
|
||||||
|
|
||||||
|
Otherwise, no action needed.
|
||||||
|
|
||||||
|
Exit.
|
||||||
|
fi
|
||||||
|
|
||||||
|
Performing operation...
|
||||||
|
|
||||||
|
[Safe, repeatable operation...]
|
||||||
|
|
||||||
|
Marking complete...
|
||||||
|
echo "$(date)" > .claude/operation-completed.flag
|
||||||
|
```
|
||||||
|
|
||||||
|
**Atomic operations:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Atomic command
|
||||||
|
---
|
||||||
|
|
||||||
|
# Atomic Operation
|
||||||
|
|
||||||
|
This operation is atomic - either fully succeeds or fully fails.
|
||||||
|
|
||||||
|
Creating temporary workspace...
|
||||||
|
TEMP_DIR=$(mktemp -d)
|
||||||
|
|
||||||
|
Performing changes in isolated environment...
|
||||||
|
[Make changes in $TEMP_DIR]
|
||||||
|
|
||||||
|
if [ $? -eq 0 ]; then
|
||||||
|
✓ Changes validated
|
||||||
|
|
||||||
|
Applying changes atomically...
|
||||||
|
mv $TEMP_DIR/* ./target/
|
||||||
|
|
||||||
|
✓ Operation complete
|
||||||
|
else
|
||||||
|
❌ Changes failed validation
|
||||||
|
|
||||||
|
Rolling back...
|
||||||
|
rm -rf $TEMP_DIR
|
||||||
|
|
||||||
|
No changes applied. Safe to retry.
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
## Testing for Distribution
|
||||||
|
|
||||||
|
### Pre-Release Checklist
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
<!--
|
||||||
|
PRE-RELEASE CHECKLIST:
|
||||||
|
|
||||||
|
Functionality:
|
||||||
|
- [ ] Works on macOS
|
||||||
|
- [ ] Works on Linux
|
||||||
|
- [ ] Works on Windows (WSL)
|
||||||
|
- [ ] All arguments tested
|
||||||
|
- [ ] Error cases handled
|
||||||
|
- [ ] Edge cases covered
|
||||||
|
|
||||||
|
User Experience:
|
||||||
|
- [ ] Clear description
|
||||||
|
- [ ] Helpful error messages
|
||||||
|
- [ ] Examples provided
|
||||||
|
- [ ] First-run experience good
|
||||||
|
- [ ] Documentation complete
|
||||||
|
|
||||||
|
Distribution:
|
||||||
|
- [ ] No hardcoded paths
|
||||||
|
- [ ] Dependencies documented
|
||||||
|
- [ ] Configuration options clear
|
||||||
|
- [ ] Version number set
|
||||||
|
- [ ] Changelog updated
|
||||||
|
|
||||||
|
Quality:
|
||||||
|
- [ ] No TODO comments
|
||||||
|
- [ ] No debug code
|
||||||
|
- [ ] Performance acceptable
|
||||||
|
- [ ] Security reviewed
|
||||||
|
- [ ] Privacy considered
|
||||||
|
|
||||||
|
Support:
|
||||||
|
- [ ] README complete
|
||||||
|
- [ ] Troubleshooting guide
|
||||||
|
- [ ] Support contact provided
|
||||||
|
- [ ] Feedback mechanism
|
||||||
|
- [ ] License specified
|
||||||
|
-->
|
||||||
|
```
|
||||||
|
|
||||||
|
### Beta Testing
|
||||||
|
|
||||||
|
**Beta release approach:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Beta command (v0.9.0)
|
||||||
|
---
|
||||||
|
|
||||||
|
# 🧪 Beta Command
|
||||||
|
|
||||||
|
**This is a beta release**
|
||||||
|
|
||||||
|
Features may change based on feedback.
|
||||||
|
|
||||||
|
BETA STATUS:
|
||||||
|
- Version: 0.9.0
|
||||||
|
- Stability: Experimental
|
||||||
|
- Support: Limited
|
||||||
|
- Feedback: Encouraged
|
||||||
|
|
||||||
|
Known limitations:
|
||||||
|
- Performance not optimized
|
||||||
|
- Some edge cases not handled
|
||||||
|
- Documentation incomplete
|
||||||
|
|
||||||
|
Help improve this command:
|
||||||
|
- Report issues: /command report-issue
|
||||||
|
- Suggest features: /command suggest
|
||||||
|
- Join beta testers: /command join-beta
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
[Command implementation...]
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Thank you for beta testing!**
|
||||||
|
|
||||||
|
Your feedback helps make this command better.
|
||||||
|
```
|
||||||
|
|
||||||
|
## Maintenance and Updates
|
||||||
|
|
||||||
|
### Update Strategy
|
||||||
|
|
||||||
|
**Versioned commands:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
<!--
|
||||||
|
VERSION STRATEGY:
|
||||||
|
|
||||||
|
Major (X.0.0): Breaking changes
|
||||||
|
- Document all breaking changes
|
||||||
|
- Provide migration guide
|
||||||
|
- Support old version briefly
|
||||||
|
|
||||||
|
Minor (x.Y.0): New features
|
||||||
|
- Backward compatible
|
||||||
|
- Announce new features
|
||||||
|
- Update examples
|
||||||
|
|
||||||
|
Patch (x.y.Z): Bug fixes
|
||||||
|
- No user-facing changes
|
||||||
|
- Update changelog
|
||||||
|
- Security fixes prioritized
|
||||||
|
|
||||||
|
Release schedule:
|
||||||
|
- Patches: As needed
|
||||||
|
- Minors: Monthly
|
||||||
|
- Majors: Annually or as needed
|
||||||
|
-->
|
||||||
|
```
|
||||||
|
|
||||||
|
**Update notifications:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Update-aware command
|
||||||
|
---
|
||||||
|
|
||||||
|
# Check for Updates
|
||||||
|
|
||||||
|
Current version: 2.1.0
|
||||||
|
Latest version: [check if available]
|
||||||
|
|
||||||
|
if [ "$CURRENT_VERSION" != "$LATEST_VERSION" ]; then
|
||||||
|
📢 UPDATE AVAILABLE
|
||||||
|
|
||||||
|
New version: $LATEST_VERSION
|
||||||
|
Current: $CURRENT_VERSION
|
||||||
|
|
||||||
|
What's new:
|
||||||
|
- Feature improvements
|
||||||
|
- Bug fixes
|
||||||
|
- Performance enhancements
|
||||||
|
|
||||||
|
Update with:
|
||||||
|
/plugin update plugin-name
|
||||||
|
|
||||||
|
Release notes: https://releases.example.com/v$LATEST_VERSION
|
||||||
|
fi
|
||||||
|
|
||||||
|
[Command continues...]
|
||||||
|
```
|
||||||
|
|
||||||
|
## Best Practices Summary
|
||||||
|
|
||||||
|
### Distribution Design
|
||||||
|
|
||||||
|
1. **Universal**: Works across platforms and environments
|
||||||
|
2. **Self-contained**: Minimal dependencies, clear requirements
|
||||||
|
3. **Graceful**: Degrades gracefully when features unavailable
|
||||||
|
4. **Forgiving**: Anticipates and handles user mistakes
|
||||||
|
5. **Helpful**: Clear errors, good defaults, excellent docs
|
||||||
|
|
||||||
|
### Marketplace Success
|
||||||
|
|
||||||
|
1. **Discoverable**: Clear name, good description, searchable keywords
|
||||||
|
2. **Professional**: Polished presentation, consistent branding
|
||||||
|
3. **Reliable**: Tested thoroughly, handles edge cases
|
||||||
|
4. **Maintainable**: Versioned, updated regularly, supported
|
||||||
|
5. **User-focused**: Great UX, responsive to feedback
|
||||||
|
|
||||||
|
### Quality Standards
|
||||||
|
|
||||||
|
1. **Complete**: Fully documented, all features working
|
||||||
|
2. **Tested**: Works in real environments, edge cases handled
|
||||||
|
3. **Secure**: No vulnerabilities, safe operations
|
||||||
|
4. **Performant**: Reasonable speed, resource-efficient
|
||||||
|
5. **Ethical**: Privacy-respecting, user consent
|
||||||
|
|
||||||
|
With these considerations, commands become marketplace-ready and delight users across diverse environments and use cases.
|
||||||
@@ -0,0 +1,609 @@
|
|||||||
|
# Plugin-Specific Command Features Reference
|
||||||
|
|
||||||
|
This reference covers features and patterns specific to commands bundled in Claude Code plugins.
|
||||||
|
|
||||||
|
## Table of Contents
|
||||||
|
|
||||||
|
- [Plugin Command Discovery](#plugin-command-discovery)
|
||||||
|
- [CLAUDE_PLUGIN_ROOT Environment Variable](#claude_plugin_root-environment-variable)
|
||||||
|
- [Plugin Command Patterns](#plugin-command-patterns)
|
||||||
|
- [Integration with Plugin Components](#integration-with-plugin-components)
|
||||||
|
- [Validation Patterns](#validation-patterns)
|
||||||
|
|
||||||
|
## Plugin Command Discovery
|
||||||
|
|
||||||
|
### Auto-Discovery
|
||||||
|
|
||||||
|
Claude Code automatically discovers commands in plugins using the following locations:
|
||||||
|
|
||||||
|
```
|
||||||
|
plugin-name/
|
||||||
|
├── commands/ # Auto-discovered commands
|
||||||
|
│ ├── foo.md # /foo (plugin:plugin-name)
|
||||||
|
│ └── bar.md # /bar (plugin:plugin-name)
|
||||||
|
└── plugin.json # Plugin manifest
|
||||||
|
```
|
||||||
|
|
||||||
|
**Key points:**
|
||||||
|
- Commands are discovered at plugin load time
|
||||||
|
- No manual registration required
|
||||||
|
- Commands appear in `/help` with "(plugin:plugin-name)" label
|
||||||
|
- Subdirectories create namespaces
|
||||||
|
|
||||||
|
### Namespaced Plugin Commands
|
||||||
|
|
||||||
|
Organize commands in subdirectories for logical grouping:
|
||||||
|
|
||||||
|
```
|
||||||
|
plugin-name/
|
||||||
|
└── commands/
|
||||||
|
├── review/
|
||||||
|
│ ├── security.md # /security (plugin:plugin-name:review)
|
||||||
|
│ └── style.md # /style (plugin:plugin-name:review)
|
||||||
|
└── deploy/
|
||||||
|
├── staging.md # /staging (plugin:plugin-name:deploy)
|
||||||
|
└── prod.md # /prod (plugin:plugin-name:deploy)
|
||||||
|
```
|
||||||
|
|
||||||
|
**Namespace behavior:**
|
||||||
|
- Subdirectory name becomes namespace
|
||||||
|
- Shown as "(plugin:plugin-name:namespace)" in `/help`
|
||||||
|
- Helps organize related commands
|
||||||
|
- Use when plugin has 5+ commands
|
||||||
|
|
||||||
|
### Command Naming Conventions
|
||||||
|
|
||||||
|
**Plugin command names should:**
|
||||||
|
1. Be descriptive and action-oriented
|
||||||
|
2. Avoid conflicts with common command names
|
||||||
|
3. Use hyphens for multi-word names
|
||||||
|
4. Consider prefixing with plugin name for uniqueness
|
||||||
|
|
||||||
|
**Examples:**
|
||||||
|
```
|
||||||
|
Good:
|
||||||
|
- /mylyn-sync (plugin-specific prefix)
|
||||||
|
- /analyze-performance (descriptive action)
|
||||||
|
- /docker-compose-up (clear purpose)
|
||||||
|
|
||||||
|
Avoid:
|
||||||
|
- /test (conflicts with common name)
|
||||||
|
- /run (too generic)
|
||||||
|
- /do-stuff (not descriptive)
|
||||||
|
```
|
||||||
|
|
||||||
|
## CLAUDE_PLUGIN_ROOT Environment Variable
|
||||||
|
|
||||||
|
### Purpose
|
||||||
|
|
||||||
|
`${CLAUDE_PLUGIN_ROOT}` is a special environment variable available in plugin commands that resolves to the absolute path of the plugin directory.
|
||||||
|
|
||||||
|
**Why it matters:**
|
||||||
|
- Enables portable paths within plugin
|
||||||
|
- Allows referencing plugin files and scripts
|
||||||
|
- Works across different installations
|
||||||
|
- Essential for multi-file plugin operations
|
||||||
|
|
||||||
|
### Basic Usage
|
||||||
|
|
||||||
|
Reference files within your plugin:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Analyze using plugin script
|
||||||
|
allowed-tools: Bash(node:*), Read
|
||||||
|
---
|
||||||
|
|
||||||
|
Run analysis: !`node ${CLAUDE_PLUGIN_ROOT}/scripts/analyze.js`
|
||||||
|
|
||||||
|
Read template: @${CLAUDE_PLUGIN_ROOT}/templates/report.md
|
||||||
|
```
|
||||||
|
|
||||||
|
**Expands to:**
|
||||||
|
```
|
||||||
|
Run analysis: !`node /path/to/plugins/plugin-name/scripts/analyze.js`
|
||||||
|
|
||||||
|
Read template: @/path/to/plugins/plugin-name/templates/report.md
|
||||||
|
```
|
||||||
|
|
||||||
|
### Common Patterns
|
||||||
|
|
||||||
|
#### 1. Executing Plugin Scripts
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Run custom linter from plugin
|
||||||
|
allowed-tools: Bash(node:*)
|
||||||
|
---
|
||||||
|
|
||||||
|
Lint results: !`node ${CLAUDE_PLUGIN_ROOT}/bin/lint.js $1`
|
||||||
|
|
||||||
|
Review the linting output and suggest fixes.
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 2. Loading Configuration Files
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Deploy using plugin configuration
|
||||||
|
allowed-tools: Read, Bash(*)
|
||||||
|
---
|
||||||
|
|
||||||
|
Configuration: @${CLAUDE_PLUGIN_ROOT}/config/deploy-config.json
|
||||||
|
|
||||||
|
Deploy application using the configuration above for $1 environment.
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 3. Accessing Plugin Resources
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Generate report from template
|
||||||
|
---
|
||||||
|
|
||||||
|
Use this template: @${CLAUDE_PLUGIN_ROOT}/templates/api-report.md
|
||||||
|
|
||||||
|
Generate a report for @$1 following the template format.
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 4. Multi-Step Plugin Workflows
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Complete plugin workflow
|
||||||
|
allowed-tools: Bash(*), Read
|
||||||
|
---
|
||||||
|
|
||||||
|
Step 1 - Prepare: !`bash ${CLAUDE_PLUGIN_ROOT}/scripts/prepare.sh $1`
|
||||||
|
Step 2 - Config: @${CLAUDE_PLUGIN_ROOT}/config/$1.json
|
||||||
|
Step 3 - Execute: !`${CLAUDE_PLUGIN_ROOT}/bin/execute $1`
|
||||||
|
|
||||||
|
Review results and report status.
|
||||||
|
```
|
||||||
|
|
||||||
|
### Best Practices
|
||||||
|
|
||||||
|
1. **Always use for plugin-internal paths:**
|
||||||
|
```markdown
|
||||||
|
# Good
|
||||||
|
@${CLAUDE_PLUGIN_ROOT}/templates/foo.md
|
||||||
|
|
||||||
|
# Bad
|
||||||
|
@./templates/foo.md # Relative to current directory, not plugin
|
||||||
|
```
|
||||||
|
|
||||||
|
2. **Validate file existence:**
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Use plugin config if exists
|
||||||
|
allowed-tools: Bash(test:*), Read
|
||||||
|
---
|
||||||
|
|
||||||
|
!`test -f ${CLAUDE_PLUGIN_ROOT}/config.json && echo "exists" || echo "missing"`
|
||||||
|
|
||||||
|
If config exists, load it: @${CLAUDE_PLUGIN_ROOT}/config.json
|
||||||
|
Otherwise, use defaults...
|
||||||
|
```
|
||||||
|
|
||||||
|
3. **Document plugin file structure:**
|
||||||
|
```markdown
|
||||||
|
<!--
|
||||||
|
Plugin structure:
|
||||||
|
${CLAUDE_PLUGIN_ROOT}/
|
||||||
|
├── scripts/analyze.js (analysis script)
|
||||||
|
├── templates/ (report templates)
|
||||||
|
└── config/ (configuration files)
|
||||||
|
-->
|
||||||
|
```
|
||||||
|
|
||||||
|
4. **Combine with arguments:**
|
||||||
|
```markdown
|
||||||
|
Run: !`${CLAUDE_PLUGIN_ROOT}/bin/process.sh $1 $2`
|
||||||
|
```
|
||||||
|
|
||||||
|
### Troubleshooting
|
||||||
|
|
||||||
|
**Variable not expanding:**
|
||||||
|
- Ensure command is loaded from plugin
|
||||||
|
- Check bash execution is allowed
|
||||||
|
- Verify syntax is exact: `${CLAUDE_PLUGIN_ROOT}`
|
||||||
|
|
||||||
|
**File not found errors:**
|
||||||
|
- Verify file exists in plugin directory
|
||||||
|
- Check file path is correct relative to plugin root
|
||||||
|
- Ensure file permissions allow reading/execution
|
||||||
|
|
||||||
|
**Path with spaces:**
|
||||||
|
- Bash commands automatically handle spaces
|
||||||
|
- File references work with spaces in paths
|
||||||
|
- No special quoting needed
|
||||||
|
|
||||||
|
## Plugin Command Patterns
|
||||||
|
|
||||||
|
### Pattern 1: Configuration-Based Commands
|
||||||
|
|
||||||
|
Commands that load plugin-specific configuration:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Deploy using plugin settings
|
||||||
|
allowed-tools: Read, Bash(*)
|
||||||
|
---
|
||||||
|
|
||||||
|
Load configuration: @${CLAUDE_PLUGIN_ROOT}/deploy-config.json
|
||||||
|
|
||||||
|
Deploy to $1 environment using:
|
||||||
|
1. Configuration settings above
|
||||||
|
2. Current git branch: !`git branch --show-current`
|
||||||
|
3. Application version: !`cat package.json | grep version`
|
||||||
|
|
||||||
|
Execute deployment and monitor progress.
|
||||||
|
```
|
||||||
|
|
||||||
|
**When to use:** Commands that need consistent settings across invocations
|
||||||
|
|
||||||
|
### Pattern 2: Template-Based Generation
|
||||||
|
|
||||||
|
Commands that use plugin templates:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Generate documentation from template
|
||||||
|
argument-hint: [component-name]
|
||||||
|
---
|
||||||
|
|
||||||
|
Template: @${CLAUDE_PLUGIN_ROOT}/templates/component-docs.md
|
||||||
|
|
||||||
|
Generate documentation for $1 component following the template structure.
|
||||||
|
Include:
|
||||||
|
- Component purpose and usage
|
||||||
|
- API reference
|
||||||
|
- Examples
|
||||||
|
- Testing guidelines
|
||||||
|
```
|
||||||
|
|
||||||
|
**When to use:** Standardized output generation
|
||||||
|
|
||||||
|
### Pattern 3: Multi-Script Workflow
|
||||||
|
|
||||||
|
Commands that orchestrate multiple plugin scripts:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Complete build and test workflow
|
||||||
|
allowed-tools: Bash(*)
|
||||||
|
---
|
||||||
|
|
||||||
|
Build: !`bash ${CLAUDE_PLUGIN_ROOT}/scripts/build.sh`
|
||||||
|
Validate: !`bash ${CLAUDE_PLUGIN_ROOT}/scripts/validate.sh`
|
||||||
|
Test: !`bash ${CLAUDE_PLUGIN_ROOT}/scripts/test.sh`
|
||||||
|
|
||||||
|
Review all outputs and report:
|
||||||
|
1. Build status
|
||||||
|
2. Validation results
|
||||||
|
3. Test results
|
||||||
|
4. Recommended next steps
|
||||||
|
```
|
||||||
|
|
||||||
|
**When to use:** Complex plugin workflows with multiple steps
|
||||||
|
|
||||||
|
### Pattern 4: Environment-Aware Commands
|
||||||
|
|
||||||
|
Commands that adapt to environment:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Deploy based on environment
|
||||||
|
argument-hint: [dev|staging|prod]
|
||||||
|
---
|
||||||
|
|
||||||
|
Environment config: @${CLAUDE_PLUGIN_ROOT}/config/$1.json
|
||||||
|
|
||||||
|
Environment check: !`echo "Deploying to: $1"`
|
||||||
|
|
||||||
|
Deploy application using $1 environment configuration.
|
||||||
|
Verify deployment and run smoke tests.
|
||||||
|
```
|
||||||
|
|
||||||
|
**When to use:** Commands that behave differently per environment
|
||||||
|
|
||||||
|
### Pattern 5: Plugin Data Management
|
||||||
|
|
||||||
|
Commands that manage plugin-specific data:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Save analysis results to plugin cache
|
||||||
|
allowed-tools: Bash(*), Read, Write
|
||||||
|
---
|
||||||
|
|
||||||
|
Cache directory: ${CLAUDE_PLUGIN_ROOT}/cache/
|
||||||
|
|
||||||
|
Analyze @$1 and save results to cache:
|
||||||
|
!`mkdir -p ${CLAUDE_PLUGIN_ROOT}/cache && date > ${CLAUDE_PLUGIN_ROOT}/cache/last-run.txt`
|
||||||
|
|
||||||
|
Store analysis for future reference and comparison.
|
||||||
|
```
|
||||||
|
|
||||||
|
**When to use:** Commands that need persistent data storage
|
||||||
|
|
||||||
|
## Integration with Plugin Components
|
||||||
|
|
||||||
|
### Invoking Plugin Agents
|
||||||
|
|
||||||
|
Commands can trigger plugin agents using the Task tool:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Deep analysis using plugin agent
|
||||||
|
argument-hint: [file-path]
|
||||||
|
---
|
||||||
|
|
||||||
|
Initiate deep code analysis of @$1 using the code-analyzer agent.
|
||||||
|
|
||||||
|
The agent will:
|
||||||
|
1. Analyze code structure
|
||||||
|
2. Identify patterns
|
||||||
|
3. Suggest improvements
|
||||||
|
4. Generate detailed report
|
||||||
|
|
||||||
|
Note: This uses the Task tool to launch the plugin's code-analyzer agent.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Key points:**
|
||||||
|
- Agent must be defined in plugin's `agents/` directory
|
||||||
|
- Claude will automatically use Task tool to launch agent
|
||||||
|
- Agent has access to same plugin resources
|
||||||
|
|
||||||
|
### Invoking Plugin Skills
|
||||||
|
|
||||||
|
Commands can reference plugin skills for specialized knowledge:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: API documentation with best practices
|
||||||
|
argument-hint: [api-file]
|
||||||
|
---
|
||||||
|
|
||||||
|
Document the API in @$1 following our API documentation standards.
|
||||||
|
|
||||||
|
Use the api-docs-standards skill to ensure documentation includes:
|
||||||
|
- Endpoint descriptions
|
||||||
|
- Parameter specifications
|
||||||
|
- Response formats
|
||||||
|
- Error codes
|
||||||
|
- Usage examples
|
||||||
|
|
||||||
|
Note: This leverages the plugin's api-docs-standards skill for consistency.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Key points:**
|
||||||
|
- Skill must be defined in plugin's `skills/` directory
|
||||||
|
- Mention skill by name to hint Claude should invoke it
|
||||||
|
- Skills provide specialized domain knowledge
|
||||||
|
|
||||||
|
### Coordinating with Plugin Hooks
|
||||||
|
|
||||||
|
Commands can be designed to work with plugin hooks:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Commit with pre-commit validation
|
||||||
|
allowed-tools: Bash(git:*)
|
||||||
|
---
|
||||||
|
|
||||||
|
Stage changes: !\`git add $1\`
|
||||||
|
|
||||||
|
Commit changes: !\`git commit -m "$2"\`
|
||||||
|
|
||||||
|
Note: This commit will trigger the plugin's pre-commit hook for validation.
|
||||||
|
Review hook output for any issues.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Key points:**
|
||||||
|
- Hooks execute automatically on events
|
||||||
|
- Commands can prepare state for hooks
|
||||||
|
- Document hook interaction in command
|
||||||
|
|
||||||
|
### Multi-Component Plugin Commands
|
||||||
|
|
||||||
|
Commands that coordinate multiple plugin components:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Comprehensive code review workflow
|
||||||
|
argument-hint: [file-path]
|
||||||
|
---
|
||||||
|
|
||||||
|
File to review: @$1
|
||||||
|
|
||||||
|
Execute comprehensive review:
|
||||||
|
|
||||||
|
1. **Static Analysis** (via plugin scripts)
|
||||||
|
!`node ${CLAUDE_PLUGIN_ROOT}/scripts/lint.js $1`
|
||||||
|
|
||||||
|
2. **Deep Review** (via plugin agent)
|
||||||
|
Launch the code-reviewer agent for detailed analysis.
|
||||||
|
|
||||||
|
3. **Best Practices** (via plugin skill)
|
||||||
|
Use the code-standards skill to ensure compliance.
|
||||||
|
|
||||||
|
4. **Documentation** (via plugin template)
|
||||||
|
Template: @${CLAUDE_PLUGIN_ROOT}/templates/review-report.md
|
||||||
|
|
||||||
|
Generate final report combining all outputs.
|
||||||
|
```
|
||||||
|
|
||||||
|
**When to use:** Complex workflows leveraging multiple plugin capabilities
|
||||||
|
|
||||||
|
## Validation Patterns
|
||||||
|
|
||||||
|
### Input Validation
|
||||||
|
|
||||||
|
Commands should validate inputs before processing:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Deploy to environment with validation
|
||||||
|
argument-hint: [environment]
|
||||||
|
---
|
||||||
|
|
||||||
|
Validate environment: !`echo "$1" | grep -E "^(dev|staging|prod)$" || echo "INVALID"`
|
||||||
|
|
||||||
|
$IF($1 in [dev, staging, prod],
|
||||||
|
Deploy to $1 environment using validated configuration,
|
||||||
|
ERROR: Invalid environment '$1'. Must be one of: dev, staging, prod
|
||||||
|
)
|
||||||
|
```
|
||||||
|
|
||||||
|
**Validation approaches:**
|
||||||
|
1. Bash validation using grep/test
|
||||||
|
2. Inline validation in prompt
|
||||||
|
3. Script-based validation
|
||||||
|
|
||||||
|
### File Existence Checks
|
||||||
|
|
||||||
|
Verify required files exist:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Process configuration file
|
||||||
|
argument-hint: [config-file]
|
||||||
|
---
|
||||||
|
|
||||||
|
Check file: !`test -f $1 && echo "EXISTS" || echo "MISSING"`
|
||||||
|
|
||||||
|
Process configuration if file exists: @$1
|
||||||
|
|
||||||
|
If file doesn't exist, explain:
|
||||||
|
- Expected location
|
||||||
|
- Required format
|
||||||
|
- How to create it
|
||||||
|
```
|
||||||
|
|
||||||
|
### Required Arguments
|
||||||
|
|
||||||
|
Validate required arguments provided:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Create deployment with version
|
||||||
|
argument-hint: [environment] [version]
|
||||||
|
---
|
||||||
|
|
||||||
|
Validate inputs: !`test -n "$1" -a -n "$2" && echo "OK" || echo "MISSING"`
|
||||||
|
|
||||||
|
$IF($1 AND $2,
|
||||||
|
Deploy version $2 to $1 environment,
|
||||||
|
ERROR: Both environment and version required. Usage: /deploy [env] [version]
|
||||||
|
)
|
||||||
|
```
|
||||||
|
|
||||||
|
### Plugin Resource Validation
|
||||||
|
|
||||||
|
Verify plugin resources available:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Run analysis with plugin tools
|
||||||
|
allowed-tools: Bash(test:*)
|
||||||
|
---
|
||||||
|
|
||||||
|
Validate plugin setup:
|
||||||
|
- Config exists: !`test -f ${CLAUDE_PLUGIN_ROOT}/config.json && echo "✓" || echo "✗"`
|
||||||
|
- Scripts exist: !`test -d ${CLAUDE_PLUGIN_ROOT}/scripts && echo "✓" || echo "✗"`
|
||||||
|
- Tools available: !`test -x ${CLAUDE_PLUGIN_ROOT}/bin/analyze && echo "✓" || echo "✗"`
|
||||||
|
|
||||||
|
If all checks pass, proceed with analysis.
|
||||||
|
Otherwise, report missing components and installation steps.
|
||||||
|
```
|
||||||
|
|
||||||
|
### Output Validation
|
||||||
|
|
||||||
|
Validate command execution results:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Build and validate output
|
||||||
|
allowed-tools: Bash(*)
|
||||||
|
---
|
||||||
|
|
||||||
|
Build: !`bash ${CLAUDE_PLUGIN_ROOT}/scripts/build.sh`
|
||||||
|
|
||||||
|
Validate output:
|
||||||
|
- Exit code: !`echo $?`
|
||||||
|
- Output exists: !`test -d dist && echo "✓" || echo "✗"`
|
||||||
|
- File count: !`find dist -type f | wc -l`
|
||||||
|
|
||||||
|
Report build status and any validation failures.
|
||||||
|
```
|
||||||
|
|
||||||
|
### Graceful Error Handling
|
||||||
|
|
||||||
|
Handle errors gracefully with helpful messages:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Process file with error handling
|
||||||
|
argument-hint: [file-path]
|
||||||
|
---
|
||||||
|
|
||||||
|
Try processing: !`node ${CLAUDE_PLUGIN_ROOT}/scripts/process.js $1 2>&1 || echo "ERROR: $?"`
|
||||||
|
|
||||||
|
If processing succeeded:
|
||||||
|
- Report results
|
||||||
|
- Suggest next steps
|
||||||
|
|
||||||
|
If processing failed:
|
||||||
|
- Explain likely causes
|
||||||
|
- Provide troubleshooting steps
|
||||||
|
- Suggest alternative approaches
|
||||||
|
```
|
||||||
|
|
||||||
|
## Best Practices Summary
|
||||||
|
|
||||||
|
### Plugin Commands Should:
|
||||||
|
|
||||||
|
1. **Use ${CLAUDE_PLUGIN_ROOT} for all plugin-internal paths**
|
||||||
|
- Scripts, templates, configuration, resources
|
||||||
|
|
||||||
|
2. **Validate inputs early**
|
||||||
|
- Check required arguments
|
||||||
|
- Verify file existence
|
||||||
|
- Validate argument formats
|
||||||
|
|
||||||
|
3. **Document plugin structure**
|
||||||
|
- Explain required files
|
||||||
|
- Document script purposes
|
||||||
|
- Clarify dependencies
|
||||||
|
|
||||||
|
4. **Integrate with plugin components**
|
||||||
|
- Reference agents for complex tasks
|
||||||
|
- Use skills for specialized knowledge
|
||||||
|
- Coordinate with hooks when relevant
|
||||||
|
|
||||||
|
5. **Provide helpful error messages**
|
||||||
|
- Explain what went wrong
|
||||||
|
- Suggest how to fix
|
||||||
|
- Offer alternatives
|
||||||
|
|
||||||
|
6. **Handle edge cases**
|
||||||
|
- Missing files
|
||||||
|
- Invalid arguments
|
||||||
|
- Failed script execution
|
||||||
|
- Missing dependencies
|
||||||
|
|
||||||
|
7. **Keep commands focused**
|
||||||
|
- One clear purpose per command
|
||||||
|
- Delegate complex logic to scripts
|
||||||
|
- Use agents for multi-step workflows
|
||||||
|
|
||||||
|
8. **Test across installations**
|
||||||
|
- Verify paths work everywhere
|
||||||
|
- Test with different arguments
|
||||||
|
- Validate error cases
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
For general command development, see main SKILL.md.
|
||||||
|
For command examples, see examples/ directory.
|
||||||
702
skills/command-development/references/testing-strategies.md
Normal file
702
skills/command-development/references/testing-strategies.md
Normal file
@@ -0,0 +1,702 @@
|
|||||||
|
# Command Testing Strategies
|
||||||
|
|
||||||
|
Comprehensive strategies for testing slash commands before deployment and distribution.
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
|
||||||
|
Testing commands ensures they work correctly, handle edge cases, and provide good user experience. A systematic testing approach catches issues early and builds confidence in command reliability.
|
||||||
|
|
||||||
|
## Testing Levels
|
||||||
|
|
||||||
|
### Level 1: Syntax and Structure Validation
|
||||||
|
|
||||||
|
**What to test:**
|
||||||
|
- YAML frontmatter syntax
|
||||||
|
- Markdown format
|
||||||
|
- File location and naming
|
||||||
|
|
||||||
|
**How to test:**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Validate YAML frontmatter
|
||||||
|
head -n 20 .claude/commands/my-command.md | grep -A 10 "^---"
|
||||||
|
|
||||||
|
# Check for closing frontmatter marker
|
||||||
|
head -n 20 .claude/commands/my-command.md | grep -c "^---" # Should be 2
|
||||||
|
|
||||||
|
# Verify file has .md extension
|
||||||
|
ls .claude/commands/*.md
|
||||||
|
|
||||||
|
# Check file is in correct location
|
||||||
|
test -f .claude/commands/my-command.md && echo "Found" || echo "Missing"
|
||||||
|
```
|
||||||
|
|
||||||
|
**Automated validation script:**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
# validate-command.sh
|
||||||
|
|
||||||
|
COMMAND_FILE="$1"
|
||||||
|
|
||||||
|
if [ ! -f "$COMMAND_FILE" ]; then
|
||||||
|
echo "ERROR: File not found: $COMMAND_FILE"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check .md extension
|
||||||
|
if [[ ! "$COMMAND_FILE" =~ \.md$ ]]; then
|
||||||
|
echo "ERROR: File must have .md extension"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Validate YAML frontmatter if present
|
||||||
|
if head -n 1 "$COMMAND_FILE" | grep -q "^---"; then
|
||||||
|
# Count frontmatter markers
|
||||||
|
MARKERS=$(head -n 50 "$COMMAND_FILE" | grep -c "^---")
|
||||||
|
if [ "$MARKERS" -ne 2 ]; then
|
||||||
|
echo "ERROR: Invalid YAML frontmatter (need exactly 2 '---' markers)"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
echo "✓ YAML frontmatter syntax valid"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check for empty file
|
||||||
|
if [ ! -s "$COMMAND_FILE" ]; then
|
||||||
|
echo "ERROR: File is empty"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "✓ Command file structure valid"
|
||||||
|
```
|
||||||
|
|
||||||
|
### Level 2: Frontmatter Field Validation
|
||||||
|
|
||||||
|
**What to test:**
|
||||||
|
- Field types correct
|
||||||
|
- Values in valid ranges
|
||||||
|
- Required fields present (if any)
|
||||||
|
|
||||||
|
**Validation script:**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
# validate-frontmatter.sh
|
||||||
|
|
||||||
|
COMMAND_FILE="$1"
|
||||||
|
|
||||||
|
# Extract YAML frontmatter
|
||||||
|
FRONTMATTER=$(sed -n '/^---$/,/^---$/p' "$COMMAND_FILE" | sed '1d;$d')
|
||||||
|
|
||||||
|
if [ -z "$FRONTMATTER" ]; then
|
||||||
|
echo "No frontmatter to validate"
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check 'model' field if present
|
||||||
|
if echo "$FRONTMATTER" | grep -q "^model:"; then
|
||||||
|
MODEL=$(echo "$FRONTMATTER" | grep "^model:" | cut -d: -f2 | tr -d ' ')
|
||||||
|
if ! echo "sonnet opus haiku" | grep -qw "$MODEL"; then
|
||||||
|
echo "ERROR: Invalid model '$MODEL' (must be sonnet, opus, or haiku)"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
echo "✓ Model field valid: $MODEL"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check 'allowed-tools' field format
|
||||||
|
if echo "$FRONTMATTER" | grep -q "^allowed-tools:"; then
|
||||||
|
echo "✓ allowed-tools field present"
|
||||||
|
# Could add more sophisticated validation here
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check 'description' length
|
||||||
|
if echo "$FRONTMATTER" | grep -q "^description:"; then
|
||||||
|
DESC=$(echo "$FRONTMATTER" | grep "^description:" | cut -d: -f2-)
|
||||||
|
LENGTH=${#DESC}
|
||||||
|
if [ "$LENGTH" -gt 80 ]; then
|
||||||
|
echo "WARNING: Description length $LENGTH (recommend < 60 chars)"
|
||||||
|
else
|
||||||
|
echo "✓ Description length acceptable: $LENGTH chars"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "✓ Frontmatter fields valid"
|
||||||
|
```
|
||||||
|
|
||||||
|
### Level 3: Manual Command Invocation
|
||||||
|
|
||||||
|
**What to test:**
|
||||||
|
- Command appears in `/help`
|
||||||
|
- Command executes without errors
|
||||||
|
- Output is as expected
|
||||||
|
|
||||||
|
**Test procedure:**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# 1. Start Claude Code
|
||||||
|
claude --debug
|
||||||
|
|
||||||
|
# 2. Check command appears in help
|
||||||
|
> /help
|
||||||
|
# Look for your command in the list
|
||||||
|
|
||||||
|
# 3. Invoke command without arguments
|
||||||
|
> /my-command
|
||||||
|
# Check for reasonable error or behavior
|
||||||
|
|
||||||
|
# 4. Invoke with valid arguments
|
||||||
|
> /my-command arg1 arg2
|
||||||
|
# Verify expected behavior
|
||||||
|
|
||||||
|
# 5. Check debug logs
|
||||||
|
tail -f ~/.claude/debug-logs/latest
|
||||||
|
# Look for errors or warnings
|
||||||
|
```
|
||||||
|
|
||||||
|
### Level 4: Argument Testing
|
||||||
|
|
||||||
|
**What to test:**
|
||||||
|
- Positional arguments work ($1, $2, etc.)
|
||||||
|
- $ARGUMENTS captures all arguments
|
||||||
|
- Missing arguments handled gracefully
|
||||||
|
- Invalid arguments detected
|
||||||
|
|
||||||
|
**Test matrix:**
|
||||||
|
|
||||||
|
| Test Case | Command | Expected Result |
|
||||||
|
|-----------|---------|-----------------|
|
||||||
|
| No args | `/cmd` | Graceful handling or useful message |
|
||||||
|
| One arg | `/cmd arg1` | $1 substituted correctly |
|
||||||
|
| Two args | `/cmd arg1 arg2` | $1 and $2 substituted |
|
||||||
|
| Extra args | `/cmd a b c d` | All captured or extras ignored appropriately |
|
||||||
|
| Special chars | `/cmd "arg with spaces"` | Quotes handled correctly |
|
||||||
|
| Empty arg | `/cmd ""` | Empty string handled |
|
||||||
|
|
||||||
|
**Test script:**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
# test-command-arguments.sh
|
||||||
|
|
||||||
|
COMMAND="$1"
|
||||||
|
|
||||||
|
echo "Testing argument handling for /$COMMAND"
|
||||||
|
echo
|
||||||
|
|
||||||
|
echo "Test 1: No arguments"
|
||||||
|
echo " Command: /$COMMAND"
|
||||||
|
echo " Expected: [describe expected behavior]"
|
||||||
|
echo " Manual test required"
|
||||||
|
echo
|
||||||
|
|
||||||
|
echo "Test 2: Single argument"
|
||||||
|
echo " Command: /$COMMAND test-value"
|
||||||
|
echo " Expected: 'test-value' appears in output"
|
||||||
|
echo " Manual test required"
|
||||||
|
echo
|
||||||
|
|
||||||
|
echo "Test 3: Multiple arguments"
|
||||||
|
echo " Command: /$COMMAND arg1 arg2 arg3"
|
||||||
|
echo " Expected: All arguments used appropriately"
|
||||||
|
echo " Manual test required"
|
||||||
|
echo
|
||||||
|
|
||||||
|
echo "Test 4: Special characters"
|
||||||
|
echo " Command: /$COMMAND \"value with spaces\""
|
||||||
|
echo " Expected: Entire phrase captured"
|
||||||
|
echo " Manual test required"
|
||||||
|
```
|
||||||
|
|
||||||
|
### Level 5: File Reference Testing
|
||||||
|
|
||||||
|
**What to test:**
|
||||||
|
- @ syntax loads file contents
|
||||||
|
- Non-existent files handled
|
||||||
|
- Large files handled appropriately
|
||||||
|
- Multiple file references work
|
||||||
|
|
||||||
|
**Test procedure:**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Create test files
|
||||||
|
echo "Test content" > /tmp/test-file.txt
|
||||||
|
echo "Second file" > /tmp/test-file-2.txt
|
||||||
|
|
||||||
|
# Test single file reference
|
||||||
|
> /my-command /tmp/test-file.txt
|
||||||
|
# Verify file content is read
|
||||||
|
|
||||||
|
# Test non-existent file
|
||||||
|
> /my-command /tmp/nonexistent.txt
|
||||||
|
# Verify graceful error handling
|
||||||
|
|
||||||
|
# Test multiple files
|
||||||
|
> /my-command /tmp/test-file.txt /tmp/test-file-2.txt
|
||||||
|
# Verify both files processed
|
||||||
|
|
||||||
|
# Test large file
|
||||||
|
dd if=/dev/zero of=/tmp/large-file.bin bs=1M count=100
|
||||||
|
> /my-command /tmp/large-file.bin
|
||||||
|
# Verify reasonable behavior (may truncate or warn)
|
||||||
|
|
||||||
|
# Cleanup
|
||||||
|
rm /tmp/test-file*.txt /tmp/large-file.bin
|
||||||
|
```
|
||||||
|
|
||||||
|
### Level 6: Bash Execution Testing
|
||||||
|
|
||||||
|
**What to test:**
|
||||||
|
- !` commands execute correctly
|
||||||
|
- Command output included in prompt
|
||||||
|
- Command failures handled
|
||||||
|
- Security: only allowed commands run
|
||||||
|
|
||||||
|
**Test procedure:**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Create test command with bash execution
|
||||||
|
cat > .claude/commands/test-bash.md << 'EOF'
|
||||||
|
---
|
||||||
|
description: Test bash execution
|
||||||
|
allowed-tools: Bash(echo:*), Bash(date:*)
|
||||||
|
---
|
||||||
|
|
||||||
|
Current date: !`date`
|
||||||
|
Test output: !`echo "Hello from bash"`
|
||||||
|
|
||||||
|
Analysis of output above...
|
||||||
|
EOF
|
||||||
|
|
||||||
|
# Test in Claude Code
|
||||||
|
> /test-bash
|
||||||
|
# Verify:
|
||||||
|
# 1. Date appears correctly
|
||||||
|
# 2. Echo output appears
|
||||||
|
# 3. No errors in debug logs
|
||||||
|
|
||||||
|
# Test with disallowed command (should fail or be blocked)
|
||||||
|
cat > .claude/commands/test-forbidden.md << 'EOF'
|
||||||
|
---
|
||||||
|
description: Test forbidden command
|
||||||
|
allowed-tools: Bash(echo:*)
|
||||||
|
---
|
||||||
|
|
||||||
|
Trying forbidden: !`ls -la /`
|
||||||
|
EOF
|
||||||
|
|
||||||
|
> /test-forbidden
|
||||||
|
# Verify: Permission denied or appropriate error
|
||||||
|
```
|
||||||
|
|
||||||
|
### Level 7: Integration Testing
|
||||||
|
|
||||||
|
**What to test:**
|
||||||
|
- Commands work with other plugin components
|
||||||
|
- Commands interact correctly with each other
|
||||||
|
- State management works across invocations
|
||||||
|
- Workflow commands execute in sequence
|
||||||
|
|
||||||
|
**Test scenarios:**
|
||||||
|
|
||||||
|
**Scenario 1: Command + Hook Integration**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Setup: Command that triggers a hook
|
||||||
|
# Test: Invoke command, verify hook executes
|
||||||
|
|
||||||
|
# Command: .claude/commands/risky-operation.md
|
||||||
|
# Hook: PreToolUse that validates the operation
|
||||||
|
|
||||||
|
> /risky-operation
|
||||||
|
# Verify: Hook executes and validates before command completes
|
||||||
|
```
|
||||||
|
|
||||||
|
**Scenario 2: Command Sequence**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Setup: Multi-command workflow
|
||||||
|
> /workflow-init
|
||||||
|
# Verify: State file created
|
||||||
|
|
||||||
|
> /workflow-step2
|
||||||
|
# Verify: State file read, step 2 executes
|
||||||
|
|
||||||
|
> /workflow-complete
|
||||||
|
# Verify: State file cleaned up
|
||||||
|
```
|
||||||
|
|
||||||
|
**Scenario 3: Command + MCP Integration**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Setup: Command uses MCP tools
|
||||||
|
# Test: Verify MCP server accessible
|
||||||
|
|
||||||
|
> /mcp-command
|
||||||
|
# Verify:
|
||||||
|
# 1. MCP server starts (if stdio)
|
||||||
|
# 2. Tool calls succeed
|
||||||
|
# 3. Results included in output
|
||||||
|
```
|
||||||
|
|
||||||
|
## Automated Testing Approaches
|
||||||
|
|
||||||
|
### Command Test Suite
|
||||||
|
|
||||||
|
Create a test suite script:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
# test-commands.sh - Command test suite
|
||||||
|
|
||||||
|
TEST_DIR=".claude/commands"
|
||||||
|
FAILED_TESTS=0
|
||||||
|
|
||||||
|
echo "Command Test Suite"
|
||||||
|
echo "=================="
|
||||||
|
echo
|
||||||
|
|
||||||
|
for cmd_file in "$TEST_DIR"/*.md; do
|
||||||
|
cmd_name=$(basename "$cmd_file" .md)
|
||||||
|
echo "Testing: $cmd_name"
|
||||||
|
|
||||||
|
# Validate structure
|
||||||
|
if ./validate-command.sh "$cmd_file"; then
|
||||||
|
echo " ✓ Structure valid"
|
||||||
|
else
|
||||||
|
echo " ✗ Structure invalid"
|
||||||
|
((FAILED_TESTS++))
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Validate frontmatter
|
||||||
|
if ./validate-frontmatter.sh "$cmd_file"; then
|
||||||
|
echo " ✓ Frontmatter valid"
|
||||||
|
else
|
||||||
|
echo " ✗ Frontmatter invalid"
|
||||||
|
((FAILED_TESTS++))
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo
|
||||||
|
done
|
||||||
|
|
||||||
|
echo "=================="
|
||||||
|
echo "Tests complete"
|
||||||
|
echo "Failed: $FAILED_TESTS"
|
||||||
|
|
||||||
|
exit $FAILED_TESTS
|
||||||
|
```
|
||||||
|
|
||||||
|
### Pre-Commit Hook
|
||||||
|
|
||||||
|
Validate commands before committing:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
# .git/hooks/pre-commit
|
||||||
|
|
||||||
|
echo "Validating commands..."
|
||||||
|
|
||||||
|
COMMANDS_CHANGED=$(git diff --cached --name-only | grep "\.claude/commands/.*\.md")
|
||||||
|
|
||||||
|
if [ -z "$COMMANDS_CHANGED" ]; then
|
||||||
|
echo "No commands changed"
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
for cmd in $COMMANDS_CHANGED; do
|
||||||
|
echo "Checking: $cmd"
|
||||||
|
|
||||||
|
if ! ./scripts/validate-command.sh "$cmd"; then
|
||||||
|
echo "ERROR: Command validation failed: $cmd"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
|
||||||
|
echo "✓ All commands valid"
|
||||||
|
```
|
||||||
|
|
||||||
|
### Continuous Testing
|
||||||
|
|
||||||
|
Test commands in CI/CD:
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
# .github/workflows/test-commands.yml
|
||||||
|
name: Test Commands
|
||||||
|
|
||||||
|
on: [push, pull_request]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
test:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
|
||||||
|
- name: Validate command structure
|
||||||
|
run: |
|
||||||
|
for cmd in .claude/commands/*.md; do
|
||||||
|
echo "Testing: $cmd"
|
||||||
|
./scripts/validate-command.sh "$cmd"
|
||||||
|
done
|
||||||
|
|
||||||
|
- name: Validate frontmatter
|
||||||
|
run: |
|
||||||
|
for cmd in .claude/commands/*.md; do
|
||||||
|
./scripts/validate-frontmatter.sh "$cmd"
|
||||||
|
done
|
||||||
|
|
||||||
|
- name: Check for TODOs
|
||||||
|
run: |
|
||||||
|
if grep -r "TODO" .claude/commands/; then
|
||||||
|
echo "ERROR: TODOs found in commands"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
## Edge Case Testing
|
||||||
|
|
||||||
|
### Test Edge Cases
|
||||||
|
|
||||||
|
**Empty arguments:**
|
||||||
|
```bash
|
||||||
|
> /cmd ""
|
||||||
|
> /cmd '' ''
|
||||||
|
```
|
||||||
|
|
||||||
|
**Special characters:**
|
||||||
|
```bash
|
||||||
|
> /cmd "arg with spaces"
|
||||||
|
> /cmd arg-with-dashes
|
||||||
|
> /cmd arg_with_underscores
|
||||||
|
> /cmd arg/with/slashes
|
||||||
|
> /cmd 'arg with "quotes"'
|
||||||
|
```
|
||||||
|
|
||||||
|
**Long arguments:**
|
||||||
|
```bash
|
||||||
|
> /cmd $(python -c "print('a' * 10000)")
|
||||||
|
```
|
||||||
|
|
||||||
|
**Unusual file paths:**
|
||||||
|
```bash
|
||||||
|
> /cmd ./file
|
||||||
|
> /cmd ../file
|
||||||
|
> /cmd ~/file
|
||||||
|
> /cmd "/path with spaces/file"
|
||||||
|
```
|
||||||
|
|
||||||
|
**Bash command edge cases:**
|
||||||
|
```markdown
|
||||||
|
# Commands that might fail
|
||||||
|
!`exit 1`
|
||||||
|
!`false`
|
||||||
|
!`command-that-does-not-exist`
|
||||||
|
|
||||||
|
# Commands with special output
|
||||||
|
!`echo ""`
|
||||||
|
!`cat /dev/null`
|
||||||
|
!`yes | head -n 1000000`
|
||||||
|
```
|
||||||
|
|
||||||
|
## Performance Testing
|
||||||
|
|
||||||
|
### Response Time Testing
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
# test-command-performance.sh
|
||||||
|
|
||||||
|
COMMAND="$1"
|
||||||
|
|
||||||
|
echo "Testing performance of /$COMMAND"
|
||||||
|
echo
|
||||||
|
|
||||||
|
for i in {1..5}; do
|
||||||
|
echo "Run $i:"
|
||||||
|
START=$(date +%s%N)
|
||||||
|
|
||||||
|
# Invoke command (manual step - record time)
|
||||||
|
echo " Invoke: /$COMMAND"
|
||||||
|
echo " Start time: $START"
|
||||||
|
echo " (Record end time manually)"
|
||||||
|
echo
|
||||||
|
done
|
||||||
|
|
||||||
|
echo "Analyze results:"
|
||||||
|
echo " - Average response time"
|
||||||
|
echo " - Variance"
|
||||||
|
echo " - Acceptable threshold: < 3 seconds for fast commands"
|
||||||
|
```
|
||||||
|
|
||||||
|
### Resource Usage Testing
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Monitor Claude Code during command execution
|
||||||
|
# In terminal 1:
|
||||||
|
claude --debug
|
||||||
|
|
||||||
|
# In terminal 2:
|
||||||
|
watch -n 1 'ps aux | grep claude'
|
||||||
|
|
||||||
|
# Execute command and observe:
|
||||||
|
# - Memory usage
|
||||||
|
# - CPU usage
|
||||||
|
# - Process count
|
||||||
|
```
|
||||||
|
|
||||||
|
## User Experience Testing
|
||||||
|
|
||||||
|
### Usability Checklist
|
||||||
|
|
||||||
|
- [ ] Command name is intuitive
|
||||||
|
- [ ] Description is clear in `/help`
|
||||||
|
- [ ] Arguments are well-documented
|
||||||
|
- [ ] Error messages are helpful
|
||||||
|
- [ ] Output is formatted readably
|
||||||
|
- [ ] Long-running commands show progress
|
||||||
|
- [ ] Results are actionable
|
||||||
|
- [ ] Edge cases have good UX
|
||||||
|
|
||||||
|
### User Acceptance Testing
|
||||||
|
|
||||||
|
Recruit testers:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
# Testing Guide for Beta Testers
|
||||||
|
|
||||||
|
## Command: /my-new-command
|
||||||
|
|
||||||
|
### Test Scenarios
|
||||||
|
|
||||||
|
1. **Basic usage:**
|
||||||
|
- Run: `/my-new-command`
|
||||||
|
- Expected: [describe]
|
||||||
|
- Rate clarity: 1-5
|
||||||
|
|
||||||
|
2. **With arguments:**
|
||||||
|
- Run: `/my-new-command arg1 arg2`
|
||||||
|
- Expected: [describe]
|
||||||
|
- Rate usefulness: 1-5
|
||||||
|
|
||||||
|
3. **Error case:**
|
||||||
|
- Run: `/my-new-command invalid-input`
|
||||||
|
- Expected: Helpful error message
|
||||||
|
- Rate error message: 1-5
|
||||||
|
|
||||||
|
### Feedback Questions
|
||||||
|
|
||||||
|
1. Was the command easy to understand?
|
||||||
|
2. Did the output meet your expectations?
|
||||||
|
3. What would you change?
|
||||||
|
4. Would you use this command regularly?
|
||||||
|
```
|
||||||
|
|
||||||
|
## Testing Checklist
|
||||||
|
|
||||||
|
Before releasing a command:
|
||||||
|
|
||||||
|
### Structure
|
||||||
|
- [ ] File in correct location
|
||||||
|
- [ ] Correct .md extension
|
||||||
|
- [ ] Valid YAML frontmatter (if present)
|
||||||
|
- [ ] Markdown syntax correct
|
||||||
|
|
||||||
|
### Functionality
|
||||||
|
- [ ] Command appears in `/help`
|
||||||
|
- [ ] Description is clear
|
||||||
|
- [ ] Command executes without errors
|
||||||
|
- [ ] Arguments work as expected
|
||||||
|
- [ ] File references work
|
||||||
|
- [ ] Bash execution works (if used)
|
||||||
|
|
||||||
|
### Edge Cases
|
||||||
|
- [ ] Missing arguments handled
|
||||||
|
- [ ] Invalid arguments detected
|
||||||
|
- [ ] Non-existent files handled
|
||||||
|
- [ ] Special characters work
|
||||||
|
- [ ] Long inputs handled
|
||||||
|
|
||||||
|
### Integration
|
||||||
|
- [ ] Works with other commands
|
||||||
|
- [ ] Works with hooks (if applicable)
|
||||||
|
- [ ] Works with MCP (if applicable)
|
||||||
|
- [ ] State management works
|
||||||
|
|
||||||
|
### Quality
|
||||||
|
- [ ] Performance acceptable
|
||||||
|
- [ ] No security issues
|
||||||
|
- [ ] Error messages helpful
|
||||||
|
- [ ] Output formatted well
|
||||||
|
- [ ] Documentation complete
|
||||||
|
|
||||||
|
### Distribution
|
||||||
|
- [ ] Tested by others
|
||||||
|
- [ ] Feedback incorporated
|
||||||
|
- [ ] README updated
|
||||||
|
- [ ] Examples provided
|
||||||
|
|
||||||
|
## Debugging Failed Tests
|
||||||
|
|
||||||
|
### Common Issues and Solutions
|
||||||
|
|
||||||
|
**Issue: Command not appearing in /help**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Check file location
|
||||||
|
ls -la .claude/commands/my-command.md
|
||||||
|
|
||||||
|
# Check permissions
|
||||||
|
chmod 644 .claude/commands/my-command.md
|
||||||
|
|
||||||
|
# Check syntax
|
||||||
|
head -n 20 .claude/commands/my-command.md
|
||||||
|
|
||||||
|
# Restart Claude Code
|
||||||
|
claude --debug
|
||||||
|
```
|
||||||
|
|
||||||
|
**Issue: Arguments not substituting**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Verify syntax
|
||||||
|
grep '\$1' .claude/commands/my-command.md
|
||||||
|
grep '\$ARGUMENTS' .claude/commands/my-command.md
|
||||||
|
|
||||||
|
# Test with simple command first
|
||||||
|
echo "Test: \$1 and \$2" > .claude/commands/test-args.md
|
||||||
|
```
|
||||||
|
|
||||||
|
**Issue: Bash commands not executing**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Check allowed-tools
|
||||||
|
grep "allowed-tools" .claude/commands/my-command.md
|
||||||
|
|
||||||
|
# Verify command syntax
|
||||||
|
grep '!\`' .claude/commands/my-command.md
|
||||||
|
|
||||||
|
# Test command manually
|
||||||
|
date
|
||||||
|
echo "test"
|
||||||
|
```
|
||||||
|
|
||||||
|
**Issue: File references not working**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Check @ syntax
|
||||||
|
grep '@' .claude/commands/my-command.md
|
||||||
|
|
||||||
|
# Verify file exists
|
||||||
|
ls -la /path/to/referenced/file
|
||||||
|
|
||||||
|
# Check permissions
|
||||||
|
chmod 644 /path/to/referenced/file
|
||||||
|
```
|
||||||
|
|
||||||
|
## Best Practices
|
||||||
|
|
||||||
|
1. **Test early, test often**: Validate as you develop
|
||||||
|
2. **Automate validation**: Use scripts for repeatable checks
|
||||||
|
3. **Test edge cases**: Don't just test the happy path
|
||||||
|
4. **Get feedback**: Have others test before wide release
|
||||||
|
5. **Document tests**: Keep test scenarios for regression testing
|
||||||
|
6. **Monitor in production**: Watch for issues after release
|
||||||
|
7. **Iterate**: Improve based on real usage data
|
||||||
711
skills/hook-development/SKILL.md
Normal file
711
skills/hook-development/SKILL.md
Normal file
@@ -0,0 +1,711 @@
|
|||||||
|
---
|
||||||
|
name: hook-development
|
||||||
|
description: This skill should be used when the user asks to "create a hook", "add a PreToolUse/PostToolUse/Stop hook", "validate tool use", "implement prompt-based hooks", "use ${CLAUDE_PLUGIN_ROOT}", "set up event-driven automation", "block dangerous commands", or mentions hook events (PreToolUse, PostToolUse, Stop, SubagentStop, SessionStart, SessionEnd, UserPromptSubmit, PreCompact, Notification). Provides comprehensive guidance for creating and implementing Claude Code plugin hooks with focus on advanced prompt-based hooks API.
|
||||||
|
---
|
||||||
|
|
||||||
|
# Hook Development for Claude Code Plugins
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
|
||||||
|
Hooks are event-driven automation scripts that execute in response to Claude Code events. Use hooks to validate operations, enforce policies, add context, and integrate external tools into workflows.
|
||||||
|
|
||||||
|
**Key capabilities:**
|
||||||
|
- Validate tool calls before execution (PreToolUse)
|
||||||
|
- React to tool results (PostToolUse)
|
||||||
|
- Enforce completion standards (Stop, SubagentStop)
|
||||||
|
- Load project context (SessionStart)
|
||||||
|
- Automate workflows across the development lifecycle
|
||||||
|
|
||||||
|
## Hook Types
|
||||||
|
|
||||||
|
### Prompt-Based Hooks (Recommended)
|
||||||
|
|
||||||
|
Use LLM-driven decision making for context-aware validation:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"type": "prompt",
|
||||||
|
"prompt": "Evaluate if this tool use is appropriate: $TOOL_INPUT",
|
||||||
|
"timeout": 30
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Supported events:** Stop, SubagentStop, UserPromptSubmit, PreToolUse
|
||||||
|
|
||||||
|
**Benefits:**
|
||||||
|
- Context-aware decisions based on natural language reasoning
|
||||||
|
- Flexible evaluation logic without bash scripting
|
||||||
|
- Better edge case handling
|
||||||
|
- Easier to maintain and extend
|
||||||
|
|
||||||
|
### Command Hooks
|
||||||
|
|
||||||
|
Execute bash commands for deterministic checks:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"command": "bash ${CLAUDE_PLUGIN_ROOT}/scripts/validate.sh",
|
||||||
|
"timeout": 60
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use for:**
|
||||||
|
- Fast deterministic validations
|
||||||
|
- File system operations
|
||||||
|
- External tool integrations
|
||||||
|
- Performance-critical checks
|
||||||
|
|
||||||
|
## Hook Configuration Formats
|
||||||
|
|
||||||
|
### Plugin hooks.json Format
|
||||||
|
|
||||||
|
**For plugin hooks** in `hooks/hooks.json`, use wrapper format:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"description": "Brief explanation of hooks (optional)",
|
||||||
|
"hooks": {
|
||||||
|
"PreToolUse": [...],
|
||||||
|
"Stop": [...],
|
||||||
|
"SessionStart": [...]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Key points:**
|
||||||
|
- `description` field is optional
|
||||||
|
- `hooks` field is required wrapper containing actual hook events
|
||||||
|
- This is the **plugin-specific format**
|
||||||
|
|
||||||
|
**Example:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"description": "Validation hooks for code quality",
|
||||||
|
"hooks": {
|
||||||
|
"PreToolUse": [
|
||||||
|
{
|
||||||
|
"matcher": "Write",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"command": "${CLAUDE_PLUGIN_ROOT}/hooks/validate.sh"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Settings Format (Direct)
|
||||||
|
|
||||||
|
**For user settings** in `.claude/settings.json`, use direct format:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"PreToolUse": [...],
|
||||||
|
"Stop": [...],
|
||||||
|
"SessionStart": [...]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Key points:**
|
||||||
|
- No wrapper - events directly at top level
|
||||||
|
- No description field
|
||||||
|
- This is the **settings format**
|
||||||
|
|
||||||
|
**Important:** The examples below show the hook event structure that goes inside either format. For plugin hooks.json, wrap these in `{"hooks": {...}}`.
|
||||||
|
|
||||||
|
## Hook Events
|
||||||
|
|
||||||
|
### PreToolUse
|
||||||
|
|
||||||
|
Execute before any tool runs. Use to approve, deny, or modify tool calls.
|
||||||
|
|
||||||
|
**Example (prompt-based):**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"PreToolUse": [
|
||||||
|
{
|
||||||
|
"matcher": "Write|Edit",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "prompt",
|
||||||
|
"prompt": "Validate file write safety. Check: system paths, credentials, path traversal, sensitive content. Return 'approve' or 'deny'."
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Output for PreToolUse:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"hookSpecificOutput": {
|
||||||
|
"permissionDecision": "allow|deny|ask",
|
||||||
|
"updatedInput": {"field": "modified_value"}
|
||||||
|
},
|
||||||
|
"systemMessage": "Explanation for Claude"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### PostToolUse
|
||||||
|
|
||||||
|
Execute after tool completes. Use to react to results, provide feedback, or log.
|
||||||
|
|
||||||
|
**Example:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"PostToolUse": [
|
||||||
|
{
|
||||||
|
"matcher": "Edit",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "prompt",
|
||||||
|
"prompt": "Analyze edit result for potential issues: syntax errors, security vulnerabilities, breaking changes. Provide feedback."
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Output behavior:**
|
||||||
|
- Exit 0: stdout shown in transcript
|
||||||
|
- Exit 2: stderr fed back to Claude
|
||||||
|
- systemMessage included in context
|
||||||
|
|
||||||
|
### Stop
|
||||||
|
|
||||||
|
Execute when main agent considers stopping. Use to validate completeness.
|
||||||
|
|
||||||
|
**Example:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"Stop": [
|
||||||
|
{
|
||||||
|
"matcher": "*",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "prompt",
|
||||||
|
"prompt": "Verify task completion: tests run, build succeeded, questions answered. Return 'approve' to stop or 'block' with reason to continue."
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Decision output:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"decision": "approve|block",
|
||||||
|
"reason": "Explanation",
|
||||||
|
"systemMessage": "Additional context"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### SubagentStop
|
||||||
|
|
||||||
|
Execute when subagent considers stopping. Use to ensure subagent completed its task.
|
||||||
|
|
||||||
|
Similar to Stop hook, but for subagents.
|
||||||
|
|
||||||
|
### UserPromptSubmit
|
||||||
|
|
||||||
|
Execute when user submits a prompt. Use to add context, validate, or block prompts.
|
||||||
|
|
||||||
|
**Example:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"UserPromptSubmit": [
|
||||||
|
{
|
||||||
|
"matcher": "*",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "prompt",
|
||||||
|
"prompt": "Check if prompt requires security guidance. If discussing auth, permissions, or API security, return relevant warnings."
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### SessionStart
|
||||||
|
|
||||||
|
Execute when Claude Code session begins. Use to load context and set environment.
|
||||||
|
|
||||||
|
**Example:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"SessionStart": [
|
||||||
|
{
|
||||||
|
"matcher": "*",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"command": "bash ${CLAUDE_PLUGIN_ROOT}/scripts/load-context.sh"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Special capability:** Persist environment variables using `$CLAUDE_ENV_FILE`:
|
||||||
|
```bash
|
||||||
|
echo "export PROJECT_TYPE=nodejs" >> "$CLAUDE_ENV_FILE"
|
||||||
|
```
|
||||||
|
|
||||||
|
See `examples/load-context.sh` for complete example.
|
||||||
|
|
||||||
|
### SessionEnd
|
||||||
|
|
||||||
|
Execute when session ends. Use for cleanup, logging, and state preservation.
|
||||||
|
|
||||||
|
### PreCompact
|
||||||
|
|
||||||
|
Execute before context compaction. Use to add critical information to preserve.
|
||||||
|
|
||||||
|
### Notification
|
||||||
|
|
||||||
|
Execute when Claude sends notifications. Use to react to user notifications.
|
||||||
|
|
||||||
|
## Hook Output Format
|
||||||
|
|
||||||
|
### Standard Output (All Hooks)
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"continue": true,
|
||||||
|
"suppressOutput": false,
|
||||||
|
"systemMessage": "Message for Claude"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
- `continue`: If false, halt processing (default true)
|
||||||
|
- `suppressOutput`: Hide output from transcript (default false)
|
||||||
|
- `systemMessage`: Message shown to Claude
|
||||||
|
|
||||||
|
### Exit Codes
|
||||||
|
|
||||||
|
- `0` - Success (stdout shown in transcript)
|
||||||
|
- `2` - Blocking error (stderr fed back to Claude)
|
||||||
|
- Other - Non-blocking error
|
||||||
|
|
||||||
|
## Hook Input Format
|
||||||
|
|
||||||
|
All hooks receive JSON via stdin with common fields:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"session_id": "abc123",
|
||||||
|
"transcript_path": "/path/to/transcript.txt",
|
||||||
|
"cwd": "/current/working/dir",
|
||||||
|
"permission_mode": "ask|allow",
|
||||||
|
"hook_event_name": "PreToolUse"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Event-specific fields:**
|
||||||
|
|
||||||
|
- **PreToolUse/PostToolUse:** `tool_name`, `tool_input`, `tool_result`
|
||||||
|
- **UserPromptSubmit:** `user_prompt`
|
||||||
|
- **Stop/SubagentStop:** `reason`
|
||||||
|
|
||||||
|
Access fields in prompts using `$TOOL_INPUT`, `$TOOL_RESULT`, `$USER_PROMPT`, etc.
|
||||||
|
|
||||||
|
## Environment Variables
|
||||||
|
|
||||||
|
Available in all command hooks:
|
||||||
|
|
||||||
|
- `$CLAUDE_PROJECT_DIR` - Project root path
|
||||||
|
- `$CLAUDE_PLUGIN_ROOT` - Plugin directory (use for portable paths)
|
||||||
|
- `$CLAUDE_ENV_FILE` - SessionStart only: persist env vars here
|
||||||
|
- `$CLAUDE_CODE_REMOTE` - Set if running in remote context
|
||||||
|
|
||||||
|
**Always use ${CLAUDE_PLUGIN_ROOT} in hook commands for portability:**
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"command": "bash ${CLAUDE_PLUGIN_ROOT}/scripts/validate.sh"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## Plugin Hook Configuration
|
||||||
|
|
||||||
|
In plugins, define hooks in `hooks/hooks.json`:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"PreToolUse": [
|
||||||
|
{
|
||||||
|
"matcher": "Write|Edit",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "prompt",
|
||||||
|
"prompt": "Validate file write safety"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"Stop": [
|
||||||
|
{
|
||||||
|
"matcher": "*",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "prompt",
|
||||||
|
"prompt": "Verify task completion"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"SessionStart": [
|
||||||
|
{
|
||||||
|
"matcher": "*",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"command": "bash ${CLAUDE_PLUGIN_ROOT}/scripts/load-context.sh",
|
||||||
|
"timeout": 10
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Plugin hooks merge with user's hooks and run in parallel.
|
||||||
|
|
||||||
|
## Matchers
|
||||||
|
|
||||||
|
### Tool Name Matching
|
||||||
|
|
||||||
|
**Exact match:**
|
||||||
|
```json
|
||||||
|
"matcher": "Write"
|
||||||
|
```
|
||||||
|
|
||||||
|
**Multiple tools:**
|
||||||
|
```json
|
||||||
|
"matcher": "Read|Write|Edit"
|
||||||
|
```
|
||||||
|
|
||||||
|
**Wildcard (all tools):**
|
||||||
|
```json
|
||||||
|
"matcher": "*"
|
||||||
|
```
|
||||||
|
|
||||||
|
**Regex patterns:**
|
||||||
|
```json
|
||||||
|
"matcher": "mcp__.*__delete.*" // All MCP delete tools
|
||||||
|
```
|
||||||
|
|
||||||
|
**Note:** Matchers are case-sensitive.
|
||||||
|
|
||||||
|
### Common Patterns
|
||||||
|
|
||||||
|
```json
|
||||||
|
// All MCP tools
|
||||||
|
"matcher": "mcp__.*"
|
||||||
|
|
||||||
|
// Specific plugin's MCP tools
|
||||||
|
"matcher": "mcp__plugin_asana_.*"
|
||||||
|
|
||||||
|
// All file operations
|
||||||
|
"matcher": "Read|Write|Edit"
|
||||||
|
|
||||||
|
// Bash commands only
|
||||||
|
"matcher": "Bash"
|
||||||
|
```
|
||||||
|
|
||||||
|
## Security Best Practices
|
||||||
|
|
||||||
|
### Input Validation
|
||||||
|
|
||||||
|
Always validate inputs in command hooks:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
input=$(cat)
|
||||||
|
tool_name=$(echo "$input" | jq -r '.tool_name')
|
||||||
|
|
||||||
|
# Validate tool name format
|
||||||
|
if [[ ! "$tool_name" =~ ^[a-zA-Z0-9_]+$ ]]; then
|
||||||
|
echo '{"decision": "deny", "reason": "Invalid tool name"}' >&2
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
### Path Safety
|
||||||
|
|
||||||
|
Check for path traversal and sensitive files:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
file_path=$(echo "$input" | jq -r '.tool_input.file_path')
|
||||||
|
|
||||||
|
# Deny path traversal
|
||||||
|
if [[ "$file_path" == *".."* ]]; then
|
||||||
|
echo '{"decision": "deny", "reason": "Path traversal detected"}' >&2
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Deny sensitive files
|
||||||
|
if [[ "$file_path" == *".env"* ]]; then
|
||||||
|
echo '{"decision": "deny", "reason": "Sensitive file"}' >&2
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
See `examples/validate-write.sh` and `examples/validate-bash.sh` for complete examples.
|
||||||
|
|
||||||
|
### Quote All Variables
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# GOOD: Quoted
|
||||||
|
echo "$file_path"
|
||||||
|
cd "$CLAUDE_PROJECT_DIR"
|
||||||
|
|
||||||
|
# BAD: Unquoted (injection risk)
|
||||||
|
echo $file_path
|
||||||
|
cd $CLAUDE_PROJECT_DIR
|
||||||
|
```
|
||||||
|
|
||||||
|
### Set Appropriate Timeouts
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"command": "bash script.sh",
|
||||||
|
"timeout": 10
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Defaults:** Command hooks (60s), Prompt hooks (30s)
|
||||||
|
|
||||||
|
## Performance Considerations
|
||||||
|
|
||||||
|
### Parallel Execution
|
||||||
|
|
||||||
|
All matching hooks run **in parallel**:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"PreToolUse": [
|
||||||
|
{
|
||||||
|
"matcher": "Write",
|
||||||
|
"hooks": [
|
||||||
|
{"type": "command", "command": "check1.sh"}, // Parallel
|
||||||
|
{"type": "command", "command": "check2.sh"}, // Parallel
|
||||||
|
{"type": "prompt", "prompt": "Validate..."} // Parallel
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Design implications:**
|
||||||
|
- Hooks don't see each other's output
|
||||||
|
- Non-deterministic ordering
|
||||||
|
- Design for independence
|
||||||
|
|
||||||
|
### Optimization
|
||||||
|
|
||||||
|
1. Use command hooks for quick deterministic checks
|
||||||
|
2. Use prompt hooks for complex reasoning
|
||||||
|
3. Cache validation results in temp files
|
||||||
|
4. Minimize I/O in hot paths
|
||||||
|
|
||||||
|
## Temporarily Active Hooks
|
||||||
|
|
||||||
|
Create hooks that activate conditionally by checking for a flag file or configuration:
|
||||||
|
|
||||||
|
**Pattern: Flag file activation**
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
# Only active when flag file exists
|
||||||
|
FLAG_FILE="$CLAUDE_PROJECT_DIR/.enable-strict-validation"
|
||||||
|
|
||||||
|
if [ ! -f "$FLAG_FILE" ]; then
|
||||||
|
# Flag not present, skip validation
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Flag present, run validation
|
||||||
|
input=$(cat)
|
||||||
|
# ... validation logic ...
|
||||||
|
```
|
||||||
|
|
||||||
|
**Pattern: Configuration-based activation**
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
# Check configuration for activation
|
||||||
|
CONFIG_FILE="$CLAUDE_PROJECT_DIR/.claude/plugin-config.json"
|
||||||
|
|
||||||
|
if [ -f "$CONFIG_FILE" ]; then
|
||||||
|
enabled=$(jq -r '.strictMode // false' "$CONFIG_FILE")
|
||||||
|
if [ "$enabled" != "true" ]; then
|
||||||
|
exit 0 # Not enabled, skip
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Enabled, run hook logic
|
||||||
|
input=$(cat)
|
||||||
|
# ... hook logic ...
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use cases:**
|
||||||
|
- Enable strict validation only when needed
|
||||||
|
- Temporary debugging hooks
|
||||||
|
- Project-specific hook behavior
|
||||||
|
- Feature flags for hooks
|
||||||
|
|
||||||
|
**Best practice:** Document activation mechanism in plugin README so users know how to enable/disable temporary hooks.
|
||||||
|
|
||||||
|
## Hook Lifecycle and Limitations
|
||||||
|
|
||||||
|
### Hooks Load at Session Start
|
||||||
|
|
||||||
|
**Important:** Hooks are loaded when Claude Code session starts. Changes to hook configuration require restarting Claude Code.
|
||||||
|
|
||||||
|
**Cannot hot-swap hooks:**
|
||||||
|
- Editing `hooks/hooks.json` won't affect current session
|
||||||
|
- Adding new hook scripts won't be recognized
|
||||||
|
- Changing hook commands/prompts won't update
|
||||||
|
- Must restart Claude Code: exit and run `claude` again
|
||||||
|
|
||||||
|
**To test hook changes:**
|
||||||
|
1. Edit hook configuration or scripts
|
||||||
|
2. Exit Claude Code session
|
||||||
|
3. Restart: `claude` or `cc`
|
||||||
|
4. New hook configuration loads
|
||||||
|
5. Test hooks with `claude --debug`
|
||||||
|
|
||||||
|
### Hook Validation at Startup
|
||||||
|
|
||||||
|
Hooks are validated when Claude Code starts:
|
||||||
|
- Invalid JSON in hooks.json causes loading failure
|
||||||
|
- Missing scripts cause warnings
|
||||||
|
- Syntax errors reported in debug mode
|
||||||
|
|
||||||
|
Use `/hooks` command to review loaded hooks in current session.
|
||||||
|
|
||||||
|
## Debugging Hooks
|
||||||
|
|
||||||
|
### Enable Debug Mode
|
||||||
|
|
||||||
|
```bash
|
||||||
|
claude --debug
|
||||||
|
```
|
||||||
|
|
||||||
|
Look for hook registration, execution logs, input/output JSON, and timing information.
|
||||||
|
|
||||||
|
### Test Hook Scripts
|
||||||
|
|
||||||
|
Test command hooks directly:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
echo '{"tool_name": "Write", "tool_input": {"file_path": "/test"}}' | \
|
||||||
|
bash ${CLAUDE_PLUGIN_ROOT}/scripts/validate.sh
|
||||||
|
|
||||||
|
echo "Exit code: $?"
|
||||||
|
```
|
||||||
|
|
||||||
|
### Validate JSON Output
|
||||||
|
|
||||||
|
Ensure hooks output valid JSON:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
output=$(./your-hook.sh < test-input.json)
|
||||||
|
echo "$output" | jq .
|
||||||
|
```
|
||||||
|
|
||||||
|
## Quick Reference
|
||||||
|
|
||||||
|
### Hook Events Summary
|
||||||
|
|
||||||
|
| Event | When | Use For |
|
||||||
|
|-------|------|---------|
|
||||||
|
| PreToolUse | Before tool | Validation, modification |
|
||||||
|
| PostToolUse | After tool | Feedback, logging |
|
||||||
|
| UserPromptSubmit | User input | Context, validation |
|
||||||
|
| Stop | Agent stopping | Completeness check |
|
||||||
|
| SubagentStop | Subagent done | Task validation |
|
||||||
|
| SessionStart | Session begins | Context loading |
|
||||||
|
| SessionEnd | Session ends | Cleanup, logging |
|
||||||
|
| PreCompact | Before compact | Preserve context |
|
||||||
|
| Notification | User notified | Logging, reactions |
|
||||||
|
|
||||||
|
### Best Practices
|
||||||
|
|
||||||
|
**DO:**
|
||||||
|
- ✅ Use prompt-based hooks for complex logic
|
||||||
|
- ✅ Use ${CLAUDE_PLUGIN_ROOT} for portability
|
||||||
|
- ✅ Validate all inputs in command hooks
|
||||||
|
- ✅ Quote all bash variables
|
||||||
|
- ✅ Set appropriate timeouts
|
||||||
|
- ✅ Return structured JSON output
|
||||||
|
- ✅ Test hooks thoroughly
|
||||||
|
|
||||||
|
**DON'T:**
|
||||||
|
- ❌ Use hardcoded paths
|
||||||
|
- ❌ Trust user input without validation
|
||||||
|
- ❌ Create long-running hooks
|
||||||
|
- ❌ Rely on hook execution order
|
||||||
|
- ❌ Modify global state unpredictably
|
||||||
|
- ❌ Log sensitive information
|
||||||
|
|
||||||
|
## Additional Resources
|
||||||
|
|
||||||
|
### Reference Files
|
||||||
|
|
||||||
|
For detailed patterns and advanced techniques, consult:
|
||||||
|
|
||||||
|
- **`references/patterns.md`** - Common hook patterns (8+ proven patterns)
|
||||||
|
- **`references/migration.md`** - Migrating from basic to advanced hooks
|
||||||
|
- **`references/advanced.md`** - Advanced use cases and techniques
|
||||||
|
|
||||||
|
### Example Hook Scripts
|
||||||
|
|
||||||
|
Working examples in `examples/`:
|
||||||
|
|
||||||
|
- **`validate-write.sh`** - File write validation example
|
||||||
|
- **`validate-bash.sh`** - Bash command validation example
|
||||||
|
- **`load-context.sh`** - SessionStart context loading example
|
||||||
|
|
||||||
|
### Utility Scripts
|
||||||
|
|
||||||
|
Development tools in `scripts/`:
|
||||||
|
|
||||||
|
- **`validate-hook-schema.sh`** - Validate hooks.json structure and syntax
|
||||||
|
- **`test-hook.sh`** - Test hooks with sample input before deployment
|
||||||
|
- **`hook-linter.sh`** - Check hook scripts for common issues and best practices
|
||||||
|
|
||||||
|
### External Resources
|
||||||
|
|
||||||
|
- **Official Docs**: https://docs.claude.com/en/docs/claude-code/hooks
|
||||||
|
- **Examples**: See security-guidance plugin in marketplace
|
||||||
|
- **Testing**: Use `claude --debug` for detailed logs
|
||||||
|
- **Validation**: Use `jq` to validate hook JSON output
|
||||||
|
|
||||||
|
## Implementation Workflow
|
||||||
|
|
||||||
|
To implement hooks in a plugin:
|
||||||
|
|
||||||
|
1. Identify events to hook into (PreToolUse, Stop, SessionStart, etc.)
|
||||||
|
2. Decide between prompt-based (flexible) or command (deterministic) hooks
|
||||||
|
3. Write hook configuration in `hooks/hooks.json`
|
||||||
|
4. For command hooks, create hook scripts
|
||||||
|
5. Use ${CLAUDE_PLUGIN_ROOT} for all file references
|
||||||
|
6. Validate configuration with `scripts/validate-hook-schema.sh hooks/hooks.json`
|
||||||
|
7. Test hooks with `scripts/test-hook.sh` before deployment
|
||||||
|
8. Test in Claude Code with `claude --debug`
|
||||||
|
9. Document hooks in plugin README
|
||||||
|
|
||||||
|
Focus on prompt-based hooks for most use cases. Reserve command hooks for performance-critical or deterministic checks.
|
||||||
55
skills/hook-development/examples/load-context.sh
Executable file
55
skills/hook-development/examples/load-context.sh
Executable file
@@ -0,0 +1,55 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
# Example SessionStart hook for loading project context
|
||||||
|
# This script detects project type and sets environment variables
|
||||||
|
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
# Navigate to project directory
|
||||||
|
cd "$CLAUDE_PROJECT_DIR" || exit 1
|
||||||
|
|
||||||
|
echo "Loading project context..."
|
||||||
|
|
||||||
|
# Detect project type and set environment
|
||||||
|
if [ -f "package.json" ]; then
|
||||||
|
echo "📦 Node.js project detected"
|
||||||
|
echo "export PROJECT_TYPE=nodejs" >> "$CLAUDE_ENV_FILE"
|
||||||
|
|
||||||
|
# Check if TypeScript
|
||||||
|
if [ -f "tsconfig.json" ]; then
|
||||||
|
echo "export USES_TYPESCRIPT=true" >> "$CLAUDE_ENV_FILE"
|
||||||
|
fi
|
||||||
|
|
||||||
|
elif [ -f "Cargo.toml" ]; then
|
||||||
|
echo "🦀 Rust project detected"
|
||||||
|
echo "export PROJECT_TYPE=rust" >> "$CLAUDE_ENV_FILE"
|
||||||
|
|
||||||
|
elif [ -f "go.mod" ]; then
|
||||||
|
echo "🐹 Go project detected"
|
||||||
|
echo "export PROJECT_TYPE=go" >> "$CLAUDE_ENV_FILE"
|
||||||
|
|
||||||
|
elif [ -f "pyproject.toml" ] || [ -f "setup.py" ]; then
|
||||||
|
echo "🐍 Python project detected"
|
||||||
|
echo "export PROJECT_TYPE=python" >> "$CLAUDE_ENV_FILE"
|
||||||
|
|
||||||
|
elif [ -f "pom.xml" ]; then
|
||||||
|
echo "☕ Java (Maven) project detected"
|
||||||
|
echo "export PROJECT_TYPE=java" >> "$CLAUDE_ENV_FILE"
|
||||||
|
echo "export BUILD_SYSTEM=maven" >> "$CLAUDE_ENV_FILE"
|
||||||
|
|
||||||
|
elif [ -f "build.gradle" ] || [ -f "build.gradle.kts" ]; then
|
||||||
|
echo "☕ Java/Kotlin (Gradle) project detected"
|
||||||
|
echo "export PROJECT_TYPE=java" >> "$CLAUDE_ENV_FILE"
|
||||||
|
echo "export BUILD_SYSTEM=gradle" >> "$CLAUDE_ENV_FILE"
|
||||||
|
|
||||||
|
else
|
||||||
|
echo "❓ Unknown project type"
|
||||||
|
echo "export PROJECT_TYPE=unknown" >> "$CLAUDE_ENV_FILE"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check for CI configuration
|
||||||
|
if [ -f ".github/workflows" ] || [ -f ".gitlab-ci.yml" ] || [ -f ".circleci/config.yml" ]; then
|
||||||
|
echo "export HAS_CI=true" >> "$CLAUDE_ENV_FILE"
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "Project context loaded successfully"
|
||||||
|
exit 0
|
||||||
43
skills/hook-development/examples/validate-bash.sh
Executable file
43
skills/hook-development/examples/validate-bash.sh
Executable file
@@ -0,0 +1,43 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
# Example PreToolUse hook for validating Bash commands
|
||||||
|
# This script demonstrates bash command validation patterns
|
||||||
|
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
# Read input from stdin
|
||||||
|
input=$(cat)
|
||||||
|
|
||||||
|
# Extract command
|
||||||
|
command=$(echo "$input" | jq -r '.tool_input.command // empty')
|
||||||
|
|
||||||
|
# Validate command exists
|
||||||
|
if [ -z "$command" ]; then
|
||||||
|
echo '{"continue": true}' # No command to validate
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check for obviously safe commands (quick approval)
|
||||||
|
if [[ "$command" =~ ^(ls|pwd|echo|date|whoami)(\s|$) ]]; then
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check for destructive operations
|
||||||
|
if [[ "$command" == *"rm -rf"* ]] || [[ "$command" == *"rm -fr"* ]]; then
|
||||||
|
echo '{"hookSpecificOutput": {"permissionDecision": "deny"}, "systemMessage": "Dangerous command detected: rm -rf"}' >&2
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check for other dangerous commands
|
||||||
|
if [[ "$command" == *"dd if="* ]] || [[ "$command" == *"mkfs"* ]] || [[ "$command" == *"> /dev/"* ]]; then
|
||||||
|
echo '{"hookSpecificOutput": {"permissionDecision": "deny"}, "systemMessage": "Dangerous system operation detected"}' >&2
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check for privilege escalation
|
||||||
|
if [[ "$command" == sudo* ]] || [[ "$command" == su* ]]; then
|
||||||
|
echo '{"hookSpecificOutput": {"permissionDecision": "ask"}, "systemMessage": "Command requires elevated privileges"}' >&2
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Approve the operation
|
||||||
|
exit 0
|
||||||
38
skills/hook-development/examples/validate-write.sh
Executable file
38
skills/hook-development/examples/validate-write.sh
Executable file
@@ -0,0 +1,38 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
# Example PreToolUse hook for validating Write/Edit operations
|
||||||
|
# This script demonstrates file write validation patterns
|
||||||
|
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
# Read input from stdin
|
||||||
|
input=$(cat)
|
||||||
|
|
||||||
|
# Extract file path and content
|
||||||
|
file_path=$(echo "$input" | jq -r '.tool_input.file_path // empty')
|
||||||
|
|
||||||
|
# Validate path exists
|
||||||
|
if [ -z "$file_path" ]; then
|
||||||
|
echo '{"continue": true}' # No path to validate
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check for path traversal
|
||||||
|
if [[ "$file_path" == *".."* ]]; then
|
||||||
|
echo '{"hookSpecificOutput": {"permissionDecision": "deny"}, "systemMessage": "Path traversal detected in: '"$file_path"'"}' >&2
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check for system directories
|
||||||
|
if [[ "$file_path" == /etc/* ]] || [[ "$file_path" == /sys/* ]] || [[ "$file_path" == /usr/* ]]; then
|
||||||
|
echo '{"hookSpecificOutput": {"permissionDecision": "deny"}, "systemMessage": "Cannot write to system directory: '"$file_path"'"}' >&2
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check for sensitive files
|
||||||
|
if [[ "$file_path" == *.env ]] || [[ "$file_path" == *secret* ]] || [[ "$file_path" == *credentials* ]]; then
|
||||||
|
echo '{"hookSpecificOutput": {"permissionDecision": "ask"}, "systemMessage": "Writing to potentially sensitive file: '"$file_path"'"}' >&2
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Approve the operation
|
||||||
|
exit 0
|
||||||
479
skills/hook-development/references/advanced.md
Normal file
479
skills/hook-development/references/advanced.md
Normal file
@@ -0,0 +1,479 @@
|
|||||||
|
# Advanced Hook Use Cases
|
||||||
|
|
||||||
|
This reference covers advanced hook patterns and techniques for sophisticated automation workflows.
|
||||||
|
|
||||||
|
## Multi-Stage Validation
|
||||||
|
|
||||||
|
Combine command and prompt hooks for layered validation:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"PreToolUse": [
|
||||||
|
{
|
||||||
|
"matcher": "Bash",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"command": "bash ${CLAUDE_PLUGIN_ROOT}/scripts/quick-check.sh",
|
||||||
|
"timeout": 5
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "prompt",
|
||||||
|
"prompt": "Deep analysis of bash command: $TOOL_INPUT",
|
||||||
|
"timeout": 15
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use case:** Fast deterministic checks followed by intelligent analysis
|
||||||
|
|
||||||
|
**Example quick-check.sh:**
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
input=$(cat)
|
||||||
|
command=$(echo "$input" | jq -r '.tool_input.command')
|
||||||
|
|
||||||
|
# Immediate approval for safe commands
|
||||||
|
if [[ "$command" =~ ^(ls|pwd|echo|date|whoami)$ ]]; then
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Let prompt hook handle complex cases
|
||||||
|
exit 0
|
||||||
|
```
|
||||||
|
|
||||||
|
The command hook quickly approves obviously safe commands, while the prompt hook analyzes everything else.
|
||||||
|
|
||||||
|
## Conditional Hook Execution
|
||||||
|
|
||||||
|
Execute hooks based on environment or context:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
# Only run in CI environment
|
||||||
|
if [ -z "$CI" ]; then
|
||||||
|
echo '{"continue": true}' # Skip in non-CI
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Run validation logic in CI
|
||||||
|
input=$(cat)
|
||||||
|
# ... validation code ...
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use cases:**
|
||||||
|
- Different behavior in CI vs local development
|
||||||
|
- Project-specific validation
|
||||||
|
- User-specific rules
|
||||||
|
|
||||||
|
**Example: Skip certain checks for trusted users:**
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
# Skip detailed checks for admin users
|
||||||
|
if [ "$USER" = "admin" ]; then
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Full validation for other users
|
||||||
|
input=$(cat)
|
||||||
|
# ... validation code ...
|
||||||
|
```
|
||||||
|
|
||||||
|
## Hook Chaining via State
|
||||||
|
|
||||||
|
Share state between hooks using temporary files:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Hook 1: Analyze and save state
|
||||||
|
#!/bin/bash
|
||||||
|
input=$(cat)
|
||||||
|
command=$(echo "$input" | jq -r '.tool_input.command')
|
||||||
|
|
||||||
|
# Analyze command
|
||||||
|
risk_level=$(calculate_risk "$command")
|
||||||
|
echo "$risk_level" > /tmp/hook-state-$$
|
||||||
|
|
||||||
|
exit 0
|
||||||
|
```
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Hook 2: Use saved state
|
||||||
|
#!/bin/bash
|
||||||
|
risk_level=$(cat /tmp/hook-state-$$ 2>/dev/null || echo "unknown")
|
||||||
|
|
||||||
|
if [ "$risk_level" = "high" ]; then
|
||||||
|
echo "High risk operation detected" >&2
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
**Important:** This only works for sequential hook events (e.g., PreToolUse then PostToolUse), not parallel hooks.
|
||||||
|
|
||||||
|
## Dynamic Hook Configuration
|
||||||
|
|
||||||
|
Modify hook behavior based on project configuration:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
cd "$CLAUDE_PROJECT_DIR" || exit 1
|
||||||
|
|
||||||
|
# Read project-specific config
|
||||||
|
if [ -f ".claude-hooks-config.json" ]; then
|
||||||
|
strict_mode=$(jq -r '.strict_mode' .claude-hooks-config.json)
|
||||||
|
|
||||||
|
if [ "$strict_mode" = "true" ]; then
|
||||||
|
# Apply strict validation
|
||||||
|
# ...
|
||||||
|
else
|
||||||
|
# Apply lenient validation
|
||||||
|
# ...
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
**Example .claude-hooks-config.json:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"strict_mode": true,
|
||||||
|
"allowed_commands": ["ls", "pwd", "grep"],
|
||||||
|
"forbidden_paths": ["/etc", "/sys"]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## Context-Aware Prompt Hooks
|
||||||
|
|
||||||
|
Use transcript and session context for intelligent decisions:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"Stop": [
|
||||||
|
{
|
||||||
|
"matcher": "*",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "prompt",
|
||||||
|
"prompt": "Review the full transcript at $TRANSCRIPT_PATH. Check: 1) Were tests run after code changes? 2) Did the build succeed? 3) Were all user questions answered? 4) Is there any unfinished work? Return 'approve' only if everything is complete."
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
The LLM can read the transcript file and make context-aware decisions.
|
||||||
|
|
||||||
|
## Performance Optimization
|
||||||
|
|
||||||
|
### Caching Validation Results
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
input=$(cat)
|
||||||
|
file_path=$(echo "$input" | jq -r '.tool_input.file_path')
|
||||||
|
cache_key=$(echo -n "$file_path" | md5sum | cut -d' ' -f1)
|
||||||
|
cache_file="/tmp/hook-cache-$cache_key"
|
||||||
|
|
||||||
|
# Check cache
|
||||||
|
if [ -f "$cache_file" ]; then
|
||||||
|
cache_age=$(($(date +%s) - $(stat -f%m "$cache_file" 2>/dev/null || stat -c%Y "$cache_file")))
|
||||||
|
if [ "$cache_age" -lt 300 ]; then # 5 minute cache
|
||||||
|
cat "$cache_file"
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Perform validation
|
||||||
|
result='{"decision": "approve"}'
|
||||||
|
|
||||||
|
# Cache result
|
||||||
|
echo "$result" > "$cache_file"
|
||||||
|
echo "$result"
|
||||||
|
```
|
||||||
|
|
||||||
|
### Parallel Execution Optimization
|
||||||
|
|
||||||
|
Since hooks run in parallel, design them to be independent:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"PreToolUse": [
|
||||||
|
{
|
||||||
|
"matcher": "Write",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"command": "bash check-size.sh", // Independent
|
||||||
|
"timeout": 2
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"command": "bash check-path.sh", // Independent
|
||||||
|
"timeout": 2
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "prompt",
|
||||||
|
"prompt": "Check content safety", // Independent
|
||||||
|
"timeout": 10
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
All three hooks run simultaneously, reducing total latency.
|
||||||
|
|
||||||
|
## Cross-Event Workflows
|
||||||
|
|
||||||
|
Coordinate hooks across different events:
|
||||||
|
|
||||||
|
**SessionStart - Set up tracking:**
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
# Initialize session tracking
|
||||||
|
echo "0" > /tmp/test-count-$$
|
||||||
|
echo "0" > /tmp/build-count-$$
|
||||||
|
```
|
||||||
|
|
||||||
|
**PostToolUse - Track events:**
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
input=$(cat)
|
||||||
|
tool_name=$(echo "$input" | jq -r '.tool_name')
|
||||||
|
|
||||||
|
if [ "$tool_name" = "Bash" ]; then
|
||||||
|
command=$(echo "$input" | jq -r '.tool_result')
|
||||||
|
if [[ "$command" == *"test"* ]]; then
|
||||||
|
count=$(cat /tmp/test-count-$$ 2>/dev/null || echo "0")
|
||||||
|
echo $((count + 1)) > /tmp/test-count-$$
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
**Stop - Verify based on tracking:**
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
test_count=$(cat /tmp/test-count-$$ 2>/dev/null || echo "0")
|
||||||
|
|
||||||
|
if [ "$test_count" -eq 0 ]; then
|
||||||
|
echo '{"decision": "block", "reason": "No tests were run"}' >&2
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
## Integration with External Systems
|
||||||
|
|
||||||
|
### Slack Notifications
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
input=$(cat)
|
||||||
|
tool_name=$(echo "$input" | jq -r '.tool_name')
|
||||||
|
decision="blocked"
|
||||||
|
|
||||||
|
# Send notification to Slack
|
||||||
|
curl -X POST "$SLACK_WEBHOOK" \
|
||||||
|
-H 'Content-Type: application/json' \
|
||||||
|
-d "{\"text\": \"Hook ${decision} ${tool_name} operation\"}" \
|
||||||
|
2>/dev/null
|
||||||
|
|
||||||
|
echo '{"decision": "deny"}' >&2
|
||||||
|
exit 2
|
||||||
|
```
|
||||||
|
|
||||||
|
### Database Logging
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
input=$(cat)
|
||||||
|
|
||||||
|
# Log to database
|
||||||
|
psql "$DATABASE_URL" -c "INSERT INTO hook_logs (event, data) VALUES ('PreToolUse', '$input')" \
|
||||||
|
2>/dev/null
|
||||||
|
|
||||||
|
exit 0
|
||||||
|
```
|
||||||
|
|
||||||
|
### Metrics Collection
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
input=$(cat)
|
||||||
|
tool_name=$(echo "$input" | jq -r '.tool_name')
|
||||||
|
|
||||||
|
# Send metrics to monitoring system
|
||||||
|
echo "hook.pretooluse.${tool_name}:1|c" | nc -u -w1 statsd.local 8125
|
||||||
|
|
||||||
|
exit 0
|
||||||
|
```
|
||||||
|
|
||||||
|
## Security Patterns
|
||||||
|
|
||||||
|
### Rate Limiting
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
input=$(cat)
|
||||||
|
command=$(echo "$input" | jq -r '.tool_input.command')
|
||||||
|
|
||||||
|
# Track command frequency
|
||||||
|
rate_file="/tmp/hook-rate-$$"
|
||||||
|
current_minute=$(date +%Y%m%d%H%M)
|
||||||
|
|
||||||
|
if [ -f "$rate_file" ]; then
|
||||||
|
last_minute=$(head -1 "$rate_file")
|
||||||
|
count=$(tail -1 "$rate_file")
|
||||||
|
|
||||||
|
if [ "$current_minute" = "$last_minute" ]; then
|
||||||
|
if [ "$count" -gt 10 ]; then
|
||||||
|
echo '{"decision": "deny", "reason": "Rate limit exceeded"}' >&2
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
count=$((count + 1))
|
||||||
|
else
|
||||||
|
count=1
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
count=1
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "$current_minute" > "$rate_file"
|
||||||
|
echo "$count" >> "$rate_file"
|
||||||
|
|
||||||
|
exit 0
|
||||||
|
```
|
||||||
|
|
||||||
|
### Audit Logging
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
input=$(cat)
|
||||||
|
tool_name=$(echo "$input" | jq -r '.tool_name')
|
||||||
|
timestamp=$(date -Iseconds)
|
||||||
|
|
||||||
|
# Append to audit log
|
||||||
|
echo "$timestamp | $USER | $tool_name | $input" >> ~/.claude/audit.log
|
||||||
|
|
||||||
|
exit 0
|
||||||
|
```
|
||||||
|
|
||||||
|
### Secret Detection
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
input=$(cat)
|
||||||
|
content=$(echo "$input" | jq -r '.tool_input.content')
|
||||||
|
|
||||||
|
# Check for common secret patterns
|
||||||
|
if echo "$content" | grep -qE "(api[_-]?key|password|secret|token).{0,20}['\"]?[A-Za-z0-9]{20,}"; then
|
||||||
|
echo '{"decision": "deny", "reason": "Potential secret detected in content"}' >&2
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
|
||||||
|
exit 0
|
||||||
|
```
|
||||||
|
|
||||||
|
## Testing Advanced Hooks
|
||||||
|
|
||||||
|
### Unit Testing Hook Scripts
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# test-hook.sh
|
||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
# Test 1: Approve safe command
|
||||||
|
result=$(echo '{"tool_input": {"command": "ls"}}' | bash validate-bash.sh)
|
||||||
|
if [ $? -eq 0 ]; then
|
||||||
|
echo "✓ Test 1 passed"
|
||||||
|
else
|
||||||
|
echo "✗ Test 1 failed"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Test 2: Block dangerous command
|
||||||
|
result=$(echo '{"tool_input": {"command": "rm -rf /"}}' | bash validate-bash.sh)
|
||||||
|
if [ $? -eq 2 ]; then
|
||||||
|
echo "✓ Test 2 passed"
|
||||||
|
else
|
||||||
|
echo "✗ Test 2 failed"
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
### Integration Testing
|
||||||
|
|
||||||
|
Create test scenarios that exercise the full hook workflow:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# integration-test.sh
|
||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
# Set up test environment
|
||||||
|
export CLAUDE_PROJECT_DIR="/tmp/test-project"
|
||||||
|
export CLAUDE_PLUGIN_ROOT="$(pwd)"
|
||||||
|
mkdir -p "$CLAUDE_PROJECT_DIR"
|
||||||
|
|
||||||
|
# Test SessionStart hook
|
||||||
|
echo '{}' | bash hooks/session-start.sh
|
||||||
|
if [ -f "/tmp/session-initialized" ]; then
|
||||||
|
echo "✓ SessionStart hook works"
|
||||||
|
else
|
||||||
|
echo "✗ SessionStart hook failed"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Clean up
|
||||||
|
rm -rf "$CLAUDE_PROJECT_DIR"
|
||||||
|
```
|
||||||
|
|
||||||
|
## Best Practices for Advanced Hooks
|
||||||
|
|
||||||
|
1. **Keep hooks independent**: Don't rely on execution order
|
||||||
|
2. **Use timeouts**: Set appropriate limits for each hook type
|
||||||
|
3. **Handle errors gracefully**: Provide clear error messages
|
||||||
|
4. **Document complexity**: Explain advanced patterns in README
|
||||||
|
5. **Test thoroughly**: Cover edge cases and failure modes
|
||||||
|
6. **Monitor performance**: Track hook execution time
|
||||||
|
7. **Version configuration**: Use version control for hook configs
|
||||||
|
8. **Provide escape hatches**: Allow users to bypass hooks when needed
|
||||||
|
|
||||||
|
## Common Pitfalls
|
||||||
|
|
||||||
|
### ❌ Assuming Hook Order
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# BAD: Assumes hooks run in specific order
|
||||||
|
# Hook 1 saves state, Hook 2 reads it
|
||||||
|
# This can fail because hooks run in parallel!
|
||||||
|
```
|
||||||
|
|
||||||
|
### ❌ Long-Running Hooks
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# BAD: Hook takes 2 minutes to run
|
||||||
|
sleep 120
|
||||||
|
# This will timeout and block the workflow
|
||||||
|
```
|
||||||
|
|
||||||
|
### ❌ Uncaught Exceptions
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# BAD: Script crashes on unexpected input
|
||||||
|
file_path=$(echo "$input" | jq -r '.tool_input.file_path')
|
||||||
|
cat "$file_path" # Fails if file doesn't exist
|
||||||
|
```
|
||||||
|
|
||||||
|
### ✅ Proper Error Handling
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# GOOD: Handles errors gracefully
|
||||||
|
file_path=$(echo "$input" | jq -r '.tool_input.file_path')
|
||||||
|
if [ ! -f "$file_path" ]; then
|
||||||
|
echo '{"continue": true, "systemMessage": "File not found, skipping check"}' >&2
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
## Conclusion
|
||||||
|
|
||||||
|
Advanced hook patterns enable sophisticated automation while maintaining reliability and performance. Use these techniques when basic hooks are insufficient, but always prioritize simplicity and maintainability.
|
||||||
369
skills/hook-development/references/migration.md
Normal file
369
skills/hook-development/references/migration.md
Normal file
@@ -0,0 +1,369 @@
|
|||||||
|
# Migrating from Basic to Advanced Hooks
|
||||||
|
|
||||||
|
This guide shows how to migrate from basic command hooks to advanced prompt-based hooks for better maintainability and flexibility.
|
||||||
|
|
||||||
|
## Why Migrate?
|
||||||
|
|
||||||
|
Prompt-based hooks offer several advantages:
|
||||||
|
|
||||||
|
- **Natural language reasoning**: LLM understands context and intent
|
||||||
|
- **Better edge case handling**: Adapts to unexpected scenarios
|
||||||
|
- **No bash scripting required**: Simpler to write and maintain
|
||||||
|
- **More flexible validation**: Can handle complex logic without coding
|
||||||
|
|
||||||
|
## Migration Example: Bash Command Validation
|
||||||
|
|
||||||
|
### Before (Basic Command Hook)
|
||||||
|
|
||||||
|
**Configuration:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"PreToolUse": [
|
||||||
|
{
|
||||||
|
"matcher": "Bash",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"command": "bash validate-bash.sh"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Script (validate-bash.sh):**
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
input=$(cat)
|
||||||
|
command=$(echo "$input" | jq -r '.tool_input.command')
|
||||||
|
|
||||||
|
# Hard-coded validation logic
|
||||||
|
if [[ "$command" == *"rm -rf"* ]]; then
|
||||||
|
echo "Dangerous command detected" >&2
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
**Problems:**
|
||||||
|
- Only checks for exact "rm -rf" pattern
|
||||||
|
- Doesn't catch variations like `rm -fr` or `rm -r -f`
|
||||||
|
- Misses other dangerous commands (`dd`, `mkfs`, etc.)
|
||||||
|
- No context awareness
|
||||||
|
- Requires bash scripting knowledge
|
||||||
|
|
||||||
|
### After (Advanced Prompt Hook)
|
||||||
|
|
||||||
|
**Configuration:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"PreToolUse": [
|
||||||
|
{
|
||||||
|
"matcher": "Bash",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "prompt",
|
||||||
|
"prompt": "Command: $TOOL_INPUT.command. Analyze for: 1) Destructive operations (rm -rf, dd, mkfs, etc) 2) Privilege escalation (sudo) 3) Network operations without user consent. Return 'approve' or 'deny' with explanation.",
|
||||||
|
"timeout": 15
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Benefits:**
|
||||||
|
- Catches all variations and patterns
|
||||||
|
- Understands intent, not just literal strings
|
||||||
|
- No script file needed
|
||||||
|
- Easy to extend with new criteria
|
||||||
|
- Context-aware decisions
|
||||||
|
- Natural language explanation in denial
|
||||||
|
|
||||||
|
## Migration Example: File Write Validation
|
||||||
|
|
||||||
|
### Before (Basic Command Hook)
|
||||||
|
|
||||||
|
**Configuration:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"PreToolUse": [
|
||||||
|
{
|
||||||
|
"matcher": "Write",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"command": "bash validate-write.sh"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Script (validate-write.sh):**
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
input=$(cat)
|
||||||
|
file_path=$(echo "$input" | jq -r '.tool_input.file_path')
|
||||||
|
|
||||||
|
# Check for path traversal
|
||||||
|
if [[ "$file_path" == *".."* ]]; then
|
||||||
|
echo '{"decision": "deny", "reason": "Path traversal detected"}' >&2
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check for system paths
|
||||||
|
if [[ "$file_path" == "/etc/"* ]] || [[ "$file_path" == "/sys/"* ]]; then
|
||||||
|
echo '{"decision": "deny", "reason": "System file"}' >&2
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
**Problems:**
|
||||||
|
- Hard-coded path patterns
|
||||||
|
- Doesn't understand symlinks
|
||||||
|
- Missing edge cases (e.g., `/etc` vs `/etc/`)
|
||||||
|
- No consideration of file content
|
||||||
|
|
||||||
|
### After (Advanced Prompt Hook)
|
||||||
|
|
||||||
|
**Configuration:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"PreToolUse": [
|
||||||
|
{
|
||||||
|
"matcher": "Write|Edit",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "prompt",
|
||||||
|
"prompt": "File path: $TOOL_INPUT.file_path. Content preview: $TOOL_INPUT.content (first 200 chars). Verify: 1) Not system directories (/etc, /sys, /usr) 2) Not credentials (.env, tokens, secrets) 3) No path traversal 4) Content doesn't expose secrets. Return 'approve' or 'deny'."
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Benefits:**
|
||||||
|
- Context-aware (considers content too)
|
||||||
|
- Handles symlinks and edge cases
|
||||||
|
- Natural understanding of "system directories"
|
||||||
|
- Can detect secrets in content
|
||||||
|
- Easy to extend criteria
|
||||||
|
|
||||||
|
## When to Keep Command Hooks
|
||||||
|
|
||||||
|
Command hooks still have their place:
|
||||||
|
|
||||||
|
### 1. Deterministic Performance Checks
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
# Check file size quickly
|
||||||
|
file_path=$(echo "$input" | jq -r '.tool_input.file_path')
|
||||||
|
size=$(stat -f%z "$file_path" 2>/dev/null || stat -c%s "$file_path" 2>/dev/null)
|
||||||
|
|
||||||
|
if [ "$size" -gt 10000000 ]; then
|
||||||
|
echo '{"decision": "deny", "reason": "File too large"}' >&2
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use command hooks when:** Validation is purely mathematical or deterministic.
|
||||||
|
|
||||||
|
### 2. External Tool Integration
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
# Run security scanner
|
||||||
|
file_path=$(echo "$input" | jq -r '.tool_input.file_path')
|
||||||
|
scan_result=$(security-scanner "$file_path")
|
||||||
|
|
||||||
|
if [ "$?" -ne 0 ]; then
|
||||||
|
echo "Security scan failed: $scan_result" >&2
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use command hooks when:** Integrating with external tools that provide yes/no answers.
|
||||||
|
|
||||||
|
### 3. Very Fast Checks (< 50ms)
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
# Quick regex check
|
||||||
|
command=$(echo "$input" | jq -r '.tool_input.command')
|
||||||
|
|
||||||
|
if [[ "$command" =~ ^(ls|pwd|echo)$ ]]; then
|
||||||
|
exit 0 # Safe commands
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use command hooks when:** Performance is critical and logic is simple.
|
||||||
|
|
||||||
|
## Hybrid Approach
|
||||||
|
|
||||||
|
Combine both for multi-stage validation:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"PreToolUse": [
|
||||||
|
{
|
||||||
|
"matcher": "Bash",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"command": "bash ${CLAUDE_PLUGIN_ROOT}/scripts/quick-check.sh",
|
||||||
|
"timeout": 5
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "prompt",
|
||||||
|
"prompt": "Deep analysis of bash command: $TOOL_INPUT",
|
||||||
|
"timeout": 15
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
The command hook does fast deterministic checks, while the prompt hook handles complex reasoning.
|
||||||
|
|
||||||
|
## Migration Checklist
|
||||||
|
|
||||||
|
When migrating hooks:
|
||||||
|
|
||||||
|
- [ ] Identify the validation logic in the command hook
|
||||||
|
- [ ] Convert hard-coded patterns to natural language criteria
|
||||||
|
- [ ] Test with edge cases the old hook missed
|
||||||
|
- [ ] Verify LLM understands the intent
|
||||||
|
- [ ] Set appropriate timeout (usually 15-30s for prompt hooks)
|
||||||
|
- [ ] Document the new hook in README
|
||||||
|
- [ ] Remove or archive old script files
|
||||||
|
|
||||||
|
## Migration Tips
|
||||||
|
|
||||||
|
1. **Start with one hook**: Don't migrate everything at once
|
||||||
|
2. **Test thoroughly**: Verify prompt hook catches what command hook caught
|
||||||
|
3. **Look for improvements**: Use migration as opportunity to enhance validation
|
||||||
|
4. **Keep scripts for reference**: Archive old scripts in case you need to reference the logic
|
||||||
|
5. **Document reasoning**: Explain why prompt hook is better in README
|
||||||
|
|
||||||
|
## Complete Migration Example
|
||||||
|
|
||||||
|
### Original Plugin Structure
|
||||||
|
|
||||||
|
```
|
||||||
|
my-plugin/
|
||||||
|
├── .claude-plugin/plugin.json
|
||||||
|
├── hooks/hooks.json
|
||||||
|
└── scripts/
|
||||||
|
├── validate-bash.sh
|
||||||
|
├── validate-write.sh
|
||||||
|
└── check-tests.sh
|
||||||
|
```
|
||||||
|
|
||||||
|
### After Migration
|
||||||
|
|
||||||
|
```
|
||||||
|
my-plugin/
|
||||||
|
├── .claude-plugin/plugin.json
|
||||||
|
├── hooks/hooks.json # Now uses prompt hooks
|
||||||
|
└── scripts/ # Archive or delete
|
||||||
|
└── archive/
|
||||||
|
├── validate-bash.sh
|
||||||
|
├── validate-write.sh
|
||||||
|
└── check-tests.sh
|
||||||
|
```
|
||||||
|
|
||||||
|
### Updated hooks.json
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"PreToolUse": [
|
||||||
|
{
|
||||||
|
"matcher": "Bash",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "prompt",
|
||||||
|
"prompt": "Validate bash command safety: destructive ops, privilege escalation, network access"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"matcher": "Write|Edit",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "prompt",
|
||||||
|
"prompt": "Validate file write safety: system paths, credentials, path traversal, content secrets"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"Stop": [
|
||||||
|
{
|
||||||
|
"matcher": "*",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "prompt",
|
||||||
|
"prompt": "Verify tests were run if code was modified"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Result:** Simpler, more maintainable, more powerful.
|
||||||
|
|
||||||
|
## Common Migration Patterns
|
||||||
|
|
||||||
|
### Pattern: String Contains → Natural Language
|
||||||
|
|
||||||
|
**Before:**
|
||||||
|
```bash
|
||||||
|
if [[ "$command" == *"sudo"* ]]; then
|
||||||
|
echo "Privilege escalation" >&2
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
**After:**
|
||||||
|
```
|
||||||
|
"Check for privilege escalation (sudo, su, etc)"
|
||||||
|
```
|
||||||
|
|
||||||
|
### Pattern: Regex → Intent
|
||||||
|
|
||||||
|
**Before:**
|
||||||
|
```bash
|
||||||
|
if [[ "$file" =~ \.(env|secret|key|token)$ ]]; then
|
||||||
|
echo "Credential file" >&2
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
**After:**
|
||||||
|
```
|
||||||
|
"Verify not writing to credential files (.env, secrets, keys, tokens)"
|
||||||
|
```
|
||||||
|
|
||||||
|
### Pattern: Multiple Conditions → Criteria List
|
||||||
|
|
||||||
|
**Before:**
|
||||||
|
```bash
|
||||||
|
if [ condition1 ] || [ condition2 ] || [ condition3 ]; then
|
||||||
|
echo "Invalid" >&2
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
**After:**
|
||||||
|
```
|
||||||
|
"Check: 1) condition1 2) condition2 3) condition3. Deny if any fail."
|
||||||
|
```
|
||||||
|
|
||||||
|
## Conclusion
|
||||||
|
|
||||||
|
Migrating to prompt-based hooks makes plugins more maintainable, flexible, and powerful. Reserve command hooks for deterministic checks and external tool integration.
|
||||||
346
skills/hook-development/references/patterns.md
Normal file
346
skills/hook-development/references/patterns.md
Normal file
@@ -0,0 +1,346 @@
|
|||||||
|
# Common Hook Patterns
|
||||||
|
|
||||||
|
This reference provides common, proven patterns for implementing Claude Code hooks. Use these patterns as starting points for typical hook use cases.
|
||||||
|
|
||||||
|
## Pattern 1: Security Validation
|
||||||
|
|
||||||
|
Block dangerous file writes using prompt-based hooks:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"PreToolUse": [
|
||||||
|
{
|
||||||
|
"matcher": "Write|Edit",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "prompt",
|
||||||
|
"prompt": "File path: $TOOL_INPUT.file_path. Verify: 1) Not in /etc or system directories 2) Not .env or credentials 3) Path doesn't contain '..' traversal. Return 'approve' or 'deny'."
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use for:** Preventing writes to sensitive files or system directories.
|
||||||
|
|
||||||
|
## Pattern 2: Test Enforcement
|
||||||
|
|
||||||
|
Ensure tests run before stopping:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"Stop": [
|
||||||
|
{
|
||||||
|
"matcher": "*",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "prompt",
|
||||||
|
"prompt": "Review transcript. If code was modified (Write/Edit tools used), verify tests were executed. If no tests were run, block with reason 'Tests must be run after code changes'."
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use for:** Enforcing quality standards and preventing incomplete work.
|
||||||
|
|
||||||
|
## Pattern 3: Context Loading
|
||||||
|
|
||||||
|
Load project-specific context at session start:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"SessionStart": [
|
||||||
|
{
|
||||||
|
"matcher": "*",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"command": "bash ${CLAUDE_PLUGIN_ROOT}/scripts/load-context.sh"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Example script (load-context.sh):**
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
cd "$CLAUDE_PROJECT_DIR" || exit 1
|
||||||
|
|
||||||
|
# Detect project type
|
||||||
|
if [ -f "package.json" ]; then
|
||||||
|
echo "📦 Node.js project detected"
|
||||||
|
echo "export PROJECT_TYPE=nodejs" >> "$CLAUDE_ENV_FILE"
|
||||||
|
elif [ -f "Cargo.toml" ]; then
|
||||||
|
echo "🦀 Rust project detected"
|
||||||
|
echo "export PROJECT_TYPE=rust" >> "$CLAUDE_ENV_FILE"
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use for:** Automatically detecting and configuring project-specific settings.
|
||||||
|
|
||||||
|
## Pattern 4: Notification Logging
|
||||||
|
|
||||||
|
Log all notifications for audit or analysis:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"Notification": [
|
||||||
|
{
|
||||||
|
"matcher": "*",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"command": "bash ${CLAUDE_PLUGIN_ROOT}/scripts/log-notification.sh"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use for:** Tracking user notifications or integration with external logging systems.
|
||||||
|
|
||||||
|
## Pattern 5: MCP Tool Monitoring
|
||||||
|
|
||||||
|
Monitor and validate MCP tool usage:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"PreToolUse": [
|
||||||
|
{
|
||||||
|
"matcher": "mcp__.*__delete.*",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "prompt",
|
||||||
|
"prompt": "Deletion operation detected. Verify: Is this deletion intentional? Can it be undone? Are there backups? Return 'approve' only if safe."
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use for:** Protecting against destructive MCP operations.
|
||||||
|
|
||||||
|
## Pattern 6: Build Verification
|
||||||
|
|
||||||
|
Ensure project builds after code changes:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"Stop": [
|
||||||
|
{
|
||||||
|
"matcher": "*",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "prompt",
|
||||||
|
"prompt": "Check if code was modified. If Write/Edit tools were used, verify the project was built (npm run build, cargo build, etc). If not built, block and request build."
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use for:** Catching build errors before committing or stopping work.
|
||||||
|
|
||||||
|
## Pattern 7: Permission Confirmation
|
||||||
|
|
||||||
|
Ask user before dangerous operations:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"PreToolUse": [
|
||||||
|
{
|
||||||
|
"matcher": "Bash",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "prompt",
|
||||||
|
"prompt": "Command: $TOOL_INPUT.command. If command contains 'rm', 'delete', 'drop', or other destructive operations, return 'ask' to confirm with user. Otherwise 'approve'."
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use for:** User confirmation on potentially destructive commands.
|
||||||
|
|
||||||
|
## Pattern 8: Code Quality Checks
|
||||||
|
|
||||||
|
Run linters or formatters on file edits:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"PostToolUse": [
|
||||||
|
{
|
||||||
|
"matcher": "Write|Edit",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"command": "bash ${CLAUDE_PLUGIN_ROOT}/scripts/check-quality.sh"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Example script (check-quality.sh):**
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
input=$(cat)
|
||||||
|
file_path=$(echo "$input" | jq -r '.tool_input.file_path')
|
||||||
|
|
||||||
|
# Run linter if applicable
|
||||||
|
if [[ "$file_path" == *.js ]] || [[ "$file_path" == *.ts ]]; then
|
||||||
|
npx eslint "$file_path" 2>&1 || true
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use for:** Automatic code quality enforcement.
|
||||||
|
|
||||||
|
## Pattern Combinations
|
||||||
|
|
||||||
|
Combine multiple patterns for comprehensive protection:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"PreToolUse": [
|
||||||
|
{
|
||||||
|
"matcher": "Write|Edit",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "prompt",
|
||||||
|
"prompt": "Validate file write safety"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"matcher": "Bash",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "prompt",
|
||||||
|
"prompt": "Validate bash command safety"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"Stop": [
|
||||||
|
{
|
||||||
|
"matcher": "*",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "prompt",
|
||||||
|
"prompt": "Verify tests run and build succeeded"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"SessionStart": [
|
||||||
|
{
|
||||||
|
"matcher": "*",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"command": "bash ${CLAUDE_PLUGIN_ROOT}/scripts/load-context.sh"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
This provides multi-layered protection and automation.
|
||||||
|
|
||||||
|
## Pattern 9: Temporarily Active Hooks
|
||||||
|
|
||||||
|
Create hooks that only run when explicitly enabled via flag files:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
# Hook only active when flag file exists
|
||||||
|
FLAG_FILE="$CLAUDE_PROJECT_DIR/.enable-security-scan"
|
||||||
|
|
||||||
|
if [ ! -f "$FLAG_FILE" ]; then
|
||||||
|
# Quick exit when disabled
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Flag present, run validation
|
||||||
|
input=$(cat)
|
||||||
|
file_path=$(echo "$input" | jq -r '.tool_input.file_path')
|
||||||
|
|
||||||
|
# Run security scan
|
||||||
|
security-scanner "$file_path"
|
||||||
|
```
|
||||||
|
|
||||||
|
**Activation:**
|
||||||
|
```bash
|
||||||
|
# Enable the hook
|
||||||
|
touch .enable-security-scan
|
||||||
|
|
||||||
|
# Disable the hook
|
||||||
|
rm .enable-security-scan
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use for:**
|
||||||
|
- Temporary debugging hooks
|
||||||
|
- Feature flags for development
|
||||||
|
- Project-specific validation that's opt-in
|
||||||
|
- Performance-intensive checks only when needed
|
||||||
|
|
||||||
|
**Note:** Must restart Claude Code after creating/removing flag files for hooks to recognize changes.
|
||||||
|
|
||||||
|
## Pattern 10: Configuration-Driven Hooks
|
||||||
|
|
||||||
|
Use JSON configuration to control hook behavior:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
CONFIG_FILE="$CLAUDE_PROJECT_DIR/.claude/my-plugin.local.json"
|
||||||
|
|
||||||
|
# Read configuration
|
||||||
|
if [ -f "$CONFIG_FILE" ]; then
|
||||||
|
strict_mode=$(jq -r '.strictMode // false' "$CONFIG_FILE")
|
||||||
|
max_file_size=$(jq -r '.maxFileSize // 1000000' "$CONFIG_FILE")
|
||||||
|
else
|
||||||
|
# Defaults
|
||||||
|
strict_mode=false
|
||||||
|
max_file_size=1000000
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Skip if not in strict mode
|
||||||
|
if [ "$strict_mode" != "true" ]; then
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Apply configured limits
|
||||||
|
input=$(cat)
|
||||||
|
file_size=$(echo "$input" | jq -r '.tool_input.content | length')
|
||||||
|
|
||||||
|
if [ "$file_size" -gt "$max_file_size" ]; then
|
||||||
|
echo '{"decision": "deny", "reason": "File exceeds configured size limit"}' >&2
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
**Configuration file (.claude/my-plugin.local.json):**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"strictMode": true,
|
||||||
|
"maxFileSize": 500000,
|
||||||
|
"allowedPaths": ["/tmp", "/home/user/projects"]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use for:**
|
||||||
|
- User-configurable hook behavior
|
||||||
|
- Per-project settings
|
||||||
|
- Team-specific rules
|
||||||
|
- Dynamic validation criteria
|
||||||
164
skills/hook-development/scripts/README.md
Normal file
164
skills/hook-development/scripts/README.md
Normal file
@@ -0,0 +1,164 @@
|
|||||||
|
# Hook Development Utility Scripts
|
||||||
|
|
||||||
|
These scripts help validate, test, and lint hook implementations before deployment.
|
||||||
|
|
||||||
|
## validate-hook-schema.sh
|
||||||
|
|
||||||
|
Validates `hooks.json` configuration files for correct structure and common issues.
|
||||||
|
|
||||||
|
**Usage:**
|
||||||
|
```bash
|
||||||
|
./validate-hook-schema.sh path/to/hooks.json
|
||||||
|
```
|
||||||
|
|
||||||
|
**Checks:**
|
||||||
|
- Valid JSON syntax
|
||||||
|
- Required fields present
|
||||||
|
- Valid hook event names
|
||||||
|
- Proper hook types (command/prompt)
|
||||||
|
- Timeout values in valid ranges
|
||||||
|
- Hardcoded path detection
|
||||||
|
- Prompt hook event compatibility
|
||||||
|
|
||||||
|
**Example:**
|
||||||
|
```bash
|
||||||
|
cd my-plugin
|
||||||
|
./validate-hook-schema.sh hooks/hooks.json
|
||||||
|
```
|
||||||
|
|
||||||
|
## test-hook.sh
|
||||||
|
|
||||||
|
Tests individual hook scripts with sample input before deploying to Claude Code.
|
||||||
|
|
||||||
|
**Usage:**
|
||||||
|
```bash
|
||||||
|
./test-hook.sh [options] <hook-script> <test-input.json>
|
||||||
|
```
|
||||||
|
|
||||||
|
**Options:**
|
||||||
|
- `-v, --verbose` - Show detailed execution information
|
||||||
|
- `-t, --timeout N` - Set timeout in seconds (default: 60)
|
||||||
|
- `--create-sample <event-type>` - Generate sample test input
|
||||||
|
|
||||||
|
**Example:**
|
||||||
|
```bash
|
||||||
|
# Create sample test input
|
||||||
|
./test-hook.sh --create-sample PreToolUse > test-input.json
|
||||||
|
|
||||||
|
# Test a hook script
|
||||||
|
./test-hook.sh my-hook.sh test-input.json
|
||||||
|
|
||||||
|
# Test with verbose output and custom timeout
|
||||||
|
./test-hook.sh -v -t 30 my-hook.sh test-input.json
|
||||||
|
```
|
||||||
|
|
||||||
|
**Features:**
|
||||||
|
- Sets up proper environment variables (CLAUDE_PROJECT_DIR, CLAUDE_PLUGIN_ROOT)
|
||||||
|
- Measures execution time
|
||||||
|
- Validates output JSON
|
||||||
|
- Shows exit codes and their meanings
|
||||||
|
- Captures environment file output
|
||||||
|
|
||||||
|
## hook-linter.sh
|
||||||
|
|
||||||
|
Checks hook scripts for common issues and best practices violations.
|
||||||
|
|
||||||
|
**Usage:**
|
||||||
|
```bash
|
||||||
|
./hook-linter.sh <hook-script.sh> [hook-script2.sh ...]
|
||||||
|
```
|
||||||
|
|
||||||
|
**Checks:**
|
||||||
|
- Shebang presence
|
||||||
|
- `set -euo pipefail` usage
|
||||||
|
- Stdin input reading
|
||||||
|
- Proper error handling
|
||||||
|
- Variable quoting (injection prevention)
|
||||||
|
- Exit code usage
|
||||||
|
- Hardcoded paths
|
||||||
|
- Long-running code detection
|
||||||
|
- Error output to stderr
|
||||||
|
- Input validation
|
||||||
|
|
||||||
|
**Example:**
|
||||||
|
```bash
|
||||||
|
# Lint single script
|
||||||
|
./hook-linter.sh ../examples/validate-write.sh
|
||||||
|
|
||||||
|
# Lint multiple scripts
|
||||||
|
./hook-linter.sh ../examples/*.sh
|
||||||
|
```
|
||||||
|
|
||||||
|
## Typical Workflow
|
||||||
|
|
||||||
|
1. **Write your hook script**
|
||||||
|
```bash
|
||||||
|
vim my-plugin/scripts/my-hook.sh
|
||||||
|
```
|
||||||
|
|
||||||
|
2. **Lint the script**
|
||||||
|
```bash
|
||||||
|
./hook-linter.sh my-plugin/scripts/my-hook.sh
|
||||||
|
```
|
||||||
|
|
||||||
|
3. **Create test input**
|
||||||
|
```bash
|
||||||
|
./test-hook.sh --create-sample PreToolUse > test-input.json
|
||||||
|
# Edit test-input.json as needed
|
||||||
|
```
|
||||||
|
|
||||||
|
4. **Test the hook**
|
||||||
|
```bash
|
||||||
|
./test-hook.sh -v my-plugin/scripts/my-hook.sh test-input.json
|
||||||
|
```
|
||||||
|
|
||||||
|
5. **Add to hooks.json**
|
||||||
|
```bash
|
||||||
|
# Edit my-plugin/hooks/hooks.json
|
||||||
|
```
|
||||||
|
|
||||||
|
6. **Validate configuration**
|
||||||
|
```bash
|
||||||
|
./validate-hook-schema.sh my-plugin/hooks/hooks.json
|
||||||
|
```
|
||||||
|
|
||||||
|
7. **Test in Claude Code**
|
||||||
|
```bash
|
||||||
|
claude --debug
|
||||||
|
```
|
||||||
|
|
||||||
|
## Tips
|
||||||
|
|
||||||
|
- Always test hooks before deploying to avoid breaking user workflows
|
||||||
|
- Use verbose mode (`-v`) to debug hook behavior
|
||||||
|
- Check the linter output for security and best practice issues
|
||||||
|
- Validate hooks.json after any changes
|
||||||
|
- Create different test inputs for various scenarios (safe operations, dangerous operations, edge cases)
|
||||||
|
|
||||||
|
## Common Issues
|
||||||
|
|
||||||
|
### Hook doesn't execute
|
||||||
|
|
||||||
|
Check:
|
||||||
|
- Script has shebang (`#!/bin/bash`)
|
||||||
|
- Script is executable (`chmod +x`)
|
||||||
|
- Path in hooks.json is correct (use `${CLAUDE_PLUGIN_ROOT}`)
|
||||||
|
|
||||||
|
### Hook times out
|
||||||
|
|
||||||
|
- Reduce timeout in hooks.json
|
||||||
|
- Optimize hook script performance
|
||||||
|
- Remove long-running operations
|
||||||
|
|
||||||
|
### Hook fails silently
|
||||||
|
|
||||||
|
- Check exit codes (should be 0 or 2)
|
||||||
|
- Ensure errors go to stderr (`>&2`)
|
||||||
|
- Validate JSON output structure
|
||||||
|
|
||||||
|
### Injection vulnerabilities
|
||||||
|
|
||||||
|
- Always quote variables: `"$variable"`
|
||||||
|
- Use `set -euo pipefail`
|
||||||
|
- Validate all input fields
|
||||||
|
- Run the linter to catch issues
|
||||||
153
skills/hook-development/scripts/hook-linter.sh
Executable file
153
skills/hook-development/scripts/hook-linter.sh
Executable file
@@ -0,0 +1,153 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
# Hook Linter
|
||||||
|
# Checks hook scripts for common issues and best practices
|
||||||
|
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
# Usage
|
||||||
|
if [ $# -eq 0 ]; then
|
||||||
|
echo "Usage: $0 <hook-script.sh> [hook-script2.sh ...]"
|
||||||
|
echo ""
|
||||||
|
echo "Checks hook scripts for:"
|
||||||
|
echo " - Shebang presence"
|
||||||
|
echo " - set -euo pipefail usage"
|
||||||
|
echo " - Input reading from stdin"
|
||||||
|
echo " - Proper error handling"
|
||||||
|
echo " - Variable quoting"
|
||||||
|
echo " - Exit code usage"
|
||||||
|
echo " - Hardcoded paths"
|
||||||
|
echo " - Timeout considerations"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
check_script() {
|
||||||
|
local script="$1"
|
||||||
|
local warnings=0
|
||||||
|
local errors=0
|
||||||
|
|
||||||
|
echo "🔍 Linting: $script"
|
||||||
|
echo ""
|
||||||
|
|
||||||
|
if [ ! -f "$script" ]; then
|
||||||
|
echo "❌ Error: File not found"
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check 1: Executable
|
||||||
|
if [ ! -x "$script" ]; then
|
||||||
|
echo "⚠️ Not executable (chmod +x $script)"
|
||||||
|
((warnings++))
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check 2: Shebang
|
||||||
|
first_line=$(head -1 "$script")
|
||||||
|
if [[ ! "$first_line" =~ ^#!/ ]]; then
|
||||||
|
echo "❌ Missing shebang (#!/bin/bash)"
|
||||||
|
((errors++))
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check 3: set -euo pipefail
|
||||||
|
if ! grep -q "set -euo pipefail" "$script"; then
|
||||||
|
echo "⚠️ Missing 'set -euo pipefail' (recommended for safety)"
|
||||||
|
((warnings++))
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check 4: Reads from stdin
|
||||||
|
if ! grep -q "cat\|read" "$script"; then
|
||||||
|
echo "⚠️ Doesn't appear to read input from stdin"
|
||||||
|
((warnings++))
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check 5: Uses jq for JSON parsing
|
||||||
|
if grep -q "tool_input\|tool_name" "$script" && ! grep -q "jq" "$script"; then
|
||||||
|
echo "⚠️ Parses hook input but doesn't use jq"
|
||||||
|
((warnings++))
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check 6: Unquoted variables
|
||||||
|
if grep -E '\$[A-Za-z_][A-Za-z0-9_]*[^"]' "$script" | grep -v '#' | grep -q .; then
|
||||||
|
echo "⚠️ Potentially unquoted variables detected (injection risk)"
|
||||||
|
echo " Always use double quotes: \"\$variable\" not \$variable"
|
||||||
|
((warnings++))
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check 7: Hardcoded paths
|
||||||
|
if grep -E '^[^#]*/home/|^[^#]*/usr/|^[^#]*/opt/' "$script" | grep -q .; then
|
||||||
|
echo "⚠️ Hardcoded absolute paths detected"
|
||||||
|
echo " Use \$CLAUDE_PROJECT_DIR or \$CLAUDE_PLUGIN_ROOT"
|
||||||
|
((warnings++))
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check 8: Uses CLAUDE_PLUGIN_ROOT
|
||||||
|
if ! grep -q "CLAUDE_PLUGIN_ROOT\|CLAUDE_PROJECT_DIR" "$script"; then
|
||||||
|
echo "💡 Tip: Use \$CLAUDE_PLUGIN_ROOT for plugin-relative paths"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check 9: Exit codes
|
||||||
|
if ! grep -q "exit 0\|exit 2" "$script"; then
|
||||||
|
echo "⚠️ No explicit exit codes (should exit 0 or 2)"
|
||||||
|
((warnings++))
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check 10: JSON output for decision hooks
|
||||||
|
if grep -q "PreToolUse\|Stop" "$script"; then
|
||||||
|
if ! grep -q "permissionDecision\|decision" "$script"; then
|
||||||
|
echo "💡 Tip: PreToolUse/Stop hooks should output decision JSON"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check 11: Long-running commands
|
||||||
|
if grep -E 'sleep [0-9]{3,}|while true' "$script" | grep -v '#' | grep -q .; then
|
||||||
|
echo "⚠️ Potentially long-running code detected"
|
||||||
|
echo " Hooks should complete quickly (< 60s)"
|
||||||
|
((warnings++))
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check 12: Error messages to stderr
|
||||||
|
if grep -q 'echo.*".*error\|Error\|denied\|Denied' "$script"; then
|
||||||
|
if ! grep -q '>&2' "$script"; then
|
||||||
|
echo "⚠️ Error messages should be written to stderr (>&2)"
|
||||||
|
((warnings++))
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check 13: Input validation
|
||||||
|
if ! grep -q "if.*empty\|if.*null\|if.*-z" "$script"; then
|
||||||
|
echo "💡 Tip: Consider validating input fields aren't empty"
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo ""
|
||||||
|
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
|
||||||
|
|
||||||
|
if [ $errors -eq 0 ] && [ $warnings -eq 0 ]; then
|
||||||
|
echo "✅ No issues found"
|
||||||
|
return 0
|
||||||
|
elif [ $errors -eq 0 ]; then
|
||||||
|
echo "⚠️ Found $warnings warning(s)"
|
||||||
|
return 0
|
||||||
|
else
|
||||||
|
echo "❌ Found $errors error(s) and $warnings warning(s)"
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
echo "🔎 Hook Script Linter"
|
||||||
|
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
|
||||||
|
echo ""
|
||||||
|
|
||||||
|
total_errors=0
|
||||||
|
|
||||||
|
for script in "$@"; do
|
||||||
|
if ! check_script "$script"; then
|
||||||
|
((total_errors++))
|
||||||
|
fi
|
||||||
|
echo ""
|
||||||
|
done
|
||||||
|
|
||||||
|
if [ $total_errors -eq 0 ]; then
|
||||||
|
echo "✅ All scripts passed linting"
|
||||||
|
exit 0
|
||||||
|
else
|
||||||
|
echo "❌ $total_errors script(s) had errors"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
252
skills/hook-development/scripts/test-hook.sh
Executable file
252
skills/hook-development/scripts/test-hook.sh
Executable file
@@ -0,0 +1,252 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
# Hook Testing Helper
|
||||||
|
# Tests a hook with sample input and shows output
|
||||||
|
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
# Usage
|
||||||
|
show_usage() {
|
||||||
|
echo "Usage: $0 [options] <hook-script> <test-input.json>"
|
||||||
|
echo ""
|
||||||
|
echo "Options:"
|
||||||
|
echo " -h, --help Show this help message"
|
||||||
|
echo " -v, --verbose Show detailed execution information"
|
||||||
|
echo " -t, --timeout N Set timeout in seconds (default: 60)"
|
||||||
|
echo ""
|
||||||
|
echo "Examples:"
|
||||||
|
echo " $0 validate-bash.sh test-input.json"
|
||||||
|
echo " $0 -v -t 30 validate-write.sh write-input.json"
|
||||||
|
echo ""
|
||||||
|
echo "Creates sample test input with:"
|
||||||
|
echo " $0 --create-sample <event-type>"
|
||||||
|
exit 0
|
||||||
|
}
|
||||||
|
|
||||||
|
# Create sample input
|
||||||
|
create_sample() {
|
||||||
|
event_type="$1"
|
||||||
|
|
||||||
|
case "$event_type" in
|
||||||
|
PreToolUse)
|
||||||
|
cat <<'EOF'
|
||||||
|
{
|
||||||
|
"session_id": "test-session",
|
||||||
|
"transcript_path": "/tmp/transcript.txt",
|
||||||
|
"cwd": "/tmp/test-project",
|
||||||
|
"permission_mode": "ask",
|
||||||
|
"hook_event_name": "PreToolUse",
|
||||||
|
"tool_name": "Write",
|
||||||
|
"tool_input": {
|
||||||
|
"file_path": "/tmp/test.txt",
|
||||||
|
"content": "Test content"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
EOF
|
||||||
|
;;
|
||||||
|
PostToolUse)
|
||||||
|
cat <<'EOF'
|
||||||
|
{
|
||||||
|
"session_id": "test-session",
|
||||||
|
"transcript_path": "/tmp/transcript.txt",
|
||||||
|
"cwd": "/tmp/test-project",
|
||||||
|
"permission_mode": "ask",
|
||||||
|
"hook_event_name": "PostToolUse",
|
||||||
|
"tool_name": "Bash",
|
||||||
|
"tool_result": "Command executed successfully"
|
||||||
|
}
|
||||||
|
EOF
|
||||||
|
;;
|
||||||
|
Stop|SubagentStop)
|
||||||
|
cat <<'EOF'
|
||||||
|
{
|
||||||
|
"session_id": "test-session",
|
||||||
|
"transcript_path": "/tmp/transcript.txt",
|
||||||
|
"cwd": "/tmp/test-project",
|
||||||
|
"permission_mode": "ask",
|
||||||
|
"hook_event_name": "Stop",
|
||||||
|
"reason": "Task appears complete"
|
||||||
|
}
|
||||||
|
EOF
|
||||||
|
;;
|
||||||
|
UserPromptSubmit)
|
||||||
|
cat <<'EOF'
|
||||||
|
{
|
||||||
|
"session_id": "test-session",
|
||||||
|
"transcript_path": "/tmp/transcript.txt",
|
||||||
|
"cwd": "/tmp/test-project",
|
||||||
|
"permission_mode": "ask",
|
||||||
|
"hook_event_name": "UserPromptSubmit",
|
||||||
|
"user_prompt": "Test user prompt"
|
||||||
|
}
|
||||||
|
EOF
|
||||||
|
;;
|
||||||
|
SessionStart|SessionEnd)
|
||||||
|
cat <<'EOF'
|
||||||
|
{
|
||||||
|
"session_id": "test-session",
|
||||||
|
"transcript_path": "/tmp/transcript.txt",
|
||||||
|
"cwd": "/tmp/test-project",
|
||||||
|
"permission_mode": "ask",
|
||||||
|
"hook_event_name": "SessionStart"
|
||||||
|
}
|
||||||
|
EOF
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
echo "Unknown event type: $event_type"
|
||||||
|
echo "Valid types: PreToolUse, PostToolUse, Stop, SubagentStop, UserPromptSubmit, SessionStart, SessionEnd"
|
||||||
|
exit 1
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
}
|
||||||
|
|
||||||
|
# Parse arguments
|
||||||
|
VERBOSE=false
|
||||||
|
TIMEOUT=60
|
||||||
|
|
||||||
|
while [ $# -gt 0 ]; do
|
||||||
|
case "$1" in
|
||||||
|
-h|--help)
|
||||||
|
show_usage
|
||||||
|
;;
|
||||||
|
-v|--verbose)
|
||||||
|
VERBOSE=true
|
||||||
|
shift
|
||||||
|
;;
|
||||||
|
-t|--timeout)
|
||||||
|
TIMEOUT="$2"
|
||||||
|
shift 2
|
||||||
|
;;
|
||||||
|
--create-sample)
|
||||||
|
create_sample "$2"
|
||||||
|
exit 0
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
break
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
done
|
||||||
|
|
||||||
|
if [ $# -ne 2 ]; then
|
||||||
|
echo "Error: Missing required arguments"
|
||||||
|
echo ""
|
||||||
|
show_usage
|
||||||
|
fi
|
||||||
|
|
||||||
|
HOOK_SCRIPT="$1"
|
||||||
|
TEST_INPUT="$2"
|
||||||
|
|
||||||
|
# Validate inputs
|
||||||
|
if [ ! -f "$HOOK_SCRIPT" ]; then
|
||||||
|
echo "❌ Error: Hook script not found: $HOOK_SCRIPT"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ ! -x "$HOOK_SCRIPT" ]; then
|
||||||
|
echo "⚠️ Warning: Hook script is not executable. Attempting to run with bash..."
|
||||||
|
HOOK_SCRIPT="bash $HOOK_SCRIPT"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ ! -f "$TEST_INPUT" ]; then
|
||||||
|
echo "❌ Error: Test input not found: $TEST_INPUT"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Validate test input JSON
|
||||||
|
if ! jq empty "$TEST_INPUT" 2>/dev/null; then
|
||||||
|
echo "❌ Error: Test input is not valid JSON"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "🧪 Testing hook: $HOOK_SCRIPT"
|
||||||
|
echo "📥 Input: $TEST_INPUT"
|
||||||
|
echo ""
|
||||||
|
|
||||||
|
if [ "$VERBOSE" = true ]; then
|
||||||
|
echo "Input JSON:"
|
||||||
|
jq . "$TEST_INPUT"
|
||||||
|
echo ""
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Set up environment
|
||||||
|
export CLAUDE_PROJECT_DIR="${CLAUDE_PROJECT_DIR:-/tmp/test-project}"
|
||||||
|
export CLAUDE_PLUGIN_ROOT="${CLAUDE_PLUGIN_ROOT:-$(pwd)}"
|
||||||
|
export CLAUDE_ENV_FILE="${CLAUDE_ENV_FILE:-/tmp/test-env-$$}"
|
||||||
|
|
||||||
|
if [ "$VERBOSE" = true ]; then
|
||||||
|
echo "Environment:"
|
||||||
|
echo " CLAUDE_PROJECT_DIR=$CLAUDE_PROJECT_DIR"
|
||||||
|
echo " CLAUDE_PLUGIN_ROOT=$CLAUDE_PLUGIN_ROOT"
|
||||||
|
echo " CLAUDE_ENV_FILE=$CLAUDE_ENV_FILE"
|
||||||
|
echo ""
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Run the hook
|
||||||
|
echo "▶️ Running hook (timeout: ${TIMEOUT}s)..."
|
||||||
|
echo ""
|
||||||
|
|
||||||
|
start_time=$(date +%s)
|
||||||
|
|
||||||
|
set +e
|
||||||
|
output=$(timeout "$TIMEOUT" bash -c "cat '$TEST_INPUT' | $HOOK_SCRIPT" 2>&1)
|
||||||
|
exit_code=$?
|
||||||
|
set -e
|
||||||
|
|
||||||
|
end_time=$(date +%s)
|
||||||
|
duration=$((end_time - start_time))
|
||||||
|
|
||||||
|
# Analyze results
|
||||||
|
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
|
||||||
|
echo "Results:"
|
||||||
|
echo ""
|
||||||
|
echo "Exit Code: $exit_code"
|
||||||
|
echo "Duration: ${duration}s"
|
||||||
|
echo ""
|
||||||
|
|
||||||
|
case $exit_code in
|
||||||
|
0)
|
||||||
|
echo "✅ Hook approved/succeeded"
|
||||||
|
;;
|
||||||
|
2)
|
||||||
|
echo "🚫 Hook blocked/denied"
|
||||||
|
;;
|
||||||
|
124)
|
||||||
|
echo "⏱️ Hook timed out after ${TIMEOUT}s"
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
echo "⚠️ Hook returned unexpected exit code: $exit_code"
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
echo ""
|
||||||
|
echo "Output:"
|
||||||
|
if [ -n "$output" ]; then
|
||||||
|
echo "$output"
|
||||||
|
echo ""
|
||||||
|
|
||||||
|
# Try to parse as JSON
|
||||||
|
if echo "$output" | jq empty 2>/dev/null; then
|
||||||
|
echo "Parsed JSON output:"
|
||||||
|
echo "$output" | jq .
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
echo "(no output)"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check for environment file
|
||||||
|
if [ -f "$CLAUDE_ENV_FILE" ]; then
|
||||||
|
echo ""
|
||||||
|
echo "Environment file created:"
|
||||||
|
cat "$CLAUDE_ENV_FILE"
|
||||||
|
rm -f "$CLAUDE_ENV_FILE"
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo ""
|
||||||
|
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
|
||||||
|
|
||||||
|
if [ $exit_code -eq 0 ] || [ $exit_code -eq 2 ]; then
|
||||||
|
echo "✅ Test completed successfully"
|
||||||
|
exit 0
|
||||||
|
else
|
||||||
|
echo "❌ Test failed"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
159
skills/hook-development/scripts/validate-hook-schema.sh
Executable file
159
skills/hook-development/scripts/validate-hook-schema.sh
Executable file
@@ -0,0 +1,159 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
# Hook Schema Validator
|
||||||
|
# Validates hooks.json structure and checks for common issues
|
||||||
|
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
# Usage
|
||||||
|
if [ $# -eq 0 ]; then
|
||||||
|
echo "Usage: $0 <path/to/hooks.json>"
|
||||||
|
echo ""
|
||||||
|
echo "Validates hook configuration file for:"
|
||||||
|
echo " - Valid JSON syntax"
|
||||||
|
echo " - Required fields"
|
||||||
|
echo " - Hook type validity"
|
||||||
|
echo " - Matcher patterns"
|
||||||
|
echo " - Timeout ranges"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
HOOKS_FILE="$1"
|
||||||
|
|
||||||
|
if [ ! -f "$HOOKS_FILE" ]; then
|
||||||
|
echo "❌ Error: File not found: $HOOKS_FILE"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "🔍 Validating hooks configuration: $HOOKS_FILE"
|
||||||
|
echo ""
|
||||||
|
|
||||||
|
# Check 1: Valid JSON
|
||||||
|
echo "Checking JSON syntax..."
|
||||||
|
if ! jq empty "$HOOKS_FILE" 2>/dev/null; then
|
||||||
|
echo "❌ Invalid JSON syntax"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
echo "✅ Valid JSON"
|
||||||
|
|
||||||
|
# Check 2: Root structure
|
||||||
|
echo ""
|
||||||
|
echo "Checking root structure..."
|
||||||
|
VALID_EVENTS=("PreToolUse" "PostToolUse" "UserPromptSubmit" "Stop" "SubagentStop" "SessionStart" "SessionEnd" "PreCompact" "Notification")
|
||||||
|
|
||||||
|
for event in $(jq -r 'keys[]' "$HOOKS_FILE"); do
|
||||||
|
found=false
|
||||||
|
for valid_event in "${VALID_EVENTS[@]}"; do
|
||||||
|
if [ "$event" = "$valid_event" ]; then
|
||||||
|
found=true
|
||||||
|
break
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
|
||||||
|
if [ "$found" = false ]; then
|
||||||
|
echo "⚠️ Unknown event type: $event"
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
echo "✅ Root structure valid"
|
||||||
|
|
||||||
|
# Check 3: Validate each hook
|
||||||
|
echo ""
|
||||||
|
echo "Validating individual hooks..."
|
||||||
|
|
||||||
|
error_count=0
|
||||||
|
warning_count=0
|
||||||
|
|
||||||
|
for event in $(jq -r 'keys[]' "$HOOKS_FILE"); do
|
||||||
|
hook_count=$(jq -r ".\"$event\" | length" "$HOOKS_FILE")
|
||||||
|
|
||||||
|
for ((i=0; i<hook_count; i++)); do
|
||||||
|
# Check matcher exists
|
||||||
|
matcher=$(jq -r ".\"$event\"[$i].matcher // empty" "$HOOKS_FILE")
|
||||||
|
if [ -z "$matcher" ]; then
|
||||||
|
echo "❌ $event[$i]: Missing 'matcher' field"
|
||||||
|
((error_count++))
|
||||||
|
continue
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check hooks array exists
|
||||||
|
hooks=$(jq -r ".\"$event\"[$i].hooks // empty" "$HOOKS_FILE")
|
||||||
|
if [ -z "$hooks" ] || [ "$hooks" = "null" ]; then
|
||||||
|
echo "❌ $event[$i]: Missing 'hooks' array"
|
||||||
|
((error_count++))
|
||||||
|
continue
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Validate each hook in the array
|
||||||
|
hook_array_count=$(jq -r ".\"$event\"[$i].hooks | length" "$HOOKS_FILE")
|
||||||
|
|
||||||
|
for ((j=0; j<hook_array_count; j++)); do
|
||||||
|
hook_type=$(jq -r ".\"$event\"[$i].hooks[$j].type // empty" "$HOOKS_FILE")
|
||||||
|
|
||||||
|
if [ -z "$hook_type" ]; then
|
||||||
|
echo "❌ $event[$i].hooks[$j]: Missing 'type' field"
|
||||||
|
((error_count++))
|
||||||
|
continue
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ "$hook_type" != "command" ] && [ "$hook_type" != "prompt" ]; then
|
||||||
|
echo "❌ $event[$i].hooks[$j]: Invalid type '$hook_type' (must be 'command' or 'prompt')"
|
||||||
|
((error_count++))
|
||||||
|
continue
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check type-specific fields
|
||||||
|
if [ "$hook_type" = "command" ]; then
|
||||||
|
command=$(jq -r ".\"$event\"[$i].hooks[$j].command // empty" "$HOOKS_FILE")
|
||||||
|
if [ -z "$command" ]; then
|
||||||
|
echo "❌ $event[$i].hooks[$j]: Command hooks must have 'command' field"
|
||||||
|
((error_count++))
|
||||||
|
else
|
||||||
|
# Check for hardcoded paths
|
||||||
|
if [[ "$command" == /* ]] && [[ "$command" != *'${CLAUDE_PLUGIN_ROOT}'* ]]; then
|
||||||
|
echo "⚠️ $event[$i].hooks[$j]: Hardcoded absolute path detected. Consider using \${CLAUDE_PLUGIN_ROOT}"
|
||||||
|
((warning_count++))
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
elif [ "$hook_type" = "prompt" ]; then
|
||||||
|
prompt=$(jq -r ".\"$event\"[$i].hooks[$j].prompt // empty" "$HOOKS_FILE")
|
||||||
|
if [ -z "$prompt" ]; then
|
||||||
|
echo "❌ $event[$i].hooks[$j]: Prompt hooks must have 'prompt' field"
|
||||||
|
((error_count++))
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check if prompt-based hooks are used on supported events
|
||||||
|
if [ "$event" != "Stop" ] && [ "$event" != "SubagentStop" ] && [ "$event" != "UserPromptSubmit" ] && [ "$event" != "PreToolUse" ]; then
|
||||||
|
echo "⚠️ $event[$i].hooks[$j]: Prompt hooks may not be fully supported on $event (best on Stop, SubagentStop, UserPromptSubmit, PreToolUse)"
|
||||||
|
((warning_count++))
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check timeout
|
||||||
|
timeout=$(jq -r ".\"$event\"[$i].hooks[$j].timeout // empty" "$HOOKS_FILE")
|
||||||
|
if [ -n "$timeout" ] && [ "$timeout" != "null" ]; then
|
||||||
|
if ! [[ "$timeout" =~ ^[0-9]+$ ]]; then
|
||||||
|
echo "❌ $event[$i].hooks[$j]: Timeout must be a number"
|
||||||
|
((error_count++))
|
||||||
|
elif [ "$timeout" -gt 600 ]; then
|
||||||
|
echo "⚠️ $event[$i].hooks[$j]: Timeout $timeout seconds is very high (max 600s)"
|
||||||
|
((warning_count++))
|
||||||
|
elif [ "$timeout" -lt 5 ]; then
|
||||||
|
echo "⚠️ $event[$i].hooks[$j]: Timeout $timeout seconds is very low"
|
||||||
|
((warning_count++))
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
done
|
||||||
|
done
|
||||||
|
|
||||||
|
echo ""
|
||||||
|
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
|
||||||
|
if [ $error_count -eq 0 ] && [ $warning_count -eq 0 ]; then
|
||||||
|
echo "✅ All checks passed!"
|
||||||
|
exit 0
|
||||||
|
elif [ $error_count -eq 0 ]; then
|
||||||
|
echo "⚠️ Validation passed with $warning_count warning(s)"
|
||||||
|
exit 0
|
||||||
|
else
|
||||||
|
echo "❌ Validation failed with $error_count error(s) and $warning_count warning(s)"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
553
skills/mcp-integration/SKILL.md
Normal file
553
skills/mcp-integration/SKILL.md
Normal file
@@ -0,0 +1,553 @@
|
|||||||
|
---
|
||||||
|
name: mcp-integration
|
||||||
|
description: This skill should be used when the user asks to "add MCP server", "integrate MCP", "configure MCP in plugin", "use .mcp.json", "set up Model Context Protocol", "connect external service", mentions "${CLAUDE_PLUGIN_ROOT} with MCP", or discusses MCP server types (SSE, stdio, HTTP, WebSocket). Provides comprehensive guidance for integrating Model Context Protocol servers into Claude Code plugins for external tool and service integration.
|
||||||
|
---
|
||||||
|
|
||||||
|
# MCP Integration for Claude Code Plugins
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
|
||||||
|
Model Context Protocol (MCP) enables Claude Code plugins to integrate with external services and APIs by providing structured tool access. Use MCP integration to expose external service capabilities as tools within Claude Code.
|
||||||
|
|
||||||
|
**Key capabilities:**
|
||||||
|
- Connect to external services (databases, APIs, file systems)
|
||||||
|
- Provide 10+ related tools from a single service
|
||||||
|
- Handle OAuth and complex authentication flows
|
||||||
|
- Bundle MCP servers with plugins for automatic setup
|
||||||
|
|
||||||
|
## MCP Server Configuration Methods
|
||||||
|
|
||||||
|
Plugins can bundle MCP servers in two ways:
|
||||||
|
|
||||||
|
### Method 1: Dedicated .mcp.json (Recommended)
|
||||||
|
|
||||||
|
Create `.mcp.json` at plugin root:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"database-tools": {
|
||||||
|
"command": "${CLAUDE_PLUGIN_ROOT}/servers/db-server",
|
||||||
|
"args": ["--config", "${CLAUDE_PLUGIN_ROOT}/config.json"],
|
||||||
|
"env": {
|
||||||
|
"DB_URL": "${DB_URL}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Benefits:**
|
||||||
|
- Clear separation of concerns
|
||||||
|
- Easier to maintain
|
||||||
|
- Better for multiple servers
|
||||||
|
|
||||||
|
### Method 2: Inline in plugin.json
|
||||||
|
|
||||||
|
Add `mcpServers` field to plugin.json:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"name": "my-plugin",
|
||||||
|
"version": "1.0.0",
|
||||||
|
"mcpServers": {
|
||||||
|
"plugin-api": {
|
||||||
|
"command": "${CLAUDE_PLUGIN_ROOT}/servers/api-server",
|
||||||
|
"args": ["--port", "8080"]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Benefits:**
|
||||||
|
- Single configuration file
|
||||||
|
- Good for simple single-server plugins
|
||||||
|
|
||||||
|
## MCP Server Types
|
||||||
|
|
||||||
|
### stdio (Local Process)
|
||||||
|
|
||||||
|
Execute local MCP servers as child processes. Best for local tools and custom servers.
|
||||||
|
|
||||||
|
**Configuration:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"filesystem": {
|
||||||
|
"command": "npx",
|
||||||
|
"args": ["-y", "@modelcontextprotocol/server-filesystem", "/allowed/path"],
|
||||||
|
"env": {
|
||||||
|
"LOG_LEVEL": "debug"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use cases:**
|
||||||
|
- File system access
|
||||||
|
- Local database connections
|
||||||
|
- Custom MCP servers
|
||||||
|
- NPM-packaged MCP servers
|
||||||
|
|
||||||
|
**Process management:**
|
||||||
|
- Claude Code spawns and manages the process
|
||||||
|
- Communicates via stdin/stdout
|
||||||
|
- Terminates when Claude Code exits
|
||||||
|
|
||||||
|
### SSE (Server-Sent Events)
|
||||||
|
|
||||||
|
Connect to hosted MCP servers with OAuth support. Best for cloud services.
|
||||||
|
|
||||||
|
**Configuration:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"asana": {
|
||||||
|
"type": "sse",
|
||||||
|
"url": "https://mcp.asana.com/sse"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use cases:**
|
||||||
|
- Official hosted MCP servers (Asana, GitHub, etc.)
|
||||||
|
- Cloud services with MCP endpoints
|
||||||
|
- OAuth-based authentication
|
||||||
|
- No local installation needed
|
||||||
|
|
||||||
|
**Authentication:**
|
||||||
|
- OAuth flows handled automatically
|
||||||
|
- User prompted on first use
|
||||||
|
- Tokens managed by Claude Code
|
||||||
|
|
||||||
|
### HTTP (REST API)
|
||||||
|
|
||||||
|
Connect to RESTful MCP servers with token authentication.
|
||||||
|
|
||||||
|
**Configuration:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"api-service": {
|
||||||
|
"type": "http",
|
||||||
|
"url": "https://api.example.com/mcp",
|
||||||
|
"headers": {
|
||||||
|
"Authorization": "Bearer ${API_TOKEN}",
|
||||||
|
"X-Custom-Header": "value"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use cases:**
|
||||||
|
- REST API-based MCP servers
|
||||||
|
- Token-based authentication
|
||||||
|
- Custom API backends
|
||||||
|
- Stateless interactions
|
||||||
|
|
||||||
|
### WebSocket (Real-time)
|
||||||
|
|
||||||
|
Connect to WebSocket MCP servers for real-time bidirectional communication.
|
||||||
|
|
||||||
|
**Configuration:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"realtime-service": {
|
||||||
|
"type": "ws",
|
||||||
|
"url": "wss://mcp.example.com/ws",
|
||||||
|
"headers": {
|
||||||
|
"Authorization": "Bearer ${TOKEN}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use cases:**
|
||||||
|
- Real-time data streaming
|
||||||
|
- Persistent connections
|
||||||
|
- Push notifications from server
|
||||||
|
- Low-latency requirements
|
||||||
|
|
||||||
|
## Environment Variable Expansion
|
||||||
|
|
||||||
|
All MCP configurations support environment variable substitution:
|
||||||
|
|
||||||
|
**${CLAUDE_PLUGIN_ROOT}** - Plugin directory (always use for portability):
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"command": "${CLAUDE_PLUGIN_ROOT}/servers/my-server"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**User environment variables** - From user's shell:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"env": {
|
||||||
|
"API_KEY": "${MY_API_KEY}",
|
||||||
|
"DATABASE_URL": "${DB_URL}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Best practice:** Document all required environment variables in plugin README.
|
||||||
|
|
||||||
|
## MCP Tool Naming
|
||||||
|
|
||||||
|
When MCP servers provide tools, they're automatically prefixed:
|
||||||
|
|
||||||
|
**Format:** `mcp__plugin_<plugin-name>_<server-name>__<tool-name>`
|
||||||
|
|
||||||
|
**Example:**
|
||||||
|
- Plugin: `asana`
|
||||||
|
- Server: `asana`
|
||||||
|
- Tool: `create_task`
|
||||||
|
- **Full name:** `mcp__plugin_asana_asana__asana_create_task`
|
||||||
|
|
||||||
|
### Using MCP Tools in Commands
|
||||||
|
|
||||||
|
Pre-allow specific MCP tools in command frontmatter:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
allowed-tools: [
|
||||||
|
"mcp__plugin_asana_asana__asana_create_task",
|
||||||
|
"mcp__plugin_asana_asana__asana_search_tasks"
|
||||||
|
]
|
||||||
|
---
|
||||||
|
```
|
||||||
|
|
||||||
|
**Wildcard (use sparingly):**
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
allowed-tools: ["mcp__plugin_asana_asana__*"]
|
||||||
|
---
|
||||||
|
```
|
||||||
|
|
||||||
|
**Best practice:** Pre-allow specific tools, not wildcards, for security.
|
||||||
|
|
||||||
|
## Lifecycle Management
|
||||||
|
|
||||||
|
**Automatic startup:**
|
||||||
|
- MCP servers start when plugin enables
|
||||||
|
- Connection established before first tool use
|
||||||
|
- Restart required for configuration changes
|
||||||
|
|
||||||
|
**Lifecycle:**
|
||||||
|
1. Plugin loads
|
||||||
|
2. MCP configuration parsed
|
||||||
|
3. Server process started (stdio) or connection established (SSE/HTTP/WS)
|
||||||
|
4. Tools discovered and registered
|
||||||
|
5. Tools available as `mcp__plugin_...__...`
|
||||||
|
|
||||||
|
**Viewing servers:**
|
||||||
|
Use `/mcp` command to see all servers including plugin-provided ones.
|
||||||
|
|
||||||
|
## Authentication Patterns
|
||||||
|
|
||||||
|
### OAuth (SSE/HTTP)
|
||||||
|
|
||||||
|
OAuth handled automatically by Claude Code:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"type": "sse",
|
||||||
|
"url": "https://mcp.example.com/sse"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
User authenticates in browser on first use. No additional configuration needed.
|
||||||
|
|
||||||
|
### Token-Based (Headers)
|
||||||
|
|
||||||
|
Static or environment variable tokens:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"type": "http",
|
||||||
|
"url": "https://api.example.com",
|
||||||
|
"headers": {
|
||||||
|
"Authorization": "Bearer ${API_TOKEN}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Document required environment variables in README.
|
||||||
|
|
||||||
|
### Environment Variables (stdio)
|
||||||
|
|
||||||
|
Pass configuration to MCP server:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"command": "python",
|
||||||
|
"args": ["-m", "my_mcp_server"],
|
||||||
|
"env": {
|
||||||
|
"DATABASE_URL": "${DB_URL}",
|
||||||
|
"API_KEY": "${API_KEY}",
|
||||||
|
"LOG_LEVEL": "info"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## Integration Patterns
|
||||||
|
|
||||||
|
### Pattern 1: Simple Tool Wrapper
|
||||||
|
|
||||||
|
Commands use MCP tools with user interaction:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
# Command: create-item.md
|
||||||
|
---
|
||||||
|
allowed-tools: ["mcp__plugin_name_server__create_item"]
|
||||||
|
---
|
||||||
|
|
||||||
|
Steps:
|
||||||
|
1. Gather item details from user
|
||||||
|
2. Use mcp__plugin_name_server__create_item
|
||||||
|
3. Confirm creation
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use for:** Adding validation or preprocessing before MCP calls.
|
||||||
|
|
||||||
|
### Pattern 2: Autonomous Agent
|
||||||
|
|
||||||
|
Agents use MCP tools autonomously:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
# Agent: data-analyzer.md
|
||||||
|
|
||||||
|
Analysis Process:
|
||||||
|
1. Query data via mcp__plugin_db_server__query
|
||||||
|
2. Process and analyze results
|
||||||
|
3. Generate insights report
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use for:** Multi-step MCP workflows without user interaction.
|
||||||
|
|
||||||
|
### Pattern 3: Multi-Server Plugin
|
||||||
|
|
||||||
|
Integrate multiple MCP servers:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"github": {
|
||||||
|
"type": "sse",
|
||||||
|
"url": "https://mcp.github.com/sse"
|
||||||
|
},
|
||||||
|
"jira": {
|
||||||
|
"type": "sse",
|
||||||
|
"url": "https://mcp.jira.com/sse"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use for:** Workflows spanning multiple services.
|
||||||
|
|
||||||
|
## Security Best Practices
|
||||||
|
|
||||||
|
### Use HTTPS/WSS
|
||||||
|
|
||||||
|
Always use secure connections:
|
||||||
|
|
||||||
|
```json
|
||||||
|
✅ "url": "https://mcp.example.com/sse"
|
||||||
|
❌ "url": "http://mcp.example.com/sse"
|
||||||
|
```
|
||||||
|
|
||||||
|
### Token Management
|
||||||
|
|
||||||
|
**DO:**
|
||||||
|
- ✅ Use environment variables for tokens
|
||||||
|
- ✅ Document required env vars in README
|
||||||
|
- ✅ Let OAuth flow handle authentication
|
||||||
|
|
||||||
|
**DON'T:**
|
||||||
|
- ❌ Hardcode tokens in configuration
|
||||||
|
- ❌ Commit tokens to git
|
||||||
|
- ❌ Share tokens in documentation
|
||||||
|
|
||||||
|
### Permission Scoping
|
||||||
|
|
||||||
|
Pre-allow only necessary MCP tools:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
✅ allowed-tools: [
|
||||||
|
"mcp__plugin_api_server__read_data",
|
||||||
|
"mcp__plugin_api_server__create_item"
|
||||||
|
]
|
||||||
|
|
||||||
|
❌ allowed-tools: ["mcp__plugin_api_server__*"]
|
||||||
|
```
|
||||||
|
|
||||||
|
## Error Handling
|
||||||
|
|
||||||
|
### Connection Failures
|
||||||
|
|
||||||
|
Handle MCP server unavailability:
|
||||||
|
- Provide fallback behavior in commands
|
||||||
|
- Inform user of connection issues
|
||||||
|
- Check server URL and configuration
|
||||||
|
|
||||||
|
### Tool Call Errors
|
||||||
|
|
||||||
|
Handle failed MCP operations:
|
||||||
|
- Validate inputs before calling MCP tools
|
||||||
|
- Provide clear error messages
|
||||||
|
- Check rate limiting and quotas
|
||||||
|
|
||||||
|
### Configuration Errors
|
||||||
|
|
||||||
|
Validate MCP configuration:
|
||||||
|
- Test server connectivity during development
|
||||||
|
- Validate JSON syntax
|
||||||
|
- Check required environment variables
|
||||||
|
|
||||||
|
## Performance Considerations
|
||||||
|
|
||||||
|
### Lazy Loading
|
||||||
|
|
||||||
|
MCP servers connect on-demand:
|
||||||
|
- Not all servers connect at startup
|
||||||
|
- First tool use triggers connection
|
||||||
|
- Connection pooling managed automatically
|
||||||
|
|
||||||
|
### Batching
|
||||||
|
|
||||||
|
Batch similar requests when possible:
|
||||||
|
|
||||||
|
```
|
||||||
|
# Good: Single query with filters
|
||||||
|
tasks = search_tasks(project="X", assignee="me", limit=50)
|
||||||
|
|
||||||
|
# Avoid: Many individual queries
|
||||||
|
for id in task_ids:
|
||||||
|
task = get_task(id)
|
||||||
|
```
|
||||||
|
|
||||||
|
## Testing MCP Integration
|
||||||
|
|
||||||
|
### Local Testing
|
||||||
|
|
||||||
|
1. Configure MCP server in `.mcp.json`
|
||||||
|
2. Install plugin locally (`.claude-plugin/`)
|
||||||
|
3. Run `/mcp` to verify server appears
|
||||||
|
4. Test tool calls in commands
|
||||||
|
5. Check `claude --debug` logs for connection issues
|
||||||
|
|
||||||
|
### Validation Checklist
|
||||||
|
|
||||||
|
- [ ] MCP configuration is valid JSON
|
||||||
|
- [ ] Server URL is correct and accessible
|
||||||
|
- [ ] Required environment variables documented
|
||||||
|
- [ ] Tools appear in `/mcp` output
|
||||||
|
- [ ] Authentication works (OAuth or tokens)
|
||||||
|
- [ ] Tool calls succeed from commands
|
||||||
|
- [ ] Error cases handled gracefully
|
||||||
|
|
||||||
|
## Debugging
|
||||||
|
|
||||||
|
### Enable Debug Logging
|
||||||
|
|
||||||
|
```bash
|
||||||
|
claude --debug
|
||||||
|
```
|
||||||
|
|
||||||
|
Look for:
|
||||||
|
- MCP server connection attempts
|
||||||
|
- Tool discovery logs
|
||||||
|
- Authentication flows
|
||||||
|
- Tool call errors
|
||||||
|
|
||||||
|
### Common Issues
|
||||||
|
|
||||||
|
**Server not connecting:**
|
||||||
|
- Check URL is correct
|
||||||
|
- Verify server is running (stdio)
|
||||||
|
- Check network connectivity
|
||||||
|
- Review authentication configuration
|
||||||
|
|
||||||
|
**Tools not available:**
|
||||||
|
- Verify server connected successfully
|
||||||
|
- Check tool names match exactly
|
||||||
|
- Run `/mcp` to see available tools
|
||||||
|
- Restart Claude Code after config changes
|
||||||
|
|
||||||
|
**Authentication failing:**
|
||||||
|
- Clear cached auth tokens
|
||||||
|
- Re-authenticate
|
||||||
|
- Check token scopes and permissions
|
||||||
|
- Verify environment variables set
|
||||||
|
|
||||||
|
## Quick Reference
|
||||||
|
|
||||||
|
### MCP Server Types
|
||||||
|
|
||||||
|
| Type | Transport | Best For | Auth |
|
||||||
|
|------|-----------|----------|------|
|
||||||
|
| stdio | Process | Local tools, custom servers | Env vars |
|
||||||
|
| SSE | HTTP | Hosted services, cloud APIs | OAuth |
|
||||||
|
| HTTP | REST | API backends, token auth | Tokens |
|
||||||
|
| ws | WebSocket | Real-time, streaming | Tokens |
|
||||||
|
|
||||||
|
### Configuration Checklist
|
||||||
|
|
||||||
|
- [ ] Server type specified (stdio/SSE/HTTP/ws)
|
||||||
|
- [ ] Type-specific fields complete (command or url)
|
||||||
|
- [ ] Authentication configured
|
||||||
|
- [ ] Environment variables documented
|
||||||
|
- [ ] HTTPS/WSS used (not HTTP/WS)
|
||||||
|
- [ ] ${CLAUDE_PLUGIN_ROOT} used for paths
|
||||||
|
|
||||||
|
### Best Practices
|
||||||
|
|
||||||
|
**DO:**
|
||||||
|
- ✅ Use ${CLAUDE_PLUGIN_ROOT} for portable paths
|
||||||
|
- ✅ Document required environment variables
|
||||||
|
- ✅ Use secure connections (HTTPS/WSS)
|
||||||
|
- ✅ Pre-allow specific MCP tools in commands
|
||||||
|
- ✅ Test MCP integration before publishing
|
||||||
|
- ✅ Handle connection and tool errors gracefully
|
||||||
|
|
||||||
|
**DON'T:**
|
||||||
|
- ❌ Hardcode absolute paths
|
||||||
|
- ❌ Commit credentials to git
|
||||||
|
- ❌ Use HTTP instead of HTTPS
|
||||||
|
- ❌ Pre-allow all tools with wildcards
|
||||||
|
- ❌ Skip error handling
|
||||||
|
- ❌ Forget to document setup
|
||||||
|
|
||||||
|
## Additional Resources
|
||||||
|
|
||||||
|
### Reference Files
|
||||||
|
|
||||||
|
For detailed information, consult:
|
||||||
|
|
||||||
|
- **`references/server-types.md`** - Deep dive on each server type
|
||||||
|
- **`references/authentication.md`** - Authentication patterns and OAuth
|
||||||
|
- **`references/tool-usage.md`** - Using MCP tools in commands and agents
|
||||||
|
|
||||||
|
### Example Configurations
|
||||||
|
|
||||||
|
Working examples in `examples/`:
|
||||||
|
|
||||||
|
- **`stdio-server.json`** - Local stdio MCP server
|
||||||
|
- **`sse-server.json`** - Hosted SSE server with OAuth
|
||||||
|
- **`http-server.json`** - REST API with token auth
|
||||||
|
|
||||||
|
### External Resources
|
||||||
|
|
||||||
|
- **Official MCP Docs**: https://modelcontextprotocol.io/
|
||||||
|
- **Claude Code MCP Docs**: https://docs.claude.com/en/docs/claude-code/mcp
|
||||||
|
- **MCP SDK**: @modelcontextprotocol/sdk
|
||||||
|
- **Testing**: Use `claude --debug` and `/mcp` command
|
||||||
|
|
||||||
|
## Implementation Workflow
|
||||||
|
|
||||||
|
To add MCP integration to a plugin:
|
||||||
|
|
||||||
|
1. Choose MCP server type (stdio, SSE, HTTP, ws)
|
||||||
|
2. Create `.mcp.json` at plugin root with configuration
|
||||||
|
3. Use ${CLAUDE_PLUGIN_ROOT} for all file references
|
||||||
|
4. Document required environment variables in README
|
||||||
|
5. Test locally with `/mcp` command
|
||||||
|
6. Pre-allow MCP tools in relevant commands
|
||||||
|
7. Handle authentication (OAuth or tokens)
|
||||||
|
8. Test error cases (connection failures, auth errors)
|
||||||
|
9. Document MCP integration in plugin README
|
||||||
|
|
||||||
|
Focus on stdio for custom/local servers, SSE for hosted services with OAuth.
|
||||||
20
skills/mcp-integration/examples/http-server.json
Normal file
20
skills/mcp-integration/examples/http-server.json
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
{
|
||||||
|
"_comment": "Example HTTP MCP server configuration for REST APIs",
|
||||||
|
"rest-api": {
|
||||||
|
"type": "http",
|
||||||
|
"url": "https://api.example.com/mcp",
|
||||||
|
"headers": {
|
||||||
|
"Authorization": "Bearer ${API_TOKEN}",
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
"X-API-Version": "2024-01-01"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"internal-service": {
|
||||||
|
"type": "http",
|
||||||
|
"url": "https://api.example.com/mcp",
|
||||||
|
"headers": {
|
||||||
|
"Authorization": "Bearer ${API_TOKEN}",
|
||||||
|
"X-Service-Name": "claude-plugin"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
19
skills/mcp-integration/examples/sse-server.json
Normal file
19
skills/mcp-integration/examples/sse-server.json
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
{
|
||||||
|
"_comment": "Example SSE MCP server configuration for hosted cloud services",
|
||||||
|
"asana": {
|
||||||
|
"type": "sse",
|
||||||
|
"url": "https://mcp.asana.com/sse"
|
||||||
|
},
|
||||||
|
"github": {
|
||||||
|
"type": "sse",
|
||||||
|
"url": "https://mcp.github.com/sse"
|
||||||
|
},
|
||||||
|
"custom-service": {
|
||||||
|
"type": "sse",
|
||||||
|
"url": "https://mcp.example.com/sse",
|
||||||
|
"headers": {
|
||||||
|
"X-API-Version": "v1",
|
||||||
|
"X-Client-ID": "${CLIENT_ID}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
26
skills/mcp-integration/examples/stdio-server.json
Normal file
26
skills/mcp-integration/examples/stdio-server.json
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
{
|
||||||
|
"_comment": "Example stdio MCP server configuration for local file system access",
|
||||||
|
"filesystem": {
|
||||||
|
"command": "npx",
|
||||||
|
"args": ["-y", "@modelcontextprotocol/server-filesystem", "${CLAUDE_PROJECT_DIR}"],
|
||||||
|
"env": {
|
||||||
|
"LOG_LEVEL": "info"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"database": {
|
||||||
|
"command": "${CLAUDE_PLUGIN_ROOT}/servers/db-server.js",
|
||||||
|
"args": ["--config", "${CLAUDE_PLUGIN_ROOT}/config/db.json"],
|
||||||
|
"env": {
|
||||||
|
"DATABASE_URL": "${DATABASE_URL}",
|
||||||
|
"DB_POOL_SIZE": "10"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"custom-tools": {
|
||||||
|
"command": "python",
|
||||||
|
"args": ["-m", "my_mcp_server", "--port", "8080"],
|
||||||
|
"env": {
|
||||||
|
"API_KEY": "${CUSTOM_API_KEY}",
|
||||||
|
"DEBUG": "false"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
549
skills/mcp-integration/references/authentication.md
Normal file
549
skills/mcp-integration/references/authentication.md
Normal file
@@ -0,0 +1,549 @@
|
|||||||
|
# MCP Authentication Patterns
|
||||||
|
|
||||||
|
Complete guide to authentication methods for MCP servers in Claude Code plugins.
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
|
||||||
|
MCP servers support multiple authentication methods depending on the server type and service requirements. Choose the method that best matches your use case and security requirements.
|
||||||
|
|
||||||
|
## OAuth (Automatic)
|
||||||
|
|
||||||
|
### How It Works
|
||||||
|
|
||||||
|
Claude Code automatically handles the complete OAuth 2.0 flow for SSE and HTTP servers:
|
||||||
|
|
||||||
|
1. User attempts to use MCP tool
|
||||||
|
2. Claude Code detects authentication needed
|
||||||
|
3. Opens browser for OAuth consent
|
||||||
|
4. User authorizes in browser
|
||||||
|
5. Tokens stored securely by Claude Code
|
||||||
|
6. Automatic token refresh
|
||||||
|
|
||||||
|
### Configuration
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"service": {
|
||||||
|
"type": "sse",
|
||||||
|
"url": "https://mcp.example.com/sse"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
No additional auth configuration needed! Claude Code handles everything.
|
||||||
|
|
||||||
|
### Supported Services
|
||||||
|
|
||||||
|
**Known OAuth-enabled MCP servers:**
|
||||||
|
- Asana: `https://mcp.asana.com/sse`
|
||||||
|
- GitHub (when available)
|
||||||
|
- Google services (when available)
|
||||||
|
- Custom OAuth servers
|
||||||
|
|
||||||
|
### OAuth Scopes
|
||||||
|
|
||||||
|
OAuth scopes are determined by the MCP server. Users see required scopes during the consent flow.
|
||||||
|
|
||||||
|
**Document required scopes in your README:**
|
||||||
|
```markdown
|
||||||
|
## Authentication
|
||||||
|
|
||||||
|
This plugin requires the following Asana permissions:
|
||||||
|
- Read tasks and projects
|
||||||
|
- Create and update tasks
|
||||||
|
- Access workspace data
|
||||||
|
```
|
||||||
|
|
||||||
|
### Token Storage
|
||||||
|
|
||||||
|
Tokens are stored securely by Claude Code:
|
||||||
|
- Not accessible to plugins
|
||||||
|
- Encrypted at rest
|
||||||
|
- Automatic refresh
|
||||||
|
- Cleared on sign-out
|
||||||
|
|
||||||
|
### Troubleshooting OAuth
|
||||||
|
|
||||||
|
**Authentication loop:**
|
||||||
|
- Clear cached tokens (sign out and sign in)
|
||||||
|
- Check OAuth redirect URLs
|
||||||
|
- Verify server OAuth configuration
|
||||||
|
|
||||||
|
**Scope issues:**
|
||||||
|
- User may need to re-authorize for new scopes
|
||||||
|
- Check server documentation for required scopes
|
||||||
|
|
||||||
|
**Token expiration:**
|
||||||
|
- Claude Code auto-refreshes
|
||||||
|
- If refresh fails, prompts re-authentication
|
||||||
|
|
||||||
|
## Token-Based Authentication
|
||||||
|
|
||||||
|
### Bearer Tokens
|
||||||
|
|
||||||
|
Most common for HTTP and WebSocket servers.
|
||||||
|
|
||||||
|
**Configuration:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"api": {
|
||||||
|
"type": "http",
|
||||||
|
"url": "https://api.example.com/mcp",
|
||||||
|
"headers": {
|
||||||
|
"Authorization": "Bearer ${API_TOKEN}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Environment variable:**
|
||||||
|
```bash
|
||||||
|
export API_TOKEN="your-secret-token-here"
|
||||||
|
```
|
||||||
|
|
||||||
|
### API Keys
|
||||||
|
|
||||||
|
Alternative to Bearer tokens, often in custom headers.
|
||||||
|
|
||||||
|
**Configuration:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"api": {
|
||||||
|
"type": "http",
|
||||||
|
"url": "https://api.example.com/mcp",
|
||||||
|
"headers": {
|
||||||
|
"X-API-Key": "${API_KEY}",
|
||||||
|
"X-API-Secret": "${API_SECRET}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Custom Headers
|
||||||
|
|
||||||
|
Services may use custom authentication headers.
|
||||||
|
|
||||||
|
**Configuration:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"service": {
|
||||||
|
"type": "sse",
|
||||||
|
"url": "https://mcp.example.com/sse",
|
||||||
|
"headers": {
|
||||||
|
"X-Auth-Token": "${AUTH_TOKEN}",
|
||||||
|
"X-User-ID": "${USER_ID}",
|
||||||
|
"X-Tenant-ID": "${TENANT_ID}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Documenting Token Requirements
|
||||||
|
|
||||||
|
Always document in your README:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
## Setup
|
||||||
|
|
||||||
|
### Required Environment Variables
|
||||||
|
|
||||||
|
Set these environment variables before using the plugin:
|
||||||
|
|
||||||
|
\`\`\`bash
|
||||||
|
export API_TOKEN="your-token-here"
|
||||||
|
export API_SECRET="your-secret-here"
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
### Obtaining Tokens
|
||||||
|
|
||||||
|
1. Visit https://api.example.com/tokens
|
||||||
|
2. Create a new API token
|
||||||
|
3. Copy the token and secret
|
||||||
|
4. Set environment variables as shown above
|
||||||
|
|
||||||
|
### Token Permissions
|
||||||
|
|
||||||
|
The API token needs the following permissions:
|
||||||
|
- Read access to resources
|
||||||
|
- Write access for creating items
|
||||||
|
- Delete access (optional, for cleanup operations)
|
||||||
|
\`\`\`
|
||||||
|
```
|
||||||
|
|
||||||
|
## Environment Variable Authentication (stdio)
|
||||||
|
|
||||||
|
### Passing Credentials to Server
|
||||||
|
|
||||||
|
For stdio servers, pass credentials via environment variables:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"database": {
|
||||||
|
"command": "python",
|
||||||
|
"args": ["-m", "mcp_server_db"],
|
||||||
|
"env": {
|
||||||
|
"DATABASE_URL": "${DATABASE_URL}",
|
||||||
|
"DB_USER": "${DB_USER}",
|
||||||
|
"DB_PASSWORD": "${DB_PASSWORD}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### User Environment Variables
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# User sets these in their shell
|
||||||
|
export DATABASE_URL="postgresql://localhost/mydb"
|
||||||
|
export DB_USER="myuser"
|
||||||
|
export DB_PASSWORD="mypassword"
|
||||||
|
```
|
||||||
|
|
||||||
|
### Documentation Template
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
## Database Configuration
|
||||||
|
|
||||||
|
Set these environment variables:
|
||||||
|
|
||||||
|
\`\`\`bash
|
||||||
|
export DATABASE_URL="postgresql://host:port/database"
|
||||||
|
export DB_USER="username"
|
||||||
|
export DB_PASSWORD="password"
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
Or create a `.env` file (add to `.gitignore`):
|
||||||
|
|
||||||
|
\`\`\`
|
||||||
|
DATABASE_URL=postgresql://localhost:5432/mydb
|
||||||
|
DB_USER=myuser
|
||||||
|
DB_PASSWORD=mypassword
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
Load with: \`source .env\` or \`export $(cat .env | xargs)\`
|
||||||
|
\`\`\`
|
||||||
|
```
|
||||||
|
|
||||||
|
## Dynamic Headers
|
||||||
|
|
||||||
|
### Headers Helper Script
|
||||||
|
|
||||||
|
For tokens that change or expire, use a helper script:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"api": {
|
||||||
|
"type": "sse",
|
||||||
|
"url": "https://api.example.com",
|
||||||
|
"headersHelper": "${CLAUDE_PLUGIN_ROOT}/scripts/get-headers.sh"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Script (get-headers.sh):**
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
# Generate dynamic authentication headers
|
||||||
|
|
||||||
|
# Fetch fresh token
|
||||||
|
TOKEN=$(get-fresh-token-from-somewhere)
|
||||||
|
|
||||||
|
# Output JSON headers
|
||||||
|
cat <<EOF
|
||||||
|
{
|
||||||
|
"Authorization": "Bearer $TOKEN",
|
||||||
|
"X-Timestamp": "$(date -Iseconds)"
|
||||||
|
}
|
||||||
|
EOF
|
||||||
|
```
|
||||||
|
|
||||||
|
### Use Cases for Dynamic Headers
|
||||||
|
|
||||||
|
- Short-lived tokens that need refresh
|
||||||
|
- Tokens with HMAC signatures
|
||||||
|
- Time-based authentication
|
||||||
|
- Dynamic tenant/workspace selection
|
||||||
|
|
||||||
|
## Security Best Practices
|
||||||
|
|
||||||
|
### DO
|
||||||
|
|
||||||
|
✅ **Use environment variables:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"headers": {
|
||||||
|
"Authorization": "Bearer ${API_TOKEN}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
✅ **Document required variables in README**
|
||||||
|
|
||||||
|
✅ **Use HTTPS/WSS always**
|
||||||
|
|
||||||
|
✅ **Implement token rotation**
|
||||||
|
|
||||||
|
✅ **Store tokens securely (env vars, not files)**
|
||||||
|
|
||||||
|
✅ **Let OAuth handle authentication when available**
|
||||||
|
|
||||||
|
### DON'T
|
||||||
|
|
||||||
|
❌ **Hardcode tokens:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"headers": {
|
||||||
|
"Authorization": "Bearer sk-abc123..." // NEVER!
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
❌ **Commit tokens to git**
|
||||||
|
|
||||||
|
❌ **Share tokens in documentation**
|
||||||
|
|
||||||
|
❌ **Use HTTP instead of HTTPS**
|
||||||
|
|
||||||
|
❌ **Store tokens in plugin files**
|
||||||
|
|
||||||
|
❌ **Log tokens or sensitive headers**
|
||||||
|
|
||||||
|
## Multi-Tenancy Patterns
|
||||||
|
|
||||||
|
### Workspace/Tenant Selection
|
||||||
|
|
||||||
|
**Via environment variable:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"api": {
|
||||||
|
"type": "http",
|
||||||
|
"url": "https://api.example.com/mcp",
|
||||||
|
"headers": {
|
||||||
|
"Authorization": "Bearer ${API_TOKEN}",
|
||||||
|
"X-Workspace-ID": "${WORKSPACE_ID}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Via URL:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"api": {
|
||||||
|
"type": "http",
|
||||||
|
"url": "https://${TENANT_ID}.api.example.com/mcp"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Per-User Configuration
|
||||||
|
|
||||||
|
Users set their own workspace:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
export WORKSPACE_ID="my-workspace-123"
|
||||||
|
export TENANT_ID="my-company"
|
||||||
|
```
|
||||||
|
|
||||||
|
## Authentication Troubleshooting
|
||||||
|
|
||||||
|
### Common Issues
|
||||||
|
|
||||||
|
**401 Unauthorized:**
|
||||||
|
- Check token is set correctly
|
||||||
|
- Verify token hasn't expired
|
||||||
|
- Check token has required permissions
|
||||||
|
- Ensure header format is correct
|
||||||
|
|
||||||
|
**403 Forbidden:**
|
||||||
|
- Token valid but lacks permissions
|
||||||
|
- Check scope/permissions
|
||||||
|
- Verify workspace/tenant ID
|
||||||
|
- May need admin approval
|
||||||
|
|
||||||
|
**Token not found:**
|
||||||
|
```bash
|
||||||
|
# Check environment variable is set
|
||||||
|
echo $API_TOKEN
|
||||||
|
|
||||||
|
# If empty, set it
|
||||||
|
export API_TOKEN="your-token"
|
||||||
|
```
|
||||||
|
|
||||||
|
**Token in wrong format:**
|
||||||
|
```json
|
||||||
|
// Correct
|
||||||
|
"Authorization": "Bearer sk-abc123"
|
||||||
|
|
||||||
|
// Wrong
|
||||||
|
"Authorization": "sk-abc123"
|
||||||
|
```
|
||||||
|
|
||||||
|
### Debugging Authentication
|
||||||
|
|
||||||
|
**Enable debug mode:**
|
||||||
|
```bash
|
||||||
|
claude --debug
|
||||||
|
```
|
||||||
|
|
||||||
|
Look for:
|
||||||
|
- Authentication header values (sanitized)
|
||||||
|
- OAuth flow progress
|
||||||
|
- Token refresh attempts
|
||||||
|
- Authentication errors
|
||||||
|
|
||||||
|
**Test authentication separately:**
|
||||||
|
```bash
|
||||||
|
# Test HTTP endpoint
|
||||||
|
curl -H "Authorization: Bearer $API_TOKEN" \
|
||||||
|
https://api.example.com/mcp/health
|
||||||
|
|
||||||
|
# Should return 200 OK
|
||||||
|
```
|
||||||
|
|
||||||
|
## Migration Patterns
|
||||||
|
|
||||||
|
### From Hardcoded to Environment Variables
|
||||||
|
|
||||||
|
**Before:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"headers": {
|
||||||
|
"Authorization": "Bearer sk-hardcoded-token"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**After:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"headers": {
|
||||||
|
"Authorization": "Bearer ${API_TOKEN}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Migration steps:**
|
||||||
|
1. Add environment variable to plugin README
|
||||||
|
2. Update configuration to use ${VAR}
|
||||||
|
3. Test with variable set
|
||||||
|
4. Remove hardcoded value
|
||||||
|
5. Commit changes
|
||||||
|
|
||||||
|
### From Basic Auth to OAuth
|
||||||
|
|
||||||
|
**Before:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"headers": {
|
||||||
|
"Authorization": "Basic ${BASE64_CREDENTIALS}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**After:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"type": "sse",
|
||||||
|
"url": "https://mcp.example.com/sse"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Benefits:**
|
||||||
|
- Better security
|
||||||
|
- No credential management
|
||||||
|
- Automatic token refresh
|
||||||
|
- Scoped permissions
|
||||||
|
|
||||||
|
## Advanced Authentication
|
||||||
|
|
||||||
|
### Mutual TLS (mTLS)
|
||||||
|
|
||||||
|
Some enterprise services require client certificates.
|
||||||
|
|
||||||
|
**Not directly supported in MCP configuration.**
|
||||||
|
|
||||||
|
**Workaround:** Wrap in stdio server that handles mTLS:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"secure-api": {
|
||||||
|
"command": "${CLAUDE_PLUGIN_ROOT}/servers/mtls-wrapper",
|
||||||
|
"args": ["--cert", "${CLIENT_CERT}", "--key", "${CLIENT_KEY}"],
|
||||||
|
"env": {
|
||||||
|
"API_URL": "https://secure.example.com"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### JWT Tokens
|
||||||
|
|
||||||
|
Generate JWT tokens dynamically with headers helper:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
# generate-jwt.sh
|
||||||
|
|
||||||
|
# Generate JWT (using library or API call)
|
||||||
|
JWT=$(generate-jwt-token)
|
||||||
|
|
||||||
|
echo "{\"Authorization\": \"Bearer $JWT\"}"
|
||||||
|
```
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"headersHelper": "${CLAUDE_PLUGIN_ROOT}/scripts/generate-jwt.sh"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### HMAC Signatures
|
||||||
|
|
||||||
|
For APIs requiring request signing:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
# generate-hmac.sh
|
||||||
|
|
||||||
|
TIMESTAMP=$(date -Iseconds)
|
||||||
|
SIGNATURE=$(echo -n "$TIMESTAMP" | openssl dgst -sha256 -hmac "$SECRET_KEY" | cut -d' ' -f2)
|
||||||
|
|
||||||
|
cat <<EOF
|
||||||
|
{
|
||||||
|
"X-Timestamp": "$TIMESTAMP",
|
||||||
|
"X-Signature": "$SIGNATURE",
|
||||||
|
"X-API-Key": "$API_KEY"
|
||||||
|
}
|
||||||
|
EOF
|
||||||
|
```
|
||||||
|
|
||||||
|
## Best Practices Summary
|
||||||
|
|
||||||
|
### For Plugin Developers
|
||||||
|
|
||||||
|
1. **Prefer OAuth** when service supports it
|
||||||
|
2. **Use environment variables** for tokens
|
||||||
|
3. **Document all required variables** in README
|
||||||
|
4. **Provide setup instructions** with examples
|
||||||
|
5. **Never commit credentials**
|
||||||
|
6. **Use HTTPS/WSS only**
|
||||||
|
7. **Test authentication thoroughly**
|
||||||
|
|
||||||
|
### For Plugin Users
|
||||||
|
|
||||||
|
1. **Set environment variables** before using plugin
|
||||||
|
2. **Keep tokens secure** and private
|
||||||
|
3. **Rotate tokens regularly**
|
||||||
|
4. **Use different tokens** for dev/prod
|
||||||
|
5. **Don't commit .env files** to git
|
||||||
|
6. **Review OAuth scopes** before authorizing
|
||||||
|
|
||||||
|
## Conclusion
|
||||||
|
|
||||||
|
Choose the authentication method that matches your MCP server's requirements:
|
||||||
|
- **OAuth** for cloud services (easiest for users)
|
||||||
|
- **Bearer tokens** for API services
|
||||||
|
- **Environment variables** for stdio servers
|
||||||
|
- **Dynamic headers** for complex auth flows
|
||||||
|
|
||||||
|
Always prioritize security and provide clear setup documentation for users.
|
||||||
536
skills/mcp-integration/references/server-types.md
Normal file
536
skills/mcp-integration/references/server-types.md
Normal file
@@ -0,0 +1,536 @@
|
|||||||
|
# MCP Server Types: Deep Dive
|
||||||
|
|
||||||
|
Complete reference for all MCP server types supported in Claude Code plugins.
|
||||||
|
|
||||||
|
## stdio (Standard Input/Output)
|
||||||
|
|
||||||
|
### Overview
|
||||||
|
|
||||||
|
Execute local MCP servers as child processes with communication via stdin/stdout. Best choice for local tools, custom servers, and NPM packages.
|
||||||
|
|
||||||
|
### Configuration
|
||||||
|
|
||||||
|
**Basic:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"my-server": {
|
||||||
|
"command": "npx",
|
||||||
|
"args": ["-y", "my-mcp-server"]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**With environment:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"my-server": {
|
||||||
|
"command": "${CLAUDE_PLUGIN_ROOT}/servers/custom-server",
|
||||||
|
"args": ["--config", "${CLAUDE_PLUGIN_ROOT}/config.json"],
|
||||||
|
"env": {
|
||||||
|
"API_KEY": "${MY_API_KEY}",
|
||||||
|
"LOG_LEVEL": "debug",
|
||||||
|
"DATABASE_URL": "${DB_URL}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Process Lifecycle
|
||||||
|
|
||||||
|
1. **Startup**: Claude Code spawns process with `command` and `args`
|
||||||
|
2. **Communication**: JSON-RPC messages via stdin/stdout
|
||||||
|
3. **Lifecycle**: Process runs for entire Claude Code session
|
||||||
|
4. **Shutdown**: Process terminated when Claude Code exits
|
||||||
|
|
||||||
|
### Use Cases
|
||||||
|
|
||||||
|
**NPM Packages:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"filesystem": {
|
||||||
|
"command": "npx",
|
||||||
|
"args": ["-y", "@modelcontextprotocol/server-filesystem", "/path"]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Custom Scripts:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"custom": {
|
||||||
|
"command": "${CLAUDE_PLUGIN_ROOT}/servers/my-server.js",
|
||||||
|
"args": ["--verbose"]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Python Servers:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"python-server": {
|
||||||
|
"command": "python",
|
||||||
|
"args": ["-m", "my_mcp_server"],
|
||||||
|
"env": {
|
||||||
|
"PYTHONUNBUFFERED": "1"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Best Practices
|
||||||
|
|
||||||
|
1. **Use absolute paths or ${CLAUDE_PLUGIN_ROOT}**
|
||||||
|
2. **Set PYTHONUNBUFFERED for Python servers**
|
||||||
|
3. **Pass configuration via args or env, not stdin**
|
||||||
|
4. **Handle server crashes gracefully**
|
||||||
|
5. **Log to stderr, not stdout (stdout is for MCP protocol)**
|
||||||
|
|
||||||
|
### Troubleshooting
|
||||||
|
|
||||||
|
**Server won't start:**
|
||||||
|
- Check command exists and is executable
|
||||||
|
- Verify file paths are correct
|
||||||
|
- Check permissions
|
||||||
|
- Review `claude --debug` logs
|
||||||
|
|
||||||
|
**Communication fails:**
|
||||||
|
- Ensure server uses stdin/stdout correctly
|
||||||
|
- Check for stray print/console.log statements
|
||||||
|
- Verify JSON-RPC format
|
||||||
|
|
||||||
|
## SSE (Server-Sent Events)
|
||||||
|
|
||||||
|
### Overview
|
||||||
|
|
||||||
|
Connect to hosted MCP servers via HTTP with server-sent events for streaming. Best for cloud services and OAuth authentication.
|
||||||
|
|
||||||
|
### Configuration
|
||||||
|
|
||||||
|
**Basic:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"hosted-service": {
|
||||||
|
"type": "sse",
|
||||||
|
"url": "https://mcp.example.com/sse"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**With headers:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"service": {
|
||||||
|
"type": "sse",
|
||||||
|
"url": "https://mcp.example.com/sse",
|
||||||
|
"headers": {
|
||||||
|
"X-API-Version": "v1",
|
||||||
|
"X-Client-ID": "${CLIENT_ID}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Connection Lifecycle
|
||||||
|
|
||||||
|
1. **Initialization**: HTTP connection established to URL
|
||||||
|
2. **Handshake**: MCP protocol negotiation
|
||||||
|
3. **Streaming**: Server sends events via SSE
|
||||||
|
4. **Requests**: Client sends HTTP POST for tool calls
|
||||||
|
5. **Reconnection**: Automatic reconnection on disconnect
|
||||||
|
|
||||||
|
### Authentication
|
||||||
|
|
||||||
|
**OAuth (Automatic):**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"asana": {
|
||||||
|
"type": "sse",
|
||||||
|
"url": "https://mcp.asana.com/sse"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Claude Code handles OAuth flow:
|
||||||
|
1. User prompted to authenticate on first use
|
||||||
|
2. Opens browser for OAuth flow
|
||||||
|
3. Tokens stored securely
|
||||||
|
4. Automatic token refresh
|
||||||
|
|
||||||
|
**Custom Headers:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"service": {
|
||||||
|
"type": "sse",
|
||||||
|
"url": "https://mcp.example.com/sse",
|
||||||
|
"headers": {
|
||||||
|
"Authorization": "Bearer ${API_TOKEN}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Use Cases
|
||||||
|
|
||||||
|
**Official Services:**
|
||||||
|
- Asana: `https://mcp.asana.com/sse`
|
||||||
|
- GitHub: `https://mcp.github.com/sse`
|
||||||
|
- Other hosted MCP servers
|
||||||
|
|
||||||
|
**Custom Hosted Servers:**
|
||||||
|
Deploy your own MCP server and expose via HTTPS + SSE.
|
||||||
|
|
||||||
|
### Best Practices
|
||||||
|
|
||||||
|
1. **Always use HTTPS, never HTTP**
|
||||||
|
2. **Let OAuth handle authentication when available**
|
||||||
|
3. **Use environment variables for tokens**
|
||||||
|
4. **Handle connection failures gracefully**
|
||||||
|
5. **Document OAuth scopes required**
|
||||||
|
|
||||||
|
### Troubleshooting
|
||||||
|
|
||||||
|
**Connection refused:**
|
||||||
|
- Check URL is correct and accessible
|
||||||
|
- Verify HTTPS certificate is valid
|
||||||
|
- Check network connectivity
|
||||||
|
- Review firewall settings
|
||||||
|
|
||||||
|
**OAuth fails:**
|
||||||
|
- Clear cached tokens
|
||||||
|
- Check OAuth scopes
|
||||||
|
- Verify redirect URLs
|
||||||
|
- Re-authenticate
|
||||||
|
|
||||||
|
## HTTP (REST API)
|
||||||
|
|
||||||
|
### Overview
|
||||||
|
|
||||||
|
Connect to RESTful MCP servers via standard HTTP requests. Best for token-based auth and stateless interactions.
|
||||||
|
|
||||||
|
### Configuration
|
||||||
|
|
||||||
|
**Basic:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"api": {
|
||||||
|
"type": "http",
|
||||||
|
"url": "https://api.example.com/mcp"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**With authentication:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"api": {
|
||||||
|
"type": "http",
|
||||||
|
"url": "https://api.example.com/mcp",
|
||||||
|
"headers": {
|
||||||
|
"Authorization": "Bearer ${API_TOKEN}",
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
"X-API-Version": "2024-01-01"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Request/Response Flow
|
||||||
|
|
||||||
|
1. **Tool Discovery**: GET to discover available tools
|
||||||
|
2. **Tool Invocation**: POST with tool name and parameters
|
||||||
|
3. **Response**: JSON response with results or errors
|
||||||
|
4. **Stateless**: Each request independent
|
||||||
|
|
||||||
|
### Authentication
|
||||||
|
|
||||||
|
**Token-Based:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"headers": {
|
||||||
|
"Authorization": "Bearer ${API_TOKEN}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**API Key:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"headers": {
|
||||||
|
"X-API-Key": "${API_KEY}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Custom Auth:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"headers": {
|
||||||
|
"X-Auth-Token": "${AUTH_TOKEN}",
|
||||||
|
"X-User-ID": "${USER_ID}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Use Cases
|
||||||
|
|
||||||
|
- REST API backends
|
||||||
|
- Internal services
|
||||||
|
- Microservices
|
||||||
|
- Serverless functions
|
||||||
|
|
||||||
|
### Best Practices
|
||||||
|
|
||||||
|
1. **Use HTTPS for all connections**
|
||||||
|
2. **Store tokens in environment variables**
|
||||||
|
3. **Implement retry logic for transient failures**
|
||||||
|
4. **Handle rate limiting**
|
||||||
|
5. **Set appropriate timeouts**
|
||||||
|
|
||||||
|
### Troubleshooting
|
||||||
|
|
||||||
|
**HTTP errors:**
|
||||||
|
- 401: Check authentication headers
|
||||||
|
- 403: Verify permissions
|
||||||
|
- 429: Implement rate limiting
|
||||||
|
- 500: Check server logs
|
||||||
|
|
||||||
|
**Timeout issues:**
|
||||||
|
- Increase timeout if needed
|
||||||
|
- Check server performance
|
||||||
|
- Optimize tool implementations
|
||||||
|
|
||||||
|
## WebSocket (Real-time)
|
||||||
|
|
||||||
|
### Overview
|
||||||
|
|
||||||
|
Connect to MCP servers via WebSocket for real-time bidirectional communication. Best for streaming and low-latency applications.
|
||||||
|
|
||||||
|
### Configuration
|
||||||
|
|
||||||
|
**Basic:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"realtime": {
|
||||||
|
"type": "ws",
|
||||||
|
"url": "wss://mcp.example.com/ws"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**With authentication:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"realtime": {
|
||||||
|
"type": "ws",
|
||||||
|
"url": "wss://mcp.example.com/ws",
|
||||||
|
"headers": {
|
||||||
|
"Authorization": "Bearer ${TOKEN}",
|
||||||
|
"X-Client-ID": "${CLIENT_ID}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Connection Lifecycle
|
||||||
|
|
||||||
|
1. **Handshake**: WebSocket upgrade request
|
||||||
|
2. **Connection**: Persistent bidirectional channel
|
||||||
|
3. **Messages**: JSON-RPC over WebSocket
|
||||||
|
4. **Heartbeat**: Keep-alive messages
|
||||||
|
5. **Reconnection**: Automatic on disconnect
|
||||||
|
|
||||||
|
### Use Cases
|
||||||
|
|
||||||
|
- Real-time data streaming
|
||||||
|
- Live updates and notifications
|
||||||
|
- Collaborative editing
|
||||||
|
- Low-latency tool calls
|
||||||
|
- Push notifications from server
|
||||||
|
|
||||||
|
### Best Practices
|
||||||
|
|
||||||
|
1. **Use WSS (secure WebSocket), never WS**
|
||||||
|
2. **Implement heartbeat/ping-pong**
|
||||||
|
3. **Handle reconnection logic**
|
||||||
|
4. **Buffer messages during disconnection**
|
||||||
|
5. **Set connection timeouts**
|
||||||
|
|
||||||
|
### Troubleshooting
|
||||||
|
|
||||||
|
**Connection drops:**
|
||||||
|
- Implement reconnection logic
|
||||||
|
- Check network stability
|
||||||
|
- Verify server supports WebSocket
|
||||||
|
- Review firewall settings
|
||||||
|
|
||||||
|
**Message delivery:**
|
||||||
|
- Implement message acknowledgment
|
||||||
|
- Handle out-of-order messages
|
||||||
|
- Buffer during disconnection
|
||||||
|
|
||||||
|
## Comparison Matrix
|
||||||
|
|
||||||
|
| Feature | stdio | SSE | HTTP | WebSocket |
|
||||||
|
|---------|-------|-----|------|-----------|
|
||||||
|
| **Transport** | Process | HTTP/SSE | HTTP | WebSocket |
|
||||||
|
| **Direction** | Bidirectional | Server→Client | Request/Response | Bidirectional |
|
||||||
|
| **State** | Stateful | Stateful | Stateless | Stateful |
|
||||||
|
| **Auth** | Env vars | OAuth/Headers | Headers | Headers |
|
||||||
|
| **Use Case** | Local tools | Cloud services | REST APIs | Real-time |
|
||||||
|
| **Latency** | Lowest | Medium | Medium | Low |
|
||||||
|
| **Setup** | Easy | Medium | Easy | Medium |
|
||||||
|
| **Reconnect** | Process respawn | Automatic | N/A | Automatic |
|
||||||
|
|
||||||
|
## Choosing the Right Type
|
||||||
|
|
||||||
|
**Use stdio when:**
|
||||||
|
- Running local tools or custom servers
|
||||||
|
- Need lowest latency
|
||||||
|
- Working with file systems or local databases
|
||||||
|
- Distributing server with plugin
|
||||||
|
|
||||||
|
**Use SSE when:**
|
||||||
|
- Connecting to hosted services
|
||||||
|
- Need OAuth authentication
|
||||||
|
- Using official MCP servers (Asana, GitHub)
|
||||||
|
- Want automatic reconnection
|
||||||
|
|
||||||
|
**Use HTTP when:**
|
||||||
|
- Integrating with REST APIs
|
||||||
|
- Need stateless interactions
|
||||||
|
- Using token-based auth
|
||||||
|
- Simple request/response pattern
|
||||||
|
|
||||||
|
**Use WebSocket when:**
|
||||||
|
- Need real-time updates
|
||||||
|
- Building collaborative features
|
||||||
|
- Low-latency critical
|
||||||
|
- Bi-directional streaming required
|
||||||
|
|
||||||
|
## Migration Between Types
|
||||||
|
|
||||||
|
### From stdio to SSE
|
||||||
|
|
||||||
|
**Before (stdio):**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"local-server": {
|
||||||
|
"command": "node",
|
||||||
|
"args": ["server.js"]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**After (SSE - deploy server):**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"hosted-server": {
|
||||||
|
"type": "sse",
|
||||||
|
"url": "https://mcp.example.com/sse"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### From HTTP to WebSocket
|
||||||
|
|
||||||
|
**Before (HTTP):**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"api": {
|
||||||
|
"type": "http",
|
||||||
|
"url": "https://api.example.com/mcp"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**After (WebSocket):**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"realtime": {
|
||||||
|
"type": "ws",
|
||||||
|
"url": "wss://api.example.com/ws"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Benefits: Real-time updates, lower latency, bi-directional communication.
|
||||||
|
|
||||||
|
## Advanced Configuration
|
||||||
|
|
||||||
|
### Multiple Servers
|
||||||
|
|
||||||
|
Combine different types:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"local-db": {
|
||||||
|
"command": "npx",
|
||||||
|
"args": ["-y", "mcp-server-sqlite", "./data.db"]
|
||||||
|
},
|
||||||
|
"cloud-api": {
|
||||||
|
"type": "sse",
|
||||||
|
"url": "https://mcp.example.com/sse"
|
||||||
|
},
|
||||||
|
"internal-service": {
|
||||||
|
"type": "http",
|
||||||
|
"url": "https://api.example.com/mcp",
|
||||||
|
"headers": {
|
||||||
|
"Authorization": "Bearer ${API_TOKEN}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Conditional Configuration
|
||||||
|
|
||||||
|
Use environment variables to switch servers:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"api": {
|
||||||
|
"type": "http",
|
||||||
|
"url": "${API_URL}",
|
||||||
|
"headers": {
|
||||||
|
"Authorization": "Bearer ${API_TOKEN}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Set different values for dev/prod:
|
||||||
|
- Dev: `API_URL=http://localhost:8080/mcp`
|
||||||
|
- Prod: `API_URL=https://api.production.com/mcp`
|
||||||
|
|
||||||
|
## Security Considerations
|
||||||
|
|
||||||
|
### Stdio Security
|
||||||
|
|
||||||
|
- Validate command paths
|
||||||
|
- Don't execute user-provided commands
|
||||||
|
- Limit environment variable access
|
||||||
|
- Restrict file system access
|
||||||
|
|
||||||
|
### Network Security
|
||||||
|
|
||||||
|
- Always use HTTPS/WSS
|
||||||
|
- Validate SSL certificates
|
||||||
|
- Don't skip certificate verification
|
||||||
|
- Use secure token storage
|
||||||
|
|
||||||
|
### Token Management
|
||||||
|
|
||||||
|
- Never hardcode tokens
|
||||||
|
- Use environment variables
|
||||||
|
- Rotate tokens regularly
|
||||||
|
- Implement token refresh
|
||||||
|
- Document scopes required
|
||||||
|
|
||||||
|
## Conclusion
|
||||||
|
|
||||||
|
Choose the MCP server type based on your use case:
|
||||||
|
- **stdio** for local, custom, or NPM-packaged servers
|
||||||
|
- **SSE** for hosted services with OAuth
|
||||||
|
- **HTTP** for REST APIs with token auth
|
||||||
|
- **WebSocket** for real-time bidirectional communication
|
||||||
|
|
||||||
|
Test thoroughly and handle errors gracefully for robust MCP integration.
|
||||||
538
skills/mcp-integration/references/tool-usage.md
Normal file
538
skills/mcp-integration/references/tool-usage.md
Normal file
@@ -0,0 +1,538 @@
|
|||||||
|
# Using MCP Tools in Commands and Agents
|
||||||
|
|
||||||
|
Complete guide to using MCP tools effectively in Claude Code plugin commands and agents.
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
|
||||||
|
Once an MCP server is configured, its tools become available with the prefix `mcp__plugin_<plugin-name>_<server-name>__<tool-name>`. Use these tools in commands and agents just like built-in Claude Code tools.
|
||||||
|
|
||||||
|
## Tool Naming Convention
|
||||||
|
|
||||||
|
### Format
|
||||||
|
|
||||||
|
```
|
||||||
|
mcp__plugin_<plugin-name>_<server-name>__<tool-name>
|
||||||
|
```
|
||||||
|
|
||||||
|
### Examples
|
||||||
|
|
||||||
|
**Asana plugin with asana server:**
|
||||||
|
- `mcp__plugin_asana_asana__asana_create_task`
|
||||||
|
- `mcp__plugin_asana_asana__asana_search_tasks`
|
||||||
|
- `mcp__plugin_asana_asana__asana_get_project`
|
||||||
|
|
||||||
|
**Custom plugin with database server:**
|
||||||
|
- `mcp__plugin_myplug_database__query`
|
||||||
|
- `mcp__plugin_myplug_database__execute`
|
||||||
|
- `mcp__plugin_myplug_database__list_tables`
|
||||||
|
|
||||||
|
### Discovering Tool Names
|
||||||
|
|
||||||
|
**Use `/mcp` command:**
|
||||||
|
```bash
|
||||||
|
/mcp
|
||||||
|
```
|
||||||
|
|
||||||
|
This shows:
|
||||||
|
- All available MCP servers
|
||||||
|
- Tools provided by each server
|
||||||
|
- Tool schemas and descriptions
|
||||||
|
- Full tool names for use in configuration
|
||||||
|
|
||||||
|
## Using Tools in Commands
|
||||||
|
|
||||||
|
### Pre-Allowing Tools
|
||||||
|
|
||||||
|
Specify MCP tools in command frontmatter:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Create a new Asana task
|
||||||
|
allowed-tools: [
|
||||||
|
"mcp__plugin_asana_asana__asana_create_task"
|
||||||
|
]
|
||||||
|
---
|
||||||
|
|
||||||
|
# Create Task Command
|
||||||
|
|
||||||
|
To create a task:
|
||||||
|
1. Gather task details from user
|
||||||
|
2. Use mcp__plugin_asana_asana__asana_create_task with the details
|
||||||
|
3. Confirm creation to user
|
||||||
|
```
|
||||||
|
|
||||||
|
### Multiple Tools
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
allowed-tools: [
|
||||||
|
"mcp__plugin_asana_asana__asana_create_task",
|
||||||
|
"mcp__plugin_asana_asana__asana_search_tasks",
|
||||||
|
"mcp__plugin_asana_asana__asana_get_project"
|
||||||
|
]
|
||||||
|
---
|
||||||
|
```
|
||||||
|
|
||||||
|
### Wildcard (Use Sparingly)
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
allowed-tools: ["mcp__plugin_asana_asana__*"]
|
||||||
|
---
|
||||||
|
```
|
||||||
|
|
||||||
|
**Caution:** Only use wildcards if the command truly needs access to all tools from a server.
|
||||||
|
|
||||||
|
### Tool Usage in Command Instructions
|
||||||
|
|
||||||
|
**Example command:**
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Search and create Asana tasks
|
||||||
|
allowed-tools: [
|
||||||
|
"mcp__plugin_asana_asana__asana_search_tasks",
|
||||||
|
"mcp__plugin_asana_asana__asana_create_task"
|
||||||
|
]
|
||||||
|
---
|
||||||
|
|
||||||
|
# Asana Task Management
|
||||||
|
|
||||||
|
## Searching Tasks
|
||||||
|
|
||||||
|
To search for tasks:
|
||||||
|
1. Use mcp__plugin_asana_asana__asana_search_tasks
|
||||||
|
2. Provide search filters (assignee, project, etc.)
|
||||||
|
3. Display results to user
|
||||||
|
|
||||||
|
## Creating Tasks
|
||||||
|
|
||||||
|
To create a task:
|
||||||
|
1. Gather task details:
|
||||||
|
- Title (required)
|
||||||
|
- Description
|
||||||
|
- Project
|
||||||
|
- Assignee
|
||||||
|
- Due date
|
||||||
|
2. Use mcp__plugin_asana_asana__asana_create_task
|
||||||
|
3. Show confirmation with task link
|
||||||
|
```
|
||||||
|
|
||||||
|
## Using Tools in Agents
|
||||||
|
|
||||||
|
### Agent Configuration
|
||||||
|
|
||||||
|
Agents can use MCP tools autonomously without pre-allowing them:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
name: asana-status-updater
|
||||||
|
description: This agent should be used when the user asks to "update Asana status", "generate project report", or "sync Asana tasks"
|
||||||
|
model: inherit
|
||||||
|
color: blue
|
||||||
|
---
|
||||||
|
|
||||||
|
## Role
|
||||||
|
|
||||||
|
Autonomous agent for generating Asana project status reports.
|
||||||
|
|
||||||
|
## Process
|
||||||
|
|
||||||
|
1. **Query tasks**: Use mcp__plugin_asana_asana__asana_search_tasks to get all tasks
|
||||||
|
2. **Analyze progress**: Calculate completion rates and identify blockers
|
||||||
|
3. **Generate report**: Create formatted status update
|
||||||
|
4. **Update Asana**: Use mcp__plugin_asana_asana__asana_create_comment to post report
|
||||||
|
|
||||||
|
## Available Tools
|
||||||
|
|
||||||
|
The agent has access to all Asana MCP tools without pre-approval.
|
||||||
|
```
|
||||||
|
|
||||||
|
### Agent Tool Access
|
||||||
|
|
||||||
|
Agents have broader tool access than commands:
|
||||||
|
- Can use any tool Claude determines is necessary
|
||||||
|
- Don't need pre-allowed lists
|
||||||
|
- Should document which tools they typically use
|
||||||
|
|
||||||
|
## Tool Call Patterns
|
||||||
|
|
||||||
|
### Pattern 1: Simple Tool Call
|
||||||
|
|
||||||
|
Single tool call with validation:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
Steps:
|
||||||
|
1. Validate user provided required fields
|
||||||
|
2. Call mcp__plugin_api_server__create_item with validated data
|
||||||
|
3. Check for errors
|
||||||
|
4. Display confirmation
|
||||||
|
```
|
||||||
|
|
||||||
|
### Pattern 2: Sequential Tools
|
||||||
|
|
||||||
|
Chain multiple tool calls:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
Steps:
|
||||||
|
1. Search for existing items: mcp__plugin_api_server__search
|
||||||
|
2. If not found, create new: mcp__plugin_api_server__create
|
||||||
|
3. Add metadata: mcp__plugin_api_server__update_metadata
|
||||||
|
4. Return final item ID
|
||||||
|
```
|
||||||
|
|
||||||
|
### Pattern 3: Batch Operations
|
||||||
|
|
||||||
|
Multiple calls with same tool:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
Steps:
|
||||||
|
1. Get list of items to process
|
||||||
|
2. For each item:
|
||||||
|
- Call mcp__plugin_api_server__update_item
|
||||||
|
- Track success/failure
|
||||||
|
3. Report results summary
|
||||||
|
```
|
||||||
|
|
||||||
|
### Pattern 4: Error Handling
|
||||||
|
|
||||||
|
Graceful error handling:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
Steps:
|
||||||
|
1. Try to call mcp__plugin_api_server__get_data
|
||||||
|
2. If error (rate limit, network, etc.):
|
||||||
|
- Wait and retry (max 3 attempts)
|
||||||
|
- If still failing, inform user
|
||||||
|
- Suggest checking configuration
|
||||||
|
3. On success, process data
|
||||||
|
```
|
||||||
|
|
||||||
|
## Tool Parameters
|
||||||
|
|
||||||
|
### Understanding Tool Schemas
|
||||||
|
|
||||||
|
Each MCP tool has a schema defining its parameters. View with `/mcp`.
|
||||||
|
|
||||||
|
**Example schema:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"name": "asana_create_task",
|
||||||
|
"description": "Create a new Asana task",
|
||||||
|
"inputSchema": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"name": {
|
||||||
|
"type": "string",
|
||||||
|
"description": "Task title"
|
||||||
|
},
|
||||||
|
"notes": {
|
||||||
|
"type": "string",
|
||||||
|
"description": "Task description"
|
||||||
|
},
|
||||||
|
"workspace": {
|
||||||
|
"type": "string",
|
||||||
|
"description": "Workspace GID"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"required": ["name", "workspace"]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Calling Tools with Parameters
|
||||||
|
|
||||||
|
Claude automatically structures tool calls based on schema:
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
// Claude generates this internally
|
||||||
|
{
|
||||||
|
toolName: "mcp__plugin_asana_asana__asana_create_task",
|
||||||
|
input: {
|
||||||
|
name: "Review PR #123",
|
||||||
|
notes: "Code review for new feature",
|
||||||
|
workspace: "12345",
|
||||||
|
assignee: "67890",
|
||||||
|
due_on: "2025-01-15"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Parameter Validation
|
||||||
|
|
||||||
|
**In commands, validate before calling:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
Steps:
|
||||||
|
1. Check required parameters:
|
||||||
|
- Title is not empty
|
||||||
|
- Workspace ID is provided
|
||||||
|
- Due date is valid format (YYYY-MM-DD)
|
||||||
|
2. If validation fails, ask user to provide missing data
|
||||||
|
3. If validation passes, call MCP tool
|
||||||
|
4. Handle tool errors gracefully
|
||||||
|
```
|
||||||
|
|
||||||
|
## Response Handling
|
||||||
|
|
||||||
|
### Success Responses
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
Steps:
|
||||||
|
1. Call MCP tool
|
||||||
|
2. On success:
|
||||||
|
- Extract relevant data from response
|
||||||
|
- Format for user display
|
||||||
|
- Provide confirmation message
|
||||||
|
- Include relevant links or IDs
|
||||||
|
```
|
||||||
|
|
||||||
|
### Error Responses
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
Steps:
|
||||||
|
1. Call MCP tool
|
||||||
|
2. On error:
|
||||||
|
- Check error type (auth, rate limit, validation, etc.)
|
||||||
|
- Provide helpful error message
|
||||||
|
- Suggest remediation steps
|
||||||
|
- Don't expose internal error details to user
|
||||||
|
```
|
||||||
|
|
||||||
|
### Partial Success
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
Steps:
|
||||||
|
1. Batch operation with multiple MCP calls
|
||||||
|
2. Track successes and failures separately
|
||||||
|
3. Report summary:
|
||||||
|
- "Successfully processed 8 of 10 items"
|
||||||
|
- "Failed items: [item1, item2] due to [reason]"
|
||||||
|
- Suggest retry or manual intervention
|
||||||
|
```
|
||||||
|
|
||||||
|
## Performance Optimization
|
||||||
|
|
||||||
|
### Batching Requests
|
||||||
|
|
||||||
|
**Good: Single query with filters**
|
||||||
|
```markdown
|
||||||
|
Steps:
|
||||||
|
1. Call mcp__plugin_api_server__search with filters:
|
||||||
|
- project_id: "123"
|
||||||
|
- status: "active"
|
||||||
|
- limit: 100
|
||||||
|
2. Process all results
|
||||||
|
```
|
||||||
|
|
||||||
|
**Avoid: Many individual queries**
|
||||||
|
```markdown
|
||||||
|
Steps:
|
||||||
|
1. For each item ID:
|
||||||
|
- Call mcp__plugin_api_server__get_item
|
||||||
|
- Process item
|
||||||
|
```
|
||||||
|
|
||||||
|
### Caching Results
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
Steps:
|
||||||
|
1. Call expensive MCP operation: mcp__plugin_api_server__analyze
|
||||||
|
2. Store results in variable for reuse
|
||||||
|
3. Use cached results for subsequent operations
|
||||||
|
4. Only re-fetch if data changes
|
||||||
|
```
|
||||||
|
|
||||||
|
### Parallel Tool Calls
|
||||||
|
|
||||||
|
When tools don't depend on each other, call in parallel:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
Steps:
|
||||||
|
1. Make parallel calls (Claude handles this automatically):
|
||||||
|
- mcp__plugin_api_server__get_project
|
||||||
|
- mcp__plugin_api_server__get_users
|
||||||
|
- mcp__plugin_api_server__get_tags
|
||||||
|
2. Wait for all to complete
|
||||||
|
3. Combine results
|
||||||
|
```
|
||||||
|
|
||||||
|
## Integration Best Practices
|
||||||
|
|
||||||
|
### User Experience
|
||||||
|
|
||||||
|
**Provide feedback:**
|
||||||
|
```markdown
|
||||||
|
Steps:
|
||||||
|
1. Inform user: "Searching Asana tasks..."
|
||||||
|
2. Call mcp__plugin_asana_asana__asana_search_tasks
|
||||||
|
3. Show progress: "Found 15 tasks, analyzing..."
|
||||||
|
4. Present results
|
||||||
|
```
|
||||||
|
|
||||||
|
**Handle long operations:**
|
||||||
|
```markdown
|
||||||
|
Steps:
|
||||||
|
1. Warn user: "This may take a minute..."
|
||||||
|
2. Break into smaller steps with updates
|
||||||
|
3. Show incremental progress
|
||||||
|
4. Final summary when complete
|
||||||
|
```
|
||||||
|
|
||||||
|
### Error Messages
|
||||||
|
|
||||||
|
**Good error messages:**
|
||||||
|
```
|
||||||
|
❌ "Could not create task. Please check:
|
||||||
|
1. You're logged into Asana
|
||||||
|
2. You have access to workspace 'Engineering'
|
||||||
|
3. The project 'Q1 Goals' exists"
|
||||||
|
```
|
||||||
|
|
||||||
|
**Poor error messages:**
|
||||||
|
```
|
||||||
|
❌ "Error: MCP tool returned 403"
|
||||||
|
```
|
||||||
|
|
||||||
|
### Documentation
|
||||||
|
|
||||||
|
**Document MCP tool usage in command:**
|
||||||
|
```markdown
|
||||||
|
## MCP Tools Used
|
||||||
|
|
||||||
|
This command uses the following Asana MCP tools:
|
||||||
|
- **asana_search_tasks**: Search for tasks matching criteria
|
||||||
|
- **asana_create_task**: Create new task with details
|
||||||
|
- **asana_update_task**: Update existing task properties
|
||||||
|
|
||||||
|
Ensure you're authenticated to Asana before running this command.
|
||||||
|
```
|
||||||
|
|
||||||
|
## Testing Tool Usage
|
||||||
|
|
||||||
|
### Local Testing
|
||||||
|
|
||||||
|
1. **Configure MCP server** in `.mcp.json`
|
||||||
|
2. **Install plugin locally** in `.claude-plugin/`
|
||||||
|
3. **Verify tools available** with `/mcp`
|
||||||
|
4. **Test command** that uses tools
|
||||||
|
5. **Check debug output**: `claude --debug`
|
||||||
|
|
||||||
|
### Test Scenarios
|
||||||
|
|
||||||
|
**Test successful calls:**
|
||||||
|
```markdown
|
||||||
|
Steps:
|
||||||
|
1. Create test data in external service
|
||||||
|
2. Run command that queries this data
|
||||||
|
3. Verify correct results returned
|
||||||
|
```
|
||||||
|
|
||||||
|
**Test error cases:**
|
||||||
|
```markdown
|
||||||
|
Steps:
|
||||||
|
1. Test with missing authentication
|
||||||
|
2. Test with invalid parameters
|
||||||
|
3. Test with non-existent resources
|
||||||
|
4. Verify graceful error handling
|
||||||
|
```
|
||||||
|
|
||||||
|
**Test edge cases:**
|
||||||
|
```markdown
|
||||||
|
Steps:
|
||||||
|
1. Test with empty results
|
||||||
|
2. Test with maximum results
|
||||||
|
3. Test with special characters
|
||||||
|
4. Test with concurrent access
|
||||||
|
```
|
||||||
|
|
||||||
|
## Common Patterns
|
||||||
|
|
||||||
|
### Pattern: CRUD Operations
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
allowed-tools: [
|
||||||
|
"mcp__plugin_api_server__create_item",
|
||||||
|
"mcp__plugin_api_server__read_item",
|
||||||
|
"mcp__plugin_api_server__update_item",
|
||||||
|
"mcp__plugin_api_server__delete_item"
|
||||||
|
]
|
||||||
|
---
|
||||||
|
|
||||||
|
# Item Management
|
||||||
|
|
||||||
|
## Create
|
||||||
|
Use create_item with required fields...
|
||||||
|
|
||||||
|
## Read
|
||||||
|
Use read_item with item ID...
|
||||||
|
|
||||||
|
## Update
|
||||||
|
Use update_item with item ID and changes...
|
||||||
|
|
||||||
|
## Delete
|
||||||
|
Use delete_item with item ID (ask for confirmation first)...
|
||||||
|
```
|
||||||
|
|
||||||
|
### Pattern: Search and Process
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
Steps:
|
||||||
|
1. **Search**: mcp__plugin_api_server__search with filters
|
||||||
|
2. **Filter**: Apply additional local filtering if needed
|
||||||
|
3. **Transform**: Process each result
|
||||||
|
4. **Present**: Format and display to user
|
||||||
|
```
|
||||||
|
|
||||||
|
### Pattern: Multi-Step Workflow
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
Steps:
|
||||||
|
1. **Setup**: Gather all required information
|
||||||
|
2. **Validate**: Check data completeness
|
||||||
|
3. **Execute**: Chain of MCP tool calls:
|
||||||
|
- Create parent resource
|
||||||
|
- Create child resources
|
||||||
|
- Link resources together
|
||||||
|
- Add metadata
|
||||||
|
4. **Verify**: Confirm all steps succeeded
|
||||||
|
5. **Report**: Provide summary to user
|
||||||
|
```
|
||||||
|
|
||||||
|
## Troubleshooting
|
||||||
|
|
||||||
|
### Tools Not Available
|
||||||
|
|
||||||
|
**Check:**
|
||||||
|
- MCP server configured correctly
|
||||||
|
- Server connected (check `/mcp`)
|
||||||
|
- Tool names match exactly (case-sensitive)
|
||||||
|
- Restart Claude Code after config changes
|
||||||
|
|
||||||
|
### Tool Calls Failing
|
||||||
|
|
||||||
|
**Check:**
|
||||||
|
- Authentication is valid
|
||||||
|
- Parameters match tool schema
|
||||||
|
- Required parameters provided
|
||||||
|
- Check `claude --debug` logs
|
||||||
|
|
||||||
|
### Performance Issues
|
||||||
|
|
||||||
|
**Check:**
|
||||||
|
- Batching queries instead of individual calls
|
||||||
|
- Caching results when appropriate
|
||||||
|
- Not making unnecessary tool calls
|
||||||
|
- Parallel calls when possible
|
||||||
|
|
||||||
|
## Conclusion
|
||||||
|
|
||||||
|
Effective MCP tool usage requires:
|
||||||
|
1. **Understanding tool schemas** via `/mcp`
|
||||||
|
2. **Pre-allowing tools** in commands appropriately
|
||||||
|
3. **Handling errors gracefully**
|
||||||
|
4. **Optimizing performance** with batching and caching
|
||||||
|
5. **Providing good UX** with feedback and clear errors
|
||||||
|
6. **Testing thoroughly** before deployment
|
||||||
|
|
||||||
|
Follow these patterns for robust MCP tool integration in your plugin commands and agents.
|
||||||
543
skills/plugin-settings/SKILL.md
Normal file
543
skills/plugin-settings/SKILL.md
Normal file
@@ -0,0 +1,543 @@
|
|||||||
|
---
|
||||||
|
name: plugin-settings
|
||||||
|
description: This skill should be used when the user asks about "plugin settings", "store plugin configuration", "user-configurable plugin", ".local.md files", "plugin state files", "read YAML frontmatter", "per-project plugin settings", or wants to make plugin behavior configurable. Documents the .claude/plugin-name.local.md pattern for storing plugin-specific configuration with YAML frontmatter and markdown content.
|
||||||
|
---
|
||||||
|
|
||||||
|
# Plugin Settings Pattern for Claude Code Plugins
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
|
||||||
|
Plugins can store user-configurable settings and state in `.claude/plugin-name.local.md` files within the project directory. This pattern uses YAML frontmatter for structured configuration and markdown content for prompts or additional context.
|
||||||
|
|
||||||
|
**Key characteristics:**
|
||||||
|
- File location: `.claude/plugin-name.local.md` in project root
|
||||||
|
- Structure: YAML frontmatter + markdown body
|
||||||
|
- Purpose: Per-project plugin configuration and state
|
||||||
|
- Usage: Read from hooks, commands, and agents
|
||||||
|
- Lifecycle: User-managed (not in git, should be in `.gitignore`)
|
||||||
|
|
||||||
|
## File Structure
|
||||||
|
|
||||||
|
### Basic Template
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
enabled: true
|
||||||
|
setting1: value1
|
||||||
|
setting2: value2
|
||||||
|
numeric_setting: 42
|
||||||
|
list_setting: ["item1", "item2"]
|
||||||
|
---
|
||||||
|
|
||||||
|
# Additional Context
|
||||||
|
|
||||||
|
This markdown body can contain:
|
||||||
|
- Task descriptions
|
||||||
|
- Additional instructions
|
||||||
|
- Prompts to feed back to Claude
|
||||||
|
- Documentation or notes
|
||||||
|
```
|
||||||
|
|
||||||
|
### Example: Plugin State File
|
||||||
|
|
||||||
|
**.claude/my-plugin.local.md:**
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
enabled: true
|
||||||
|
strict_mode: false
|
||||||
|
max_retries: 3
|
||||||
|
notification_level: info
|
||||||
|
coordinator_session: team-leader
|
||||||
|
---
|
||||||
|
|
||||||
|
# Plugin Configuration
|
||||||
|
|
||||||
|
This plugin is configured for standard validation mode.
|
||||||
|
Contact @team-lead with questions.
|
||||||
|
```
|
||||||
|
|
||||||
|
## Reading Settings Files
|
||||||
|
|
||||||
|
### From Hooks (Bash Scripts)
|
||||||
|
|
||||||
|
**Pattern: Check existence and parse frontmatter**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
# Define state file path
|
||||||
|
STATE_FILE=".claude/my-plugin.local.md"
|
||||||
|
|
||||||
|
# Quick exit if file doesn't exist
|
||||||
|
if [[ ! -f "$STATE_FILE" ]]; then
|
||||||
|
exit 0 # Plugin not configured, skip
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Parse YAML frontmatter (between --- markers)
|
||||||
|
FRONTMATTER=$(sed -n '/^---$/,/^---$/{ /^---$/d; p; }' "$STATE_FILE")
|
||||||
|
|
||||||
|
# Extract individual fields
|
||||||
|
ENABLED=$(echo "$FRONTMATTER" | grep '^enabled:' | sed 's/enabled: *//' | sed 's/^"\(.*\)"$/\1/')
|
||||||
|
STRICT_MODE=$(echo "$FRONTMATTER" | grep '^strict_mode:' | sed 's/strict_mode: *//' | sed 's/^"\(.*\)"$/\1/')
|
||||||
|
|
||||||
|
# Check if enabled
|
||||||
|
if [[ "$ENABLED" != "true" ]]; then
|
||||||
|
exit 0 # Disabled
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Use configuration in hook logic
|
||||||
|
if [[ "$STRICT_MODE" == "true" ]]; then
|
||||||
|
# Apply strict validation
|
||||||
|
# ...
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
See `examples/read-settings-hook.sh` for complete working example.
|
||||||
|
|
||||||
|
### From Commands
|
||||||
|
|
||||||
|
Commands can read settings files to customize behavior:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Process data with plugin
|
||||||
|
allowed-tools: ["Read", "Bash"]
|
||||||
|
---
|
||||||
|
|
||||||
|
# Process Command
|
||||||
|
|
||||||
|
Steps:
|
||||||
|
1. Check if settings exist at `.claude/my-plugin.local.md`
|
||||||
|
2. Read configuration using Read tool
|
||||||
|
3. Parse YAML frontmatter to extract settings
|
||||||
|
4. Apply settings to processing logic
|
||||||
|
5. Execute with configured behavior
|
||||||
|
```
|
||||||
|
|
||||||
|
### From Agents
|
||||||
|
|
||||||
|
Agents can reference settings in their instructions:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
name: configured-agent
|
||||||
|
description: Agent that adapts to project settings
|
||||||
|
---
|
||||||
|
|
||||||
|
Check for plugin settings at `.claude/my-plugin.local.md`.
|
||||||
|
If present, parse YAML frontmatter and adapt behavior according to:
|
||||||
|
- enabled: Whether plugin is active
|
||||||
|
- mode: Processing mode (strict, standard, lenient)
|
||||||
|
- Additional configuration fields
|
||||||
|
```
|
||||||
|
|
||||||
|
## Parsing Techniques
|
||||||
|
|
||||||
|
### Extract Frontmatter
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Extract everything between --- markers
|
||||||
|
FRONTMATTER=$(sed -n '/^---$/,/^---$/{ /^---$/d; p; }' "$FILE")
|
||||||
|
```
|
||||||
|
|
||||||
|
### Read Individual Fields
|
||||||
|
|
||||||
|
**String fields:**
|
||||||
|
```bash
|
||||||
|
VALUE=$(echo "$FRONTMATTER" | grep '^field_name:' | sed 's/field_name: *//' | sed 's/^"\(.*\)"$/\1/')
|
||||||
|
```
|
||||||
|
|
||||||
|
**Boolean fields:**
|
||||||
|
```bash
|
||||||
|
ENABLED=$(echo "$FRONTMATTER" | grep '^enabled:' | sed 's/enabled: *//')
|
||||||
|
# Compare: if [[ "$ENABLED" == "true" ]]; then
|
||||||
|
```
|
||||||
|
|
||||||
|
**Numeric fields:**
|
||||||
|
```bash
|
||||||
|
MAX=$(echo "$FRONTMATTER" | grep '^max_value:' | sed 's/max_value: *//')
|
||||||
|
# Use: if [[ $MAX -gt 100 ]]; then
|
||||||
|
```
|
||||||
|
|
||||||
|
### Read Markdown Body
|
||||||
|
|
||||||
|
Extract content after second `---`:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Get everything after closing ---
|
||||||
|
BODY=$(awk '/^---$/{i++; next} i>=2' "$FILE")
|
||||||
|
```
|
||||||
|
|
||||||
|
## Common Patterns
|
||||||
|
|
||||||
|
### Pattern 1: Temporarily Active Hooks
|
||||||
|
|
||||||
|
Use settings file to control hook activation:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
STATE_FILE=".claude/security-scan.local.md"
|
||||||
|
|
||||||
|
# Quick exit if not configured
|
||||||
|
if [[ ! -f "$STATE_FILE" ]]; then
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Read enabled flag
|
||||||
|
FRONTMATTER=$(sed -n '/^---$/,/^---$/{ /^---$/d; p; }' "$STATE_FILE")
|
||||||
|
ENABLED=$(echo "$FRONTMATTER" | grep '^enabled:' | sed 's/enabled: *//')
|
||||||
|
|
||||||
|
if [[ "$ENABLED" != "true" ]]; then
|
||||||
|
exit 0 # Disabled
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Run hook logic
|
||||||
|
# ...
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use case:** Enable/disable hooks without editing hooks.json (requires restart).
|
||||||
|
|
||||||
|
### Pattern 2: Agent State Management
|
||||||
|
|
||||||
|
Store agent-specific state and configuration:
|
||||||
|
|
||||||
|
**.claude/multi-agent-swarm.local.md:**
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
agent_name: auth-agent
|
||||||
|
task_number: 3.5
|
||||||
|
pr_number: 1234
|
||||||
|
coordinator_session: team-leader
|
||||||
|
enabled: true
|
||||||
|
dependencies: ["Task 3.4"]
|
||||||
|
---
|
||||||
|
|
||||||
|
# Task Assignment
|
||||||
|
|
||||||
|
Implement JWT authentication for the API.
|
||||||
|
|
||||||
|
**Success Criteria:**
|
||||||
|
- Authentication endpoints created
|
||||||
|
- Tests passing
|
||||||
|
- PR created and CI green
|
||||||
|
```
|
||||||
|
|
||||||
|
Read from hooks to coordinate agents:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
AGENT_NAME=$(echo "$FRONTMATTER" | grep '^agent_name:' | sed 's/agent_name: *//')
|
||||||
|
COORDINATOR=$(echo "$FRONTMATTER" | grep '^coordinator_session:' | sed 's/coordinator_session: *//')
|
||||||
|
|
||||||
|
# Send notification to coordinator
|
||||||
|
tmux send-keys -t "$COORDINATOR" "Agent $AGENT_NAME completed task" Enter
|
||||||
|
```
|
||||||
|
|
||||||
|
### Pattern 3: Configuration-Driven Behavior
|
||||||
|
|
||||||
|
**.claude/my-plugin.local.md:**
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
validation_level: strict
|
||||||
|
max_file_size: 1000000
|
||||||
|
allowed_extensions: [".js", ".ts", ".tsx"]
|
||||||
|
enable_logging: true
|
||||||
|
---
|
||||||
|
|
||||||
|
# Validation Configuration
|
||||||
|
|
||||||
|
Strict mode enabled for this project.
|
||||||
|
All writes validated against security policies.
|
||||||
|
```
|
||||||
|
|
||||||
|
Use in hooks or commands:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
LEVEL=$(echo "$FRONTMATTER" | grep '^validation_level:' | sed 's/validation_level: *//')
|
||||||
|
|
||||||
|
case "$LEVEL" in
|
||||||
|
strict)
|
||||||
|
# Apply strict validation
|
||||||
|
;;
|
||||||
|
standard)
|
||||||
|
# Apply standard validation
|
||||||
|
;;
|
||||||
|
lenient)
|
||||||
|
# Apply lenient validation
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
```
|
||||||
|
|
||||||
|
## Creating Settings Files
|
||||||
|
|
||||||
|
### From Commands
|
||||||
|
|
||||||
|
Commands can create settings files:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
# Setup Command
|
||||||
|
|
||||||
|
Steps:
|
||||||
|
1. Ask user for configuration preferences
|
||||||
|
2. Create `.claude/my-plugin.local.md` with YAML frontmatter
|
||||||
|
3. Set appropriate values based on user input
|
||||||
|
4. Inform user that settings are saved
|
||||||
|
5. Remind user to restart Claude Code for hooks to recognize changes
|
||||||
|
```
|
||||||
|
|
||||||
|
### Template Generation
|
||||||
|
|
||||||
|
Provide template in plugin README:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
## Configuration
|
||||||
|
|
||||||
|
Create `.claude/my-plugin.local.md` in your project:
|
||||||
|
|
||||||
|
\`\`\`markdown
|
||||||
|
---
|
||||||
|
enabled: true
|
||||||
|
mode: standard
|
||||||
|
max_retries: 3
|
||||||
|
---
|
||||||
|
|
||||||
|
# Plugin Configuration
|
||||||
|
|
||||||
|
Your settings are active.
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
After creating or editing, restart Claude Code for changes to take effect.
|
||||||
|
```
|
||||||
|
|
||||||
|
## Best Practices
|
||||||
|
|
||||||
|
### File Naming
|
||||||
|
|
||||||
|
✅ **DO:**
|
||||||
|
- Use `.claude/plugin-name.local.md` format
|
||||||
|
- Match plugin name exactly
|
||||||
|
- Use `.local.md` suffix for user-local files
|
||||||
|
|
||||||
|
❌ **DON'T:**
|
||||||
|
- Use different directory (not `.claude/`)
|
||||||
|
- Use inconsistent naming
|
||||||
|
- Use `.md` without `.local` (might be committed)
|
||||||
|
|
||||||
|
### Gitignore
|
||||||
|
|
||||||
|
Always add to `.gitignore`:
|
||||||
|
|
||||||
|
```gitignore
|
||||||
|
.claude/*.local.md
|
||||||
|
.claude/*.local.json
|
||||||
|
```
|
||||||
|
|
||||||
|
Document this in plugin README.
|
||||||
|
|
||||||
|
### Defaults
|
||||||
|
|
||||||
|
Provide sensible defaults when settings file doesn't exist:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
if [[ ! -f "$STATE_FILE" ]]; then
|
||||||
|
# Use defaults
|
||||||
|
ENABLED=true
|
||||||
|
MODE=standard
|
||||||
|
else
|
||||||
|
# Read from file
|
||||||
|
# ...
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
### Validation
|
||||||
|
|
||||||
|
Validate settings values:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
MAX=$(echo "$FRONTMATTER" | grep '^max_value:' | sed 's/max_value: *//')
|
||||||
|
|
||||||
|
# Validate numeric range
|
||||||
|
if ! [[ "$MAX" =~ ^[0-9]+$ ]] || [[ $MAX -lt 1 ]] || [[ $MAX -gt 100 ]]; then
|
||||||
|
echo "⚠️ Invalid max_value in settings (must be 1-100)" >&2
|
||||||
|
MAX=10 # Use default
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
### Restart Requirement
|
||||||
|
|
||||||
|
**Important:** Settings changes require Claude Code restart.
|
||||||
|
|
||||||
|
Document in your README:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
## Changing Settings
|
||||||
|
|
||||||
|
After editing `.claude/my-plugin.local.md`:
|
||||||
|
1. Save the file
|
||||||
|
2. Exit Claude Code
|
||||||
|
3. Restart: `claude` or `cc`
|
||||||
|
4. New settings will be loaded
|
||||||
|
```
|
||||||
|
|
||||||
|
Hooks cannot be hot-swapped within a session.
|
||||||
|
|
||||||
|
## Security Considerations
|
||||||
|
|
||||||
|
### Sanitize User Input
|
||||||
|
|
||||||
|
When writing settings files from user input:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Escape quotes in user input
|
||||||
|
SAFE_VALUE=$(echo "$USER_INPUT" | sed 's/"/\\"/g')
|
||||||
|
|
||||||
|
# Write to file
|
||||||
|
cat > "$STATE_FILE" <<EOF
|
||||||
|
---
|
||||||
|
user_setting: "$SAFE_VALUE"
|
||||||
|
---
|
||||||
|
EOF
|
||||||
|
```
|
||||||
|
|
||||||
|
### Validate File Paths
|
||||||
|
|
||||||
|
If settings contain file paths:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
FILE_PATH=$(echo "$FRONTMATTER" | grep '^data_file:' | sed 's/data_file: *//')
|
||||||
|
|
||||||
|
# Check for path traversal
|
||||||
|
if [[ "$FILE_PATH" == *".."* ]]; then
|
||||||
|
echo "⚠️ Invalid path in settings (path traversal)" >&2
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
### Permissions
|
||||||
|
|
||||||
|
Settings files should be:
|
||||||
|
- Readable by user only (`chmod 600`)
|
||||||
|
- Not committed to git
|
||||||
|
- Not shared between users
|
||||||
|
|
||||||
|
## Real-World Examples
|
||||||
|
|
||||||
|
### multi-agent-swarm Plugin
|
||||||
|
|
||||||
|
**.claude/multi-agent-swarm.local.md:**
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
agent_name: auth-implementation
|
||||||
|
task_number: 3.5
|
||||||
|
pr_number: 1234
|
||||||
|
coordinator_session: team-leader
|
||||||
|
enabled: true
|
||||||
|
dependencies: ["Task 3.4"]
|
||||||
|
additional_instructions: Use JWT tokens, not sessions
|
||||||
|
---
|
||||||
|
|
||||||
|
# Task: Implement Authentication
|
||||||
|
|
||||||
|
Build JWT-based authentication for the REST API.
|
||||||
|
Coordinate with auth-agent on shared types.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Hook usage (agent-stop-notification.sh):**
|
||||||
|
- Checks if file exists (line 15-18: quick exit if not)
|
||||||
|
- Parses frontmatter to get coordinator_session, agent_name, enabled
|
||||||
|
- Sends notifications to coordinator if enabled
|
||||||
|
- Allows quick activation/deactivation via `enabled: true/false`
|
||||||
|
|
||||||
|
### ralph-wiggum Plugin
|
||||||
|
|
||||||
|
**.claude/ralph-loop.local.md:**
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
iteration: 1
|
||||||
|
max_iterations: 10
|
||||||
|
completion_promise: "All tests passing and build successful"
|
||||||
|
---
|
||||||
|
|
||||||
|
Fix all the linting errors in the project.
|
||||||
|
Make sure tests pass after each fix.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Hook usage (stop-hook.sh):**
|
||||||
|
- Checks if file exists (line 15-18: quick exit if not active)
|
||||||
|
- Reads iteration count and max_iterations
|
||||||
|
- Extracts completion_promise for loop termination
|
||||||
|
- Reads body as the prompt to feed back
|
||||||
|
- Updates iteration count on each loop
|
||||||
|
|
||||||
|
## Quick Reference
|
||||||
|
|
||||||
|
### File Location
|
||||||
|
|
||||||
|
```
|
||||||
|
project-root/
|
||||||
|
└── .claude/
|
||||||
|
└── plugin-name.local.md
|
||||||
|
```
|
||||||
|
|
||||||
|
### Frontmatter Parsing
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Extract frontmatter
|
||||||
|
FRONTMATTER=$(sed -n '/^---$/,/^---$/{ /^---$/d; p; }' "$FILE")
|
||||||
|
|
||||||
|
# Read field
|
||||||
|
VALUE=$(echo "$FRONTMATTER" | grep '^field:' | sed 's/field: *//' | sed 's/^"\(.*\)"$/\1/')
|
||||||
|
```
|
||||||
|
|
||||||
|
### Body Parsing
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Extract body (after second ---)
|
||||||
|
BODY=$(awk '/^---$/{i++; next} i>=2' "$FILE")
|
||||||
|
```
|
||||||
|
|
||||||
|
### Quick Exit Pattern
|
||||||
|
|
||||||
|
```bash
|
||||||
|
if [[ ! -f ".claude/my-plugin.local.md" ]]; then
|
||||||
|
exit 0 # Not configured
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
## Additional Resources
|
||||||
|
|
||||||
|
### Reference Files
|
||||||
|
|
||||||
|
For detailed implementation patterns:
|
||||||
|
|
||||||
|
- **`references/parsing-techniques.md`** - Complete guide to parsing YAML frontmatter and markdown bodies
|
||||||
|
- **`references/real-world-examples.md`** - Deep dive into multi-agent-swarm and ralph-wiggum implementations
|
||||||
|
|
||||||
|
### Example Files
|
||||||
|
|
||||||
|
Working examples in `examples/`:
|
||||||
|
|
||||||
|
- **`read-settings-hook.sh`** - Hook that reads and uses settings
|
||||||
|
- **`create-settings-command.md`** - Command that creates settings file
|
||||||
|
- **`example-settings.md`** - Template settings file
|
||||||
|
|
||||||
|
### Utility Scripts
|
||||||
|
|
||||||
|
Development tools in `scripts/`:
|
||||||
|
|
||||||
|
- **`validate-settings.sh`** - Validate settings file structure
|
||||||
|
- **`parse-frontmatter.sh`** - Extract frontmatter fields
|
||||||
|
|
||||||
|
## Implementation Workflow
|
||||||
|
|
||||||
|
To add settings to a plugin:
|
||||||
|
|
||||||
|
1. Design settings schema (which fields, types, defaults)
|
||||||
|
2. Create template file in plugin documentation
|
||||||
|
3. Add gitignore entry for `.claude/*.local.md`
|
||||||
|
4. Implement settings parsing in hooks/commands
|
||||||
|
5. Use quick-exit pattern (check file exists, check enabled field)
|
||||||
|
6. Document settings in plugin README with template
|
||||||
|
7. Remind users that changes require Claude Code restart
|
||||||
|
|
||||||
|
Focus on keeping settings simple and providing good defaults when settings file doesn't exist.
|
||||||
98
skills/plugin-settings/examples/create-settings-command.md
Normal file
98
skills/plugin-settings/examples/create-settings-command.md
Normal file
@@ -0,0 +1,98 @@
|
|||||||
|
---
|
||||||
|
description: "Create plugin settings file with user preferences"
|
||||||
|
allowed-tools: ["Write", "AskUserQuestion"]
|
||||||
|
---
|
||||||
|
|
||||||
|
# Create Plugin Settings
|
||||||
|
|
||||||
|
This command helps users create a `.claude/my-plugin.local.md` settings file.
|
||||||
|
|
||||||
|
## Steps
|
||||||
|
|
||||||
|
### Step 1: Ask User for Preferences
|
||||||
|
|
||||||
|
Use AskUserQuestion to gather configuration:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"questions": [
|
||||||
|
{
|
||||||
|
"question": "Enable plugin for this project?",
|
||||||
|
"header": "Enable Plugin",
|
||||||
|
"multiSelect": false,
|
||||||
|
"options": [
|
||||||
|
{
|
||||||
|
"label": "Yes",
|
||||||
|
"description": "Plugin will be active"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "No",
|
||||||
|
"description": "Plugin will be disabled"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"question": "Validation mode?",
|
||||||
|
"header": "Mode",
|
||||||
|
"multiSelect": false,
|
||||||
|
"options": [
|
||||||
|
{
|
||||||
|
"label": "Strict",
|
||||||
|
"description": "Maximum validation and security checks"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "Standard",
|
||||||
|
"description": "Balanced validation (recommended)"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "Lenient",
|
||||||
|
"description": "Minimal validation only"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Step 2: Parse Answers
|
||||||
|
|
||||||
|
Extract answers from AskUserQuestion result:
|
||||||
|
|
||||||
|
- answers["0"]: enabled (Yes/No)
|
||||||
|
- answers["1"]: mode (Strict/Standard/Lenient)
|
||||||
|
|
||||||
|
### Step 3: Create Settings File
|
||||||
|
|
||||||
|
Use Write tool to create `.claude/my-plugin.local.md`:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
enabled: <true if Yes, false if No>
|
||||||
|
validation_mode: <strict, standard, or lenient>
|
||||||
|
max_file_size: 1000000
|
||||||
|
notify_on_errors: true
|
||||||
|
---
|
||||||
|
|
||||||
|
# Plugin Configuration
|
||||||
|
|
||||||
|
Your plugin is configured with <mode> validation mode.
|
||||||
|
|
||||||
|
To modify settings, edit this file and restart Claude Code.
|
||||||
|
```
|
||||||
|
|
||||||
|
### Step 4: Inform User
|
||||||
|
|
||||||
|
Tell the user:
|
||||||
|
- Settings file created at `.claude/my-plugin.local.md`
|
||||||
|
- Current configuration summary
|
||||||
|
- How to edit manually if needed
|
||||||
|
- Reminder: Restart Claude Code for changes to take effect
|
||||||
|
- Settings file is gitignored (won't be committed)
|
||||||
|
|
||||||
|
## Implementation Notes
|
||||||
|
|
||||||
|
Always validate user input before writing:
|
||||||
|
- Check mode is valid
|
||||||
|
- Validate numeric fields are numbers
|
||||||
|
- Ensure paths don't have traversal attempts
|
||||||
|
- Sanitize any free-text fields
|
||||||
159
skills/plugin-settings/examples/example-settings.md
Normal file
159
skills/plugin-settings/examples/example-settings.md
Normal file
@@ -0,0 +1,159 @@
|
|||||||
|
# Example Plugin Settings File
|
||||||
|
|
||||||
|
## Template: Basic Configuration
|
||||||
|
|
||||||
|
**.claude/my-plugin.local.md:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
enabled: true
|
||||||
|
mode: standard
|
||||||
|
---
|
||||||
|
|
||||||
|
# My Plugin Configuration
|
||||||
|
|
||||||
|
Plugin is active in standard mode.
|
||||||
|
```
|
||||||
|
|
||||||
|
## Template: Advanced Configuration
|
||||||
|
|
||||||
|
**.claude/my-plugin.local.md:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
enabled: true
|
||||||
|
strict_mode: false
|
||||||
|
max_file_size: 1000000
|
||||||
|
allowed_extensions: [".js", ".ts", ".tsx"]
|
||||||
|
enable_logging: true
|
||||||
|
notification_level: info
|
||||||
|
retry_attempts: 3
|
||||||
|
timeout_seconds: 60
|
||||||
|
custom_path: "/path/to/data"
|
||||||
|
---
|
||||||
|
|
||||||
|
# My Plugin Advanced Configuration
|
||||||
|
|
||||||
|
This project uses custom plugin configuration with:
|
||||||
|
- Standard validation mode
|
||||||
|
- 1MB file size limit
|
||||||
|
- JavaScript/TypeScript files allowed
|
||||||
|
- Info-level logging
|
||||||
|
- 3 retry attempts
|
||||||
|
|
||||||
|
## Additional Notes
|
||||||
|
|
||||||
|
Contact @team-lead with questions about this configuration.
|
||||||
|
```
|
||||||
|
|
||||||
|
## Template: Agent State File
|
||||||
|
|
||||||
|
**.claude/multi-agent-swarm.local.md:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
agent_name: database-implementation
|
||||||
|
task_number: 4.2
|
||||||
|
pr_number: 5678
|
||||||
|
coordinator_session: team-leader
|
||||||
|
enabled: true
|
||||||
|
dependencies: ["Task 3.5", "Task 4.1"]
|
||||||
|
additional_instructions: "Use PostgreSQL, not MySQL"
|
||||||
|
---
|
||||||
|
|
||||||
|
# Task Assignment: Database Schema Implementation
|
||||||
|
|
||||||
|
Implement the database schema for the new features module.
|
||||||
|
|
||||||
|
## Requirements
|
||||||
|
|
||||||
|
- Create migration files
|
||||||
|
- Add indexes for performance
|
||||||
|
- Write tests for constraints
|
||||||
|
- Document schema in README
|
||||||
|
|
||||||
|
## Success Criteria
|
||||||
|
|
||||||
|
- Migrations run successfully
|
||||||
|
- All tests pass
|
||||||
|
- PR created with CI green
|
||||||
|
- Schema documented
|
||||||
|
|
||||||
|
## Coordination
|
||||||
|
|
||||||
|
Depends on:
|
||||||
|
- Task 3.5: API endpoint definitions
|
||||||
|
- Task 4.1: Data model design
|
||||||
|
|
||||||
|
Report status to coordinator session 'team-leader'.
|
||||||
|
```
|
||||||
|
|
||||||
|
## Template: Feature Flag Pattern
|
||||||
|
|
||||||
|
**.claude/experimental-features.local.md:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
enabled: true
|
||||||
|
features:
|
||||||
|
- ai_suggestions
|
||||||
|
- auto_formatting
|
||||||
|
- advanced_refactoring
|
||||||
|
experimental_mode: false
|
||||||
|
---
|
||||||
|
|
||||||
|
# Experimental Features Configuration
|
||||||
|
|
||||||
|
Current enabled features:
|
||||||
|
- AI-powered code suggestions
|
||||||
|
- Automatic code formatting
|
||||||
|
- Advanced refactoring tools
|
||||||
|
|
||||||
|
Experimental mode is OFF (stable features only).
|
||||||
|
```
|
||||||
|
|
||||||
|
## Usage in Hooks
|
||||||
|
|
||||||
|
These templates can be read by hooks:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Check if plugin is configured
|
||||||
|
if [[ ! -f ".claude/my-plugin.local.md" ]]; then
|
||||||
|
exit 0 # Not configured, skip hook
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Read settings
|
||||||
|
FRONTMATTER=$(sed -n '/^---$/,/^---$/{ /^---$/d; p; }' ".claude/my-plugin.local.md")
|
||||||
|
ENABLED=$(echo "$FRONTMATTER" | grep '^enabled:' | sed 's/enabled: *//')
|
||||||
|
|
||||||
|
# Apply settings
|
||||||
|
if [[ "$ENABLED" == "true" ]]; then
|
||||||
|
# Hook is active
|
||||||
|
# ...
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
## Gitignore
|
||||||
|
|
||||||
|
Always add to project `.gitignore`:
|
||||||
|
|
||||||
|
```gitignore
|
||||||
|
# Plugin settings (user-local, not committed)
|
||||||
|
.claude/*.local.md
|
||||||
|
.claude/*.local.json
|
||||||
|
```
|
||||||
|
|
||||||
|
## Editing Settings
|
||||||
|
|
||||||
|
Users can edit settings files manually:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Edit settings
|
||||||
|
vim .claude/my-plugin.local.md
|
||||||
|
|
||||||
|
# Changes take effect after restart
|
||||||
|
exit # Exit Claude Code
|
||||||
|
claude # Restart
|
||||||
|
```
|
||||||
|
|
||||||
|
Changes require Claude Code restart - hooks can't be hot-swapped.
|
||||||
65
skills/plugin-settings/examples/read-settings-hook.sh
Executable file
65
skills/plugin-settings/examples/read-settings-hook.sh
Executable file
@@ -0,0 +1,65 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
# Example hook that reads plugin settings from .claude/my-plugin.local.md
|
||||||
|
# Demonstrates the complete pattern for settings-driven hook behavior
|
||||||
|
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
# Define settings file path
|
||||||
|
SETTINGS_FILE=".claude/my-plugin.local.md"
|
||||||
|
|
||||||
|
# Quick exit if settings file doesn't exist
|
||||||
|
if [[ ! -f "$SETTINGS_FILE" ]]; then
|
||||||
|
# Plugin not configured - use defaults or skip
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Parse YAML frontmatter (everything between --- markers)
|
||||||
|
FRONTMATTER=$(sed -n '/^---$/,/^---$/{ /^---$/d; p; }' "$SETTINGS_FILE")
|
||||||
|
|
||||||
|
# Extract configuration fields
|
||||||
|
ENABLED=$(echo "$FRONTMATTER" | grep '^enabled:' | sed 's/enabled: *//' | sed 's/^"\(.*\)"$/\1/')
|
||||||
|
STRICT_MODE=$(echo "$FRONTMATTER" | grep '^strict_mode:' | sed 's/strict_mode: *//' | sed 's/^"\(.*\)"$/\1/')
|
||||||
|
MAX_SIZE=$(echo "$FRONTMATTER" | grep '^max_file_size:' | sed 's/max_file_size: *//')
|
||||||
|
|
||||||
|
# Quick exit if disabled
|
||||||
|
if [[ "$ENABLED" != "true" ]]; then
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Read hook input
|
||||||
|
input=$(cat)
|
||||||
|
file_path=$(echo "$input" | jq -r '.tool_input.file_path // empty')
|
||||||
|
|
||||||
|
# Apply configured validation
|
||||||
|
if [[ "$STRICT_MODE" == "true" ]]; then
|
||||||
|
# Strict mode: apply all checks
|
||||||
|
if [[ "$file_path" == *".."* ]]; then
|
||||||
|
echo '{"hookSpecificOutput": {"permissionDecision": "deny"}, "systemMessage": "Path traversal blocked (strict mode)"}' >&2
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ "$file_path" == *".env"* ]] || [[ "$file_path" == *"secret"* ]]; then
|
||||||
|
echo '{"hookSpecificOutput": {"permissionDecision": "deny"}, "systemMessage": "Sensitive file blocked (strict mode)"}' >&2
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
# Standard mode: basic checks only
|
||||||
|
if [[ "$file_path" == "/etc/"* ]] || [[ "$file_path" == "/sys/"* ]]; then
|
||||||
|
echo '{"hookSpecificOutput": {"permissionDecision": "deny"}, "systemMessage": "System path blocked"}' >&2
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check file size if configured
|
||||||
|
if [[ -n "$MAX_SIZE" ]] && [[ "$MAX_SIZE" =~ ^[0-9]+$ ]]; then
|
||||||
|
content=$(echo "$input" | jq -r '.tool_input.content // empty')
|
||||||
|
content_size=${#content}
|
||||||
|
|
||||||
|
if [[ $content_size -gt $MAX_SIZE ]]; then
|
||||||
|
echo '{"hookSpecificOutput": {"permissionDecision": "deny"}, "systemMessage": "File exceeds configured max size: '"$MAX_SIZE"' bytes"}' >&2
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
# All checks passed
|
||||||
|
exit 0
|
||||||
549
skills/plugin-settings/references/parsing-techniques.md
Normal file
549
skills/plugin-settings/references/parsing-techniques.md
Normal file
@@ -0,0 +1,549 @@
|
|||||||
|
# Settings File Parsing Techniques
|
||||||
|
|
||||||
|
Complete guide to parsing `.claude/plugin-name.local.md` files in bash scripts.
|
||||||
|
|
||||||
|
## File Structure
|
||||||
|
|
||||||
|
Settings files use markdown with YAML frontmatter:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
field1: value1
|
||||||
|
field2: "value with spaces"
|
||||||
|
numeric_field: 42
|
||||||
|
boolean_field: true
|
||||||
|
list_field: ["item1", "item2", "item3"]
|
||||||
|
---
|
||||||
|
|
||||||
|
# Markdown Content
|
||||||
|
|
||||||
|
This body content can be extracted separately.
|
||||||
|
It's useful for prompts, documentation, or additional context.
|
||||||
|
```
|
||||||
|
|
||||||
|
## Parsing Frontmatter
|
||||||
|
|
||||||
|
### Extract Frontmatter Block
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
FILE=".claude/my-plugin.local.md"
|
||||||
|
|
||||||
|
# Extract everything between --- markers (excluding the markers themselves)
|
||||||
|
FRONTMATTER=$(sed -n '/^---$/,/^---$/{ /^---$/d; p; }' "$FILE")
|
||||||
|
```
|
||||||
|
|
||||||
|
**How it works:**
|
||||||
|
- `sed -n` - Suppress automatic printing
|
||||||
|
- `/^---$/,/^---$/` - Range from first `---` to second `---`
|
||||||
|
- `{ /^---$/d; p; }` - Delete the `---` lines, print everything else
|
||||||
|
|
||||||
|
### Extract Individual Fields
|
||||||
|
|
||||||
|
**String fields:**
|
||||||
|
```bash
|
||||||
|
# Simple value
|
||||||
|
VALUE=$(echo "$FRONTMATTER" | grep '^field_name:' | sed 's/field_name: *//')
|
||||||
|
|
||||||
|
# Quoted value (removes surrounding quotes)
|
||||||
|
VALUE=$(echo "$FRONTMATTER" | grep '^field_name:' | sed 's/field_name: *//' | sed 's/^"\(.*\)"$/\1/')
|
||||||
|
```
|
||||||
|
|
||||||
|
**Boolean fields:**
|
||||||
|
```bash
|
||||||
|
ENABLED=$(echo "$FRONTMATTER" | grep '^enabled:' | sed 's/enabled: *//')
|
||||||
|
|
||||||
|
# Use in condition
|
||||||
|
if [[ "$ENABLED" == "true" ]]; then
|
||||||
|
# Enabled
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
**Numeric fields:**
|
||||||
|
```bash
|
||||||
|
MAX=$(echo "$FRONTMATTER" | grep '^max_value:' | sed 's/max_value: *//')
|
||||||
|
|
||||||
|
# Validate it's a number
|
||||||
|
if [[ "$MAX" =~ ^[0-9]+$ ]]; then
|
||||||
|
# Use in numeric comparison
|
||||||
|
if [[ $MAX -gt 100 ]]; then
|
||||||
|
# Too large
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
**List fields (simple):**
|
||||||
|
```bash
|
||||||
|
# YAML: list: ["item1", "item2", "item3"]
|
||||||
|
LIST=$(echo "$FRONTMATTER" | grep '^list:' | sed 's/list: *//')
|
||||||
|
# Result: ["item1", "item2", "item3"]
|
||||||
|
|
||||||
|
# For simple checks:
|
||||||
|
if [[ "$LIST" == *"item1"* ]]; then
|
||||||
|
# List contains item1
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
**List fields (proper parsing with jq):**
|
||||||
|
```bash
|
||||||
|
# For proper list handling, use yq or convert to JSON
|
||||||
|
# This requires yq to be installed (brew install yq)
|
||||||
|
|
||||||
|
# Extract list as JSON array
|
||||||
|
LIST=$(echo "$FRONTMATTER" | yq -o json '.list' 2>/dev/null)
|
||||||
|
|
||||||
|
# Iterate over items
|
||||||
|
echo "$LIST" | jq -r '.[]' | while read -r item; do
|
||||||
|
echo "Processing: $item"
|
||||||
|
done
|
||||||
|
```
|
||||||
|
|
||||||
|
## Parsing Markdown Body
|
||||||
|
|
||||||
|
### Extract Body Content
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
FILE=".claude/my-plugin.local.md"
|
||||||
|
|
||||||
|
# Extract everything after the closing ---
|
||||||
|
# Counts --- markers: first is opening, second is closing, everything after is body
|
||||||
|
BODY=$(awk '/^---$/{i++; next} i>=2' "$FILE")
|
||||||
|
```
|
||||||
|
|
||||||
|
**How it works:**
|
||||||
|
- `/^---$/` - Match `---` lines
|
||||||
|
- `{i++; next}` - Increment counter and skip the `---` line
|
||||||
|
- `i>=2` - Print all lines after second `---`
|
||||||
|
|
||||||
|
**Handles edge case:** If `---` appears in the markdown body, it still works because we only count the first two `---` at the start.
|
||||||
|
|
||||||
|
### Use Body as Prompt
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Extract body
|
||||||
|
PROMPT=$(awk '/^---$/{i++; next} i>=2' "$RALPH_STATE_FILE")
|
||||||
|
|
||||||
|
# Feed back to Claude
|
||||||
|
echo '{"decision": "block", "reason": "'"$PROMPT"'"}' | jq .
|
||||||
|
```
|
||||||
|
|
||||||
|
**Important:** Use `jq -n --arg` for safer JSON construction with user content:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
PROMPT=$(awk '/^---$/{i++; next} i>=2' "$FILE")
|
||||||
|
|
||||||
|
# Safe JSON construction
|
||||||
|
jq -n --arg prompt "$PROMPT" '{
|
||||||
|
"decision": "block",
|
||||||
|
"reason": $prompt
|
||||||
|
}'
|
||||||
|
```
|
||||||
|
|
||||||
|
## Common Parsing Patterns
|
||||||
|
|
||||||
|
### Pattern: Field with Default
|
||||||
|
|
||||||
|
```bash
|
||||||
|
VALUE=$(echo "$FRONTMATTER" | grep '^field:' | sed 's/field: *//' | sed 's/^"\(.*\)"$/\1/')
|
||||||
|
|
||||||
|
# Use default if empty
|
||||||
|
if [[ -z "$VALUE" ]]; then
|
||||||
|
VALUE="default_value"
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
### Pattern: Optional Field
|
||||||
|
|
||||||
|
```bash
|
||||||
|
OPTIONAL=$(echo "$FRONTMATTER" | grep '^optional_field:' | sed 's/optional_field: *//' | sed 's/^"\(.*\)"$/\1/')
|
||||||
|
|
||||||
|
# Only use if present
|
||||||
|
if [[ -n "$OPTIONAL" ]] && [[ "$OPTIONAL" != "null" ]]; then
|
||||||
|
# Field is set, use it
|
||||||
|
echo "Optional field: $OPTIONAL"
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
### Pattern: Multiple Fields at Once
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Parse all fields in one pass
|
||||||
|
while IFS=': ' read -r key value; do
|
||||||
|
# Remove quotes if present
|
||||||
|
value=$(echo "$value" | sed 's/^"\(.*\)"$/\1/')
|
||||||
|
|
||||||
|
case "$key" in
|
||||||
|
enabled)
|
||||||
|
ENABLED="$value"
|
||||||
|
;;
|
||||||
|
mode)
|
||||||
|
MODE="$value"
|
||||||
|
;;
|
||||||
|
max_size)
|
||||||
|
MAX_SIZE="$value"
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
done <<< "$FRONTMATTER"
|
||||||
|
```
|
||||||
|
|
||||||
|
## Updating Settings Files
|
||||||
|
|
||||||
|
### Atomic Updates
|
||||||
|
|
||||||
|
Always use temp file + atomic move to prevent corruption:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
FILE=".claude/my-plugin.local.md"
|
||||||
|
NEW_VALUE="updated_value"
|
||||||
|
|
||||||
|
# Create temp file
|
||||||
|
TEMP_FILE="${FILE}.tmp.$$"
|
||||||
|
|
||||||
|
# Update field using sed
|
||||||
|
sed "s/^field_name: .*/field_name: $NEW_VALUE/" "$FILE" > "$TEMP_FILE"
|
||||||
|
|
||||||
|
# Atomic replace
|
||||||
|
mv "$TEMP_FILE" "$FILE"
|
||||||
|
```
|
||||||
|
|
||||||
|
### Update Single Field
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Increment iteration counter
|
||||||
|
CURRENT=$(echo "$FRONTMATTER" | grep '^iteration:' | sed 's/iteration: *//')
|
||||||
|
NEXT=$((CURRENT + 1))
|
||||||
|
|
||||||
|
# Update file
|
||||||
|
TEMP_FILE="${FILE}.tmp.$$"
|
||||||
|
sed "s/^iteration: .*/iteration: $NEXT/" "$FILE" > "$TEMP_FILE"
|
||||||
|
mv "$TEMP_FILE" "$FILE"
|
||||||
|
```
|
||||||
|
|
||||||
|
### Update Multiple Fields
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Update several fields at once
|
||||||
|
TEMP_FILE="${FILE}.tmp.$$"
|
||||||
|
|
||||||
|
sed -e "s/^iteration: .*/iteration: $NEXT_ITERATION/" \
|
||||||
|
-e "s/^pr_number: .*/pr_number: $PR_NUMBER/" \
|
||||||
|
-e "s/^status: .*/status: $NEW_STATUS/" \
|
||||||
|
"$FILE" > "$TEMP_FILE"
|
||||||
|
|
||||||
|
mv "$TEMP_FILE" "$FILE"
|
||||||
|
```
|
||||||
|
|
||||||
|
## Validation Techniques
|
||||||
|
|
||||||
|
### Validate File Exists and Is Readable
|
||||||
|
|
||||||
|
```bash
|
||||||
|
FILE=".claude/my-plugin.local.md"
|
||||||
|
|
||||||
|
if [[ ! -f "$FILE" ]]; then
|
||||||
|
echo "Settings file not found" >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ ! -r "$FILE" ]]; then
|
||||||
|
echo "Settings file not readable" >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
### Validate Frontmatter Structure
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Count --- markers (should be exactly 2 at start)
|
||||||
|
MARKER_COUNT=$(grep -c '^---$' "$FILE" 2>/dev/null || echo "0")
|
||||||
|
|
||||||
|
if [[ $MARKER_COUNT -lt 2 ]]; then
|
||||||
|
echo "Invalid settings file: missing frontmatter markers" >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
### Validate Field Values
|
||||||
|
|
||||||
|
```bash
|
||||||
|
MODE=$(echo "$FRONTMATTER" | grep '^mode:' | sed 's/mode: *//')
|
||||||
|
|
||||||
|
case "$MODE" in
|
||||||
|
strict|standard|lenient)
|
||||||
|
# Valid mode
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
echo "Invalid mode: $MODE (must be strict, standard, or lenient)" >&2
|
||||||
|
exit 1
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
```
|
||||||
|
|
||||||
|
### Validate Numeric Ranges
|
||||||
|
|
||||||
|
```bash
|
||||||
|
MAX_SIZE=$(echo "$FRONTMATTER" | grep '^max_size:' | sed 's/max_size: *//')
|
||||||
|
|
||||||
|
if ! [[ "$MAX_SIZE" =~ ^[0-9]+$ ]]; then
|
||||||
|
echo "max_size must be a number" >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ $MAX_SIZE -lt 1 ]] || [[ $MAX_SIZE -gt 10000000 ]]; then
|
||||||
|
echo "max_size out of range (1-10000000)" >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
## Edge Cases and Gotchas
|
||||||
|
|
||||||
|
### Quotes in Values
|
||||||
|
|
||||||
|
YAML allows both quoted and unquoted strings:
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
# These are equivalent:
|
||||||
|
field1: value
|
||||||
|
field2: "value"
|
||||||
|
field3: 'value'
|
||||||
|
```
|
||||||
|
|
||||||
|
**Handle both:**
|
||||||
|
```bash
|
||||||
|
# Remove surrounding quotes if present
|
||||||
|
VALUE=$(echo "$FRONTMATTER" | grep '^field:' | sed 's/field: *//' | sed 's/^"\(.*\)"$/\1/' | sed "s/^'\\(.*\\)'$/\\1/")
|
||||||
|
```
|
||||||
|
|
||||||
|
### --- in Markdown Body
|
||||||
|
|
||||||
|
If the markdown body contains `---`, the parsing still works because we only match the first two:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
field: value
|
||||||
|
---
|
||||||
|
|
||||||
|
# Body
|
||||||
|
|
||||||
|
Here's a separator:
|
||||||
|
---
|
||||||
|
|
||||||
|
More content after the separator.
|
||||||
|
```
|
||||||
|
|
||||||
|
The `awk '/^---$/{i++; next} i>=2'` pattern handles this correctly.
|
||||||
|
|
||||||
|
### Empty Values
|
||||||
|
|
||||||
|
Handle missing or empty fields:
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
field1:
|
||||||
|
field2: ""
|
||||||
|
field3: null
|
||||||
|
```
|
||||||
|
|
||||||
|
**Parsing:**
|
||||||
|
```bash
|
||||||
|
VALUE=$(echo "$FRONTMATTER" | grep '^field1:' | sed 's/field1: *//')
|
||||||
|
# VALUE will be empty string
|
||||||
|
|
||||||
|
# Check for empty/null
|
||||||
|
if [[ -z "$VALUE" ]] || [[ "$VALUE" == "null" ]]; then
|
||||||
|
VALUE="default"
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
### Special Characters
|
||||||
|
|
||||||
|
Values with special characters need careful handling:
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
message: "Error: Something went wrong!"
|
||||||
|
path: "/path/with spaces/file.txt"
|
||||||
|
regex: "^[a-zA-Z0-9_]+$"
|
||||||
|
```
|
||||||
|
|
||||||
|
**Safe parsing:**
|
||||||
|
```bash
|
||||||
|
# Always quote variables when using
|
||||||
|
MESSAGE=$(echo "$FRONTMATTER" | grep '^message:' | sed 's/message: *//' | sed 's/^"\(.*\)"$/\1/')
|
||||||
|
|
||||||
|
echo "Message: $MESSAGE" # Quoted!
|
||||||
|
```
|
||||||
|
|
||||||
|
## Performance Optimization
|
||||||
|
|
||||||
|
### Cache Parsed Values
|
||||||
|
|
||||||
|
If reading settings multiple times:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Parse once
|
||||||
|
FRONTMATTER=$(sed -n '/^---$/,/^---$/{ /^---$/d; p; }' "$FILE")
|
||||||
|
|
||||||
|
# Extract multiple fields from cached frontmatter
|
||||||
|
FIELD1=$(echo "$FRONTMATTER" | grep '^field1:' | sed 's/field1: *//')
|
||||||
|
FIELD2=$(echo "$FRONTMATTER" | grep '^field2:' | sed 's/field2: *//')
|
||||||
|
FIELD3=$(echo "$FRONTMATTER" | grep '^field3:' | sed 's/field3: *//')
|
||||||
|
```
|
||||||
|
|
||||||
|
**Don't:** Re-parse file for each field.
|
||||||
|
|
||||||
|
### Lazy Loading
|
||||||
|
|
||||||
|
Only parse settings when needed:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
input=$(cat)
|
||||||
|
|
||||||
|
# Quick checks first (no file I/O)
|
||||||
|
tool_name=$(echo "$input" | jq -r '.tool_name')
|
||||||
|
if [[ "$tool_name" != "Write" ]]; then
|
||||||
|
exit 0 # Not a write operation, skip
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Only now check settings file
|
||||||
|
if [[ -f ".claude/my-plugin.local.md" ]]; then
|
||||||
|
# Parse settings
|
||||||
|
# ...
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
## Debugging
|
||||||
|
|
||||||
|
### Print Parsed Values
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
set -x # Enable debug tracing
|
||||||
|
|
||||||
|
FILE=".claude/my-plugin.local.md"
|
||||||
|
|
||||||
|
if [[ -f "$FILE" ]]; then
|
||||||
|
echo "Settings file found" >&2
|
||||||
|
|
||||||
|
FRONTMATTER=$(sed -n '/^---$/,/^---$/{ /^---$/d; p; }' "$FILE")
|
||||||
|
echo "Frontmatter:" >&2
|
||||||
|
echo "$FRONTMATTER" >&2
|
||||||
|
|
||||||
|
ENABLED=$(echo "$FRONTMATTER" | grep '^enabled:' | sed 's/enabled: *//')
|
||||||
|
echo "Enabled: $ENABLED" >&2
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
### Validate Parsing
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Show what was parsed
|
||||||
|
echo "Parsed values:" >&2
|
||||||
|
echo " enabled: $ENABLED" >&2
|
||||||
|
echo " mode: $MODE" >&2
|
||||||
|
echo " max_size: $MAX_SIZE" >&2
|
||||||
|
|
||||||
|
# Verify expected values
|
||||||
|
if [[ "$ENABLED" != "true" ]] && [[ "$ENABLED" != "false" ]]; then
|
||||||
|
echo "⚠️ Unexpected enabled value: $ENABLED" >&2
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
## Alternative: Using yq
|
||||||
|
|
||||||
|
For complex YAML, consider using `yq`:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Install: brew install yq
|
||||||
|
|
||||||
|
# Parse YAML properly
|
||||||
|
FRONTMATTER=$(sed -n '/^---$/,/^---$/{ /^---$/d; p; }' "$FILE")
|
||||||
|
|
||||||
|
# Extract fields with yq
|
||||||
|
ENABLED=$(echo "$FRONTMATTER" | yq '.enabled')
|
||||||
|
MODE=$(echo "$FRONTMATTER" | yq '.mode')
|
||||||
|
LIST=$(echo "$FRONTMATTER" | yq -o json '.list_field')
|
||||||
|
|
||||||
|
# Iterate list properly
|
||||||
|
echo "$LIST" | jq -r '.[]' | while read -r item; do
|
||||||
|
echo "Item: $item"
|
||||||
|
done
|
||||||
|
```
|
||||||
|
|
||||||
|
**Pros:**
|
||||||
|
- Proper YAML parsing
|
||||||
|
- Handles complex structures
|
||||||
|
- Better list/object support
|
||||||
|
|
||||||
|
**Cons:**
|
||||||
|
- Requires yq installation
|
||||||
|
- Additional dependency
|
||||||
|
- May not be available on all systems
|
||||||
|
|
||||||
|
**Recommendation:** Use sed/grep for simple fields, yq for complex structures.
|
||||||
|
|
||||||
|
## Complete Example
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
# Configuration
|
||||||
|
SETTINGS_FILE=".claude/my-plugin.local.md"
|
||||||
|
|
||||||
|
# Quick exit if not configured
|
||||||
|
if [[ ! -f "$SETTINGS_FILE" ]]; then
|
||||||
|
# Use defaults
|
||||||
|
ENABLED=true
|
||||||
|
MODE=standard
|
||||||
|
MAX_SIZE=1000000
|
||||||
|
else
|
||||||
|
# Parse frontmatter
|
||||||
|
FRONTMATTER=$(sed -n '/^---$/,/^---$/{ /^---$/d; p; }' "$SETTINGS_FILE")
|
||||||
|
|
||||||
|
# Extract fields with defaults
|
||||||
|
ENABLED=$(echo "$FRONTMATTER" | grep '^enabled:' | sed 's/enabled: *//')
|
||||||
|
ENABLED=${ENABLED:-true}
|
||||||
|
|
||||||
|
MODE=$(echo "$FRONTMATTER" | grep '^mode:' | sed 's/mode: *//' | sed 's/^"\(.*\)"$/\1/')
|
||||||
|
MODE=${MODE:-standard}
|
||||||
|
|
||||||
|
MAX_SIZE=$(echo "$FRONTMATTER" | grep '^max_size:' | sed 's/max_size: *//')
|
||||||
|
MAX_SIZE=${MAX_SIZE:-1000000}
|
||||||
|
|
||||||
|
# Validate values
|
||||||
|
if [[ "$ENABLED" != "true" ]] && [[ "$ENABLED" != "false" ]]; then
|
||||||
|
echo "⚠️ Invalid enabled value, using default" >&2
|
||||||
|
ENABLED=true
|
||||||
|
fi
|
||||||
|
|
||||||
|
if ! [[ "$MAX_SIZE" =~ ^[0-9]+$ ]]; then
|
||||||
|
echo "⚠️ Invalid max_size, using default" >&2
|
||||||
|
MAX_SIZE=1000000
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Quick exit if disabled
|
||||||
|
if [[ "$ENABLED" != "true" ]]; then
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Use configuration
|
||||||
|
echo "Configuration loaded: mode=$MODE, max_size=$MAX_SIZE" >&2
|
||||||
|
|
||||||
|
# Apply logic based on settings
|
||||||
|
case "$MODE" in
|
||||||
|
strict)
|
||||||
|
# Strict validation
|
||||||
|
;;
|
||||||
|
standard)
|
||||||
|
# Standard validation
|
||||||
|
;;
|
||||||
|
lenient)
|
||||||
|
# Lenient validation
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
```
|
||||||
|
|
||||||
|
This provides robust settings handling with defaults, validation, and error recovery.
|
||||||
395
skills/plugin-settings/references/real-world-examples.md
Normal file
395
skills/plugin-settings/references/real-world-examples.md
Normal file
@@ -0,0 +1,395 @@
|
|||||||
|
# Real-World Plugin Settings Examples
|
||||||
|
|
||||||
|
Detailed analysis of how production plugins use the `.claude/plugin-name.local.md` pattern.
|
||||||
|
|
||||||
|
## multi-agent-swarm Plugin
|
||||||
|
|
||||||
|
### Settings File Structure
|
||||||
|
|
||||||
|
**.claude/multi-agent-swarm.local.md:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
agent_name: auth-implementation
|
||||||
|
task_number: 3.5
|
||||||
|
pr_number: 1234
|
||||||
|
coordinator_session: team-leader
|
||||||
|
enabled: true
|
||||||
|
dependencies: ["Task 3.4"]
|
||||||
|
additional_instructions: "Use JWT tokens, not sessions"
|
||||||
|
---
|
||||||
|
|
||||||
|
# Task: Implement Authentication
|
||||||
|
|
||||||
|
Build JWT-based authentication for the REST API.
|
||||||
|
|
||||||
|
## Requirements
|
||||||
|
- JWT token generation and validation
|
||||||
|
- Refresh token flow
|
||||||
|
- Secure password hashing
|
||||||
|
|
||||||
|
## Success Criteria
|
||||||
|
- Auth endpoints implemented
|
||||||
|
- Tests passing (100% coverage)
|
||||||
|
- PR created and CI green
|
||||||
|
- Documentation updated
|
||||||
|
|
||||||
|
## Coordination
|
||||||
|
Depends on Task 3.4 (user model).
|
||||||
|
Report status to 'team-leader' session.
|
||||||
|
```
|
||||||
|
|
||||||
|
### How It's Used
|
||||||
|
|
||||||
|
**File:** `hooks/agent-stop-notification.sh`
|
||||||
|
|
||||||
|
**Purpose:** Send notifications to coordinator when agent becomes idle
|
||||||
|
|
||||||
|
**Implementation:**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
SWARM_STATE_FILE=".claude/multi-agent-swarm.local.md"
|
||||||
|
|
||||||
|
# Quick exit if no swarm active
|
||||||
|
if [[ ! -f "$SWARM_STATE_FILE" ]]; then
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Parse frontmatter
|
||||||
|
FRONTMATTER=$(sed -n '/^---$/,/^---$/{ /^---$/d; p; }' "$SWARM_STATE_FILE")
|
||||||
|
|
||||||
|
# Extract configuration
|
||||||
|
COORDINATOR_SESSION=$(echo "$FRONTMATTER" | grep '^coordinator_session:' | sed 's/coordinator_session: *//' | sed 's/^"\(.*\)"$/\1/')
|
||||||
|
AGENT_NAME=$(echo "$FRONTMATTER" | grep '^agent_name:' | sed 's/agent_name: *//' | sed 's/^"\(.*\)"$/\1/')
|
||||||
|
TASK_NUMBER=$(echo "$FRONTMATTER" | grep '^task_number:' | sed 's/task_number: *//' | sed 's/^"\(.*\)"$/\1/')
|
||||||
|
PR_NUMBER=$(echo "$FRONTMATTER" | grep '^pr_number:' | sed 's/pr_number: *//' | sed 's/^"\(.*\)"$/\1/')
|
||||||
|
ENABLED=$(echo "$FRONTMATTER" | grep '^enabled:' | sed 's/enabled: *//')
|
||||||
|
|
||||||
|
# Check if enabled
|
||||||
|
if [[ "$ENABLED" != "true" ]]; then
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Send notification to coordinator
|
||||||
|
NOTIFICATION="🤖 Agent ${AGENT_NAME} (Task ${TASK_NUMBER}, PR #${PR_NUMBER}) is idle."
|
||||||
|
|
||||||
|
if tmux has-session -t "$COORDINATOR_SESSION" 2>/dev/null; then
|
||||||
|
tmux send-keys -t "$COORDINATOR_SESSION" "$NOTIFICATION" Enter
|
||||||
|
sleep 0.5
|
||||||
|
tmux send-keys -t "$COORDINATOR_SESSION" Enter
|
||||||
|
fi
|
||||||
|
|
||||||
|
exit 0
|
||||||
|
```
|
||||||
|
|
||||||
|
**Key patterns:**
|
||||||
|
1. **Quick exit** (line 7-9): Returns immediately if file doesn't exist
|
||||||
|
2. **Field extraction** (lines 11-17): Parses each frontmatter field
|
||||||
|
3. **Enabled check** (lines 19-21): Respects enabled flag
|
||||||
|
4. **Action based on settings** (lines 23-29): Uses coordinator_session to send notification
|
||||||
|
|
||||||
|
### Creation
|
||||||
|
|
||||||
|
**File:** `commands/launch-swarm.md`
|
||||||
|
|
||||||
|
Settings files are created during swarm launch with:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cat > "$WORKTREE_PATH/.claude/multi-agent-swarm.local.md" <<EOF
|
||||||
|
---
|
||||||
|
agent_name: $AGENT_NAME
|
||||||
|
task_number: $TASK_ID
|
||||||
|
pr_number: TBD
|
||||||
|
coordinator_session: $COORDINATOR_SESSION
|
||||||
|
enabled: true
|
||||||
|
dependencies: [$DEPENDENCIES]
|
||||||
|
additional_instructions: "$EXTRA_INSTRUCTIONS"
|
||||||
|
---
|
||||||
|
|
||||||
|
# Task: $TASK_DESCRIPTION
|
||||||
|
|
||||||
|
$TASK_DETAILS
|
||||||
|
EOF
|
||||||
|
```
|
||||||
|
|
||||||
|
### Updates
|
||||||
|
|
||||||
|
PR number updated after PR creation:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Update pr_number field
|
||||||
|
sed "s/^pr_number: .*/pr_number: $PR_NUM/" \
|
||||||
|
".claude/multi-agent-swarm.local.md" > temp.md
|
||||||
|
mv temp.md ".claude/multi-agent-swarm.local.md"
|
||||||
|
```
|
||||||
|
|
||||||
|
## ralph-wiggum Plugin
|
||||||
|
|
||||||
|
### Settings File Structure
|
||||||
|
|
||||||
|
**.claude/ralph-loop.local.md:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
iteration: 1
|
||||||
|
max_iterations: 10
|
||||||
|
completion_promise: "All tests passing and build successful"
|
||||||
|
started_at: "2025-01-15T14:30:00Z"
|
||||||
|
---
|
||||||
|
|
||||||
|
Fix all the linting errors in the project.
|
||||||
|
Make sure tests pass after each fix.
|
||||||
|
Document any changes needed in CLAUDE.md.
|
||||||
|
```
|
||||||
|
|
||||||
|
### How It's Used
|
||||||
|
|
||||||
|
**File:** `hooks/stop-hook.sh`
|
||||||
|
|
||||||
|
**Purpose:** Prevent session exit and loop Claude's output back as input
|
||||||
|
|
||||||
|
**Implementation:**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
RALPH_STATE_FILE=".claude/ralph-loop.local.md"
|
||||||
|
|
||||||
|
# Quick exit if no active loop
|
||||||
|
if [[ ! -f "$RALPH_STATE_FILE" ]]; then
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Parse frontmatter
|
||||||
|
FRONTMATTER=$(sed -n '/^---$/,/^---$/{ /^---$/d; p; }' "$RALPH_STATE_FILE")
|
||||||
|
|
||||||
|
# Extract configuration
|
||||||
|
ITERATION=$(echo "$FRONTMATTER" | grep '^iteration:' | sed 's/iteration: *//')
|
||||||
|
MAX_ITERATIONS=$(echo "$FRONTMATTER" | grep '^max_iterations:' | sed 's/max_iterations: *//')
|
||||||
|
COMPLETION_PROMISE=$(echo "$FRONTMATTER" | grep '^completion_promise:' | sed 's/completion_promise: *//' | sed 's/^"\(.*\)"$/\1/')
|
||||||
|
|
||||||
|
# Check max iterations
|
||||||
|
if [[ $MAX_ITERATIONS -gt 0 ]] && [[ $ITERATION -ge $MAX_ITERATIONS ]]; then
|
||||||
|
echo "🛑 Ralph loop: Max iterations ($MAX_ITERATIONS) reached."
|
||||||
|
rm "$RALPH_STATE_FILE"
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Get transcript and check for completion promise
|
||||||
|
TRANSCRIPT_PATH=$(echo "$HOOK_INPUT" | jq -r '.transcript_path')
|
||||||
|
LAST_OUTPUT=$(grep '"role":"assistant"' "$TRANSCRIPT_PATH" | tail -1 | jq -r '.message.content | map(select(.type == "text")) | map(.text) | join("\n")')
|
||||||
|
|
||||||
|
# Check for completion
|
||||||
|
if [[ "$COMPLETION_PROMISE" != "null" ]] && [[ -n "$COMPLETION_PROMISE" ]]; then
|
||||||
|
PROMISE_TEXT=$(echo "$LAST_OUTPUT" | perl -0777 -pe 's/.*?<promise>(.*?)<\/promise>.*/$1/s; s/^\s+|\s+$//g')
|
||||||
|
|
||||||
|
if [[ "$PROMISE_TEXT" = "$COMPLETION_PROMISE" ]]; then
|
||||||
|
echo "✅ Ralph loop: Detected completion"
|
||||||
|
rm "$RALPH_STATE_FILE"
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Continue loop - increment iteration
|
||||||
|
NEXT_ITERATION=$((ITERATION + 1))
|
||||||
|
|
||||||
|
# Extract prompt from markdown body
|
||||||
|
PROMPT_TEXT=$(awk '/^---$/{i++; next} i>=2' "$RALPH_STATE_FILE")
|
||||||
|
|
||||||
|
# Update iteration counter
|
||||||
|
TEMP_FILE="${RALPH_STATE_FILE}.tmp.$$"
|
||||||
|
sed "s/^iteration: .*/iteration: $NEXT_ITERATION/" "$RALPH_STATE_FILE" > "$TEMP_FILE"
|
||||||
|
mv "$TEMP_FILE" "$RALPH_STATE_FILE"
|
||||||
|
|
||||||
|
# Block exit and feed prompt back
|
||||||
|
jq -n \
|
||||||
|
--arg prompt "$PROMPT_TEXT" \
|
||||||
|
--arg msg "🔄 Ralph iteration $NEXT_ITERATION" \
|
||||||
|
'{
|
||||||
|
"decision": "block",
|
||||||
|
"reason": $prompt,
|
||||||
|
"systemMessage": $msg
|
||||||
|
}'
|
||||||
|
|
||||||
|
exit 0
|
||||||
|
```
|
||||||
|
|
||||||
|
**Key patterns:**
|
||||||
|
1. **Quick exit** (line 7-9): Skip if not active
|
||||||
|
2. **Iteration tracking** (lines 11-20): Count and enforce max iterations
|
||||||
|
3. **Promise detection** (lines 25-33): Check for completion signal in output
|
||||||
|
4. **Prompt extraction** (line 38): Read markdown body as next prompt
|
||||||
|
5. **State update** (lines 40-43): Increment iteration atomically
|
||||||
|
6. **Loop continuation** (lines 45-53): Block exit and feed prompt back
|
||||||
|
|
||||||
|
### Creation
|
||||||
|
|
||||||
|
**File:** `scripts/setup-ralph-loop.sh`
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
PROMPT="$1"
|
||||||
|
MAX_ITERATIONS="${2:-0}"
|
||||||
|
COMPLETION_PROMISE="${3:-}"
|
||||||
|
|
||||||
|
# Create state file
|
||||||
|
cat > ".claude/ralph-loop.local.md" <<EOF
|
||||||
|
---
|
||||||
|
iteration: 1
|
||||||
|
max_iterations: $MAX_ITERATIONS
|
||||||
|
completion_promise: "$COMPLETION_PROMISE"
|
||||||
|
started_at: "$(date -Iseconds)"
|
||||||
|
---
|
||||||
|
|
||||||
|
$PROMPT
|
||||||
|
EOF
|
||||||
|
|
||||||
|
echo "Ralph loop initialized: .claude/ralph-loop.local.md"
|
||||||
|
```
|
||||||
|
|
||||||
|
## Pattern Comparison
|
||||||
|
|
||||||
|
| Feature | multi-agent-swarm | ralph-wiggum |
|
||||||
|
|---------|-------------------|--------------|
|
||||||
|
| **File** | `.claude/multi-agent-swarm.local.md` | `.claude/ralph-loop.local.md` |
|
||||||
|
| **Purpose** | Agent coordination state | Loop iteration state |
|
||||||
|
| **Frontmatter** | Agent metadata | Loop configuration |
|
||||||
|
| **Body** | Task assignment | Prompt to loop |
|
||||||
|
| **Updates** | PR number, status | Iteration counter |
|
||||||
|
| **Deletion** | Manual or on completion | On loop exit |
|
||||||
|
| **Hook** | Stop (notifications) | Stop (loop control) |
|
||||||
|
|
||||||
|
## Best Practices from Real Plugins
|
||||||
|
|
||||||
|
### 1. Quick Exit Pattern
|
||||||
|
|
||||||
|
Both plugins check file existence first:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
if [[ ! -f "$STATE_FILE" ]]; then
|
||||||
|
exit 0 # Not active
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
**Why:** Avoids errors when plugin isn't configured and performs fast.
|
||||||
|
|
||||||
|
### 2. Enabled Flag
|
||||||
|
|
||||||
|
Both use an `enabled` field for explicit control:
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
enabled: true
|
||||||
|
```
|
||||||
|
|
||||||
|
**Why:** Allows temporary deactivation without deleting file.
|
||||||
|
|
||||||
|
### 3. Atomic Updates
|
||||||
|
|
||||||
|
Both use temp file + atomic move:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
TEMP_FILE="${FILE}.tmp.$$"
|
||||||
|
sed "s/^field: .*/field: $NEW_VALUE/" "$FILE" > "$TEMP_FILE"
|
||||||
|
mv "$TEMP_FILE" "$FILE"
|
||||||
|
```
|
||||||
|
|
||||||
|
**Why:** Prevents corruption if process is interrupted.
|
||||||
|
|
||||||
|
### 4. Quote Handling
|
||||||
|
|
||||||
|
Both strip surrounding quotes from YAML values:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
sed 's/^"\(.*\)"$/\1/'
|
||||||
|
```
|
||||||
|
|
||||||
|
**Why:** YAML allows both `field: value` and `field: "value"`.
|
||||||
|
|
||||||
|
### 5. Error Handling
|
||||||
|
|
||||||
|
Both handle missing/corrupt files gracefully:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
if [[ ! -f "$FILE" ]]; then
|
||||||
|
exit 0 # No error, just not configured
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ -z "$CRITICAL_FIELD" ]]; then
|
||||||
|
echo "Settings file corrupt" >&2
|
||||||
|
rm "$FILE" # Clean up
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
**Why:** Fails gracefully instead of crashing.
|
||||||
|
|
||||||
|
## Anti-Patterns to Avoid
|
||||||
|
|
||||||
|
### ❌ Hardcoded Paths
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# BAD
|
||||||
|
FILE="/Users/alice/.claude/my-plugin.local.md"
|
||||||
|
|
||||||
|
# GOOD
|
||||||
|
FILE=".claude/my-plugin.local.md"
|
||||||
|
```
|
||||||
|
|
||||||
|
### ❌ Unquoted Variables
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# BAD
|
||||||
|
echo $VALUE
|
||||||
|
|
||||||
|
# GOOD
|
||||||
|
echo "$VALUE"
|
||||||
|
```
|
||||||
|
|
||||||
|
### ❌ Non-Atomic Updates
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# BAD: Can corrupt file if interrupted
|
||||||
|
sed -i "s/field: .*/field: $VALUE/" "$FILE"
|
||||||
|
|
||||||
|
# GOOD: Atomic
|
||||||
|
TEMP_FILE="${FILE}.tmp.$$"
|
||||||
|
sed "s/field: .*/field: $VALUE/" "$FILE" > "$TEMP_FILE"
|
||||||
|
mv "$TEMP_FILE" "$FILE"
|
||||||
|
```
|
||||||
|
|
||||||
|
### ❌ No Default Values
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# BAD: Fails if field missing
|
||||||
|
if [[ $MAX -gt 100 ]]; then
|
||||||
|
# MAX might be empty!
|
||||||
|
fi
|
||||||
|
|
||||||
|
# GOOD: Provide default
|
||||||
|
MAX=${MAX:-10}
|
||||||
|
```
|
||||||
|
|
||||||
|
### ❌ Ignoring Edge Cases
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# BAD: Assumes exactly 2 --- markers
|
||||||
|
sed -n '/^---$/,/^---$/{ /^---$/d; p; }'
|
||||||
|
|
||||||
|
# GOOD: Handles --- in body
|
||||||
|
awk '/^---$/{i++; next} i>=2' # For body
|
||||||
|
```
|
||||||
|
|
||||||
|
## Conclusion
|
||||||
|
|
||||||
|
The `.claude/plugin-name.local.md` pattern provides:
|
||||||
|
- Simple, human-readable configuration
|
||||||
|
- Version-control friendly (gitignored)
|
||||||
|
- Per-project settings
|
||||||
|
- Easy parsing with standard bash tools
|
||||||
|
- Supports both structured config (YAML) and freeform content (markdown)
|
||||||
|
|
||||||
|
Use this pattern for any plugin that needs user-configurable behavior or state persistence.
|
||||||
59
skills/plugin-settings/scripts/parse-frontmatter.sh
Executable file
59
skills/plugin-settings/scripts/parse-frontmatter.sh
Executable file
@@ -0,0 +1,59 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
# Frontmatter Parser Utility
|
||||||
|
# Extracts YAML frontmatter from .local.md files
|
||||||
|
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
# Usage
|
||||||
|
show_usage() {
|
||||||
|
echo "Usage: $0 <settings-file.md> [field-name]"
|
||||||
|
echo ""
|
||||||
|
echo "Examples:"
|
||||||
|
echo " # Show all frontmatter"
|
||||||
|
echo " $0 .claude/my-plugin.local.md"
|
||||||
|
echo ""
|
||||||
|
echo " # Extract specific field"
|
||||||
|
echo " $0 .claude/my-plugin.local.md enabled"
|
||||||
|
echo ""
|
||||||
|
echo " # Extract and use in script"
|
||||||
|
echo " ENABLED=\$($0 .claude/my-plugin.local.md enabled)"
|
||||||
|
exit 0
|
||||||
|
}
|
||||||
|
|
||||||
|
if [ $# -eq 0 ] || [ "$1" = "-h" ] || [ "$1" = "--help" ]; then
|
||||||
|
show_usage
|
||||||
|
fi
|
||||||
|
|
||||||
|
FILE="$1"
|
||||||
|
FIELD="${2:-}"
|
||||||
|
|
||||||
|
# Validate file
|
||||||
|
if [ ! -f "$FILE" ]; then
|
||||||
|
echo "Error: File not found: $FILE" >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Extract frontmatter
|
||||||
|
FRONTMATTER=$(sed -n '/^---$/,/^---$/{ /^---$/d; p; }' "$FILE")
|
||||||
|
|
||||||
|
if [ -z "$FRONTMATTER" ]; then
|
||||||
|
echo "Error: No frontmatter found in $FILE" >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# If no field specified, output all frontmatter
|
||||||
|
if [ -z "$FIELD" ]; then
|
||||||
|
echo "$FRONTMATTER"
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Extract specific field
|
||||||
|
VALUE=$(echo "$FRONTMATTER" | grep "^${FIELD}:" | sed "s/${FIELD}: *//" | sed 's/^"\(.*\)"$/\1/' | sed "s/^'\\(.*\\)'$/\\1/")
|
||||||
|
|
||||||
|
if [ -z "$VALUE" ]; then
|
||||||
|
echo "Error: Field '$FIELD' not found in frontmatter" >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "$VALUE"
|
||||||
|
exit 0
|
||||||
101
skills/plugin-settings/scripts/validate-settings.sh
Executable file
101
skills/plugin-settings/scripts/validate-settings.sh
Executable file
@@ -0,0 +1,101 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
# Settings File Validator
|
||||||
|
# Validates .claude/plugin-name.local.md structure
|
||||||
|
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
# Usage
|
||||||
|
if [ $# -eq 0 ]; then
|
||||||
|
echo "Usage: $0 <path/to/settings.local.md>"
|
||||||
|
echo ""
|
||||||
|
echo "Validates plugin settings file for:"
|
||||||
|
echo " - File existence and readability"
|
||||||
|
echo " - YAML frontmatter structure"
|
||||||
|
echo " - Required --- markers"
|
||||||
|
echo " - Field format"
|
||||||
|
echo ""
|
||||||
|
echo "Example: $0 .claude/my-plugin.local.md"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
SETTINGS_FILE="$1"
|
||||||
|
|
||||||
|
echo "🔍 Validating settings file: $SETTINGS_FILE"
|
||||||
|
echo ""
|
||||||
|
|
||||||
|
# Check 1: File exists
|
||||||
|
if [ ! -f "$SETTINGS_FILE" ]; then
|
||||||
|
echo "❌ File not found: $SETTINGS_FILE"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
echo "✅ File exists"
|
||||||
|
|
||||||
|
# Check 2: File is readable
|
||||||
|
if [ ! -r "$SETTINGS_FILE" ]; then
|
||||||
|
echo "❌ File is not readable"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
echo "✅ File is readable"
|
||||||
|
|
||||||
|
# Check 3: Has frontmatter markers
|
||||||
|
MARKER_COUNT=$(grep -c '^---$' "$SETTINGS_FILE" 2>/dev/null || echo "0")
|
||||||
|
|
||||||
|
if [ "$MARKER_COUNT" -lt 2 ]; then
|
||||||
|
echo "❌ Invalid frontmatter: found $MARKER_COUNT '---' markers (need at least 2)"
|
||||||
|
echo " Expected format:"
|
||||||
|
echo " ---"
|
||||||
|
echo " field: value"
|
||||||
|
echo " ---"
|
||||||
|
echo " Content..."
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
echo "✅ Frontmatter markers present"
|
||||||
|
|
||||||
|
# Check 4: Extract and validate frontmatter
|
||||||
|
FRONTMATTER=$(sed -n '/^---$/,/^---$/{ /^---$/d; p; }' "$SETTINGS_FILE")
|
||||||
|
|
||||||
|
if [ -z "$FRONTMATTER" ]; then
|
||||||
|
echo "❌ Empty frontmatter (nothing between --- markers)"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
echo "✅ Frontmatter not empty"
|
||||||
|
|
||||||
|
# Check 5: Frontmatter has valid YAML-like structure
|
||||||
|
if ! echo "$FRONTMATTER" | grep -q ':'; then
|
||||||
|
echo "⚠️ Warning: Frontmatter has no key:value pairs"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check 6: Look for common fields
|
||||||
|
echo ""
|
||||||
|
echo "Detected fields:"
|
||||||
|
echo "$FRONTMATTER" | grep '^[a-z_][a-z0-9_]*:' | while IFS=':' read -r key value; do
|
||||||
|
echo " - $key: ${value:0:50}"
|
||||||
|
done
|
||||||
|
|
||||||
|
# Check 7: Validate common boolean fields
|
||||||
|
for field in enabled strict_mode; do
|
||||||
|
VALUE=$(echo "$FRONTMATTER" | grep "^${field}:" | sed "s/${field}: *//" || true)
|
||||||
|
if [ -n "$VALUE" ]; then
|
||||||
|
if [ "$VALUE" != "true" ] && [ "$VALUE" != "false" ]; then
|
||||||
|
echo "⚠️ Field '$field' should be boolean (true/false), got: $VALUE"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
|
||||||
|
# Check 8: Check body exists
|
||||||
|
BODY=$(awk '/^---$/{i++; next} i>=2' "$SETTINGS_FILE")
|
||||||
|
|
||||||
|
echo ""
|
||||||
|
if [ -n "$BODY" ]; then
|
||||||
|
BODY_LINES=$(echo "$BODY" | wc -l | tr -d ' ')
|
||||||
|
echo "✅ Markdown body present ($BODY_LINES lines)"
|
||||||
|
else
|
||||||
|
echo "⚠️ No markdown body (frontmatter only)"
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo ""
|
||||||
|
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
|
||||||
|
echo "✅ Settings file structure is valid"
|
||||||
|
echo ""
|
||||||
|
echo "Reminder: Changes to this file require restarting Claude Code"
|
||||||
|
exit 0
|
||||||
109
skills/plugin-structure/README.md
Normal file
109
skills/plugin-structure/README.md
Normal file
@@ -0,0 +1,109 @@
|
|||||||
|
# Plugin Structure Skill
|
||||||
|
|
||||||
|
Comprehensive guidance on Claude Code plugin architecture, directory layout, and best practices.
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
|
||||||
|
This skill provides detailed knowledge about:
|
||||||
|
- Plugin directory structure and organization
|
||||||
|
- `plugin.json` manifest configuration
|
||||||
|
- Component organization (commands, agents, skills, hooks)
|
||||||
|
- Auto-discovery mechanisms
|
||||||
|
- Portable path references with `${CLAUDE_PLUGIN_ROOT}`
|
||||||
|
- File naming conventions
|
||||||
|
|
||||||
|
## Skill Structure
|
||||||
|
|
||||||
|
### SKILL.md (1,619 words)
|
||||||
|
|
||||||
|
Core skill content covering:
|
||||||
|
- Directory structure overview
|
||||||
|
- Plugin manifest (plugin.json) fields
|
||||||
|
- Component organization patterns
|
||||||
|
- ${CLAUDE_PLUGIN_ROOT} usage
|
||||||
|
- File naming conventions
|
||||||
|
- Auto-discovery mechanism
|
||||||
|
- Best practices
|
||||||
|
- Common patterns
|
||||||
|
- Troubleshooting
|
||||||
|
|
||||||
|
### References
|
||||||
|
|
||||||
|
Detailed documentation for deep dives:
|
||||||
|
|
||||||
|
- **manifest-reference.md**: Complete `plugin.json` field reference
|
||||||
|
- All field descriptions and examples
|
||||||
|
- Path resolution rules
|
||||||
|
- Validation guidelines
|
||||||
|
- Minimal vs. complete manifest examples
|
||||||
|
|
||||||
|
- **component-patterns.md**: Advanced organization patterns
|
||||||
|
- Component lifecycle (discovery, activation)
|
||||||
|
- Command organization patterns
|
||||||
|
- Agent organization patterns
|
||||||
|
- Skill organization patterns
|
||||||
|
- Hook organization patterns
|
||||||
|
- Script organization patterns
|
||||||
|
- Cross-component patterns
|
||||||
|
- Best practices for scalability
|
||||||
|
|
||||||
|
### Examples
|
||||||
|
|
||||||
|
Three complete plugin examples:
|
||||||
|
|
||||||
|
- **minimal-plugin.md**: Simplest possible plugin
|
||||||
|
- Single command
|
||||||
|
- Minimal manifest
|
||||||
|
- When to use this pattern
|
||||||
|
|
||||||
|
- **standard-plugin.md**: Well-structured production plugin
|
||||||
|
- Multiple components (commands, agents, skills, hooks)
|
||||||
|
- Complete manifest with metadata
|
||||||
|
- Rich skill structure
|
||||||
|
- Integration between components
|
||||||
|
|
||||||
|
- **advanced-plugin.md**: Enterprise-grade plugin
|
||||||
|
- Multi-level organization
|
||||||
|
- MCP server integration
|
||||||
|
- Shared libraries
|
||||||
|
- Configuration management
|
||||||
|
- Security automation
|
||||||
|
- Monitoring integration
|
||||||
|
|
||||||
|
## When This Skill Triggers
|
||||||
|
|
||||||
|
Claude Code activates this skill when users:
|
||||||
|
- Ask to "create a plugin" or "scaffold a plugin"
|
||||||
|
- Need to "understand plugin structure"
|
||||||
|
- Want to "organize plugin components"
|
||||||
|
- Need to "set up plugin.json"
|
||||||
|
- Ask about "${CLAUDE_PLUGIN_ROOT}" usage
|
||||||
|
- Want to "add commands/agents/skills/hooks"
|
||||||
|
- Need "configure auto-discovery" help
|
||||||
|
- Ask about plugin architecture or best practices
|
||||||
|
|
||||||
|
## Progressive Disclosure
|
||||||
|
|
||||||
|
The skill uses progressive disclosure to manage context:
|
||||||
|
|
||||||
|
1. **SKILL.md** (~1600 words): Core concepts and workflows
|
||||||
|
2. **References** (~6000 words): Detailed field references and patterns
|
||||||
|
3. **Examples** (~8000 words): Complete working examples
|
||||||
|
|
||||||
|
Claude loads references and examples only as needed based on the task.
|
||||||
|
|
||||||
|
## Related Skills
|
||||||
|
|
||||||
|
This skill works well with:
|
||||||
|
- **hook-development**: For creating plugin hooks
|
||||||
|
- **mcp-integration**: For integrating MCP servers (when available)
|
||||||
|
- **marketplace-publishing**: For publishing plugins (when available)
|
||||||
|
|
||||||
|
## Maintenance
|
||||||
|
|
||||||
|
To update this skill:
|
||||||
|
1. Keep SKILL.md lean and focused on core concepts
|
||||||
|
2. Move detailed information to references/
|
||||||
|
3. Add new examples/ for common patterns
|
||||||
|
4. Update version in SKILL.md frontmatter
|
||||||
|
5. Ensure all documentation uses imperative/infinitive form
|
||||||
475
skills/plugin-structure/SKILL.md
Normal file
475
skills/plugin-structure/SKILL.md
Normal file
@@ -0,0 +1,475 @@
|
|||||||
|
---
|
||||||
|
name: plugin-structure
|
||||||
|
description: This skill should be used when the user asks to "create a plugin", "scaffold a plugin", "understand plugin structure", "organize plugin components", "set up plugin.json", "use ${CLAUDE_PLUGIN_ROOT}", "add commands/agents/skills/hooks", "configure auto-discovery", or needs guidance on plugin directory layout, manifest configuration, component organization, file naming conventions, or Claude Code plugin architecture best practices.
|
||||||
|
---
|
||||||
|
|
||||||
|
# Plugin Structure for Claude Code
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
|
||||||
|
Claude Code plugins follow a standardized directory structure with automatic component discovery. Understanding this structure enables creating well-organized, maintainable plugins that integrate seamlessly with Claude Code.
|
||||||
|
|
||||||
|
**Key concepts:**
|
||||||
|
- Conventional directory layout for automatic discovery
|
||||||
|
- Manifest-driven configuration in `.claude-plugin/plugin.json`
|
||||||
|
- Component-based organization (commands, agents, skills, hooks)
|
||||||
|
- Portable path references using `${CLAUDE_PLUGIN_ROOT}`
|
||||||
|
- Explicit vs. auto-discovered component loading
|
||||||
|
|
||||||
|
## Directory Structure
|
||||||
|
|
||||||
|
Every Claude Code plugin follows this organizational pattern:
|
||||||
|
|
||||||
|
```
|
||||||
|
plugin-name/
|
||||||
|
├── .claude-plugin/
|
||||||
|
│ └── plugin.json # Required: Plugin manifest
|
||||||
|
├── commands/ # Slash commands (.md files)
|
||||||
|
├── agents/ # Subagent definitions (.md files)
|
||||||
|
├── skills/ # Agent skills (subdirectories)
|
||||||
|
│ └── skill-name/
|
||||||
|
│ └── SKILL.md # Required for each skill
|
||||||
|
├── hooks/
|
||||||
|
│ └── hooks.json # Event handler configuration
|
||||||
|
├── .mcp.json # MCP server definitions
|
||||||
|
└── scripts/ # Helper scripts and utilities
|
||||||
|
```
|
||||||
|
|
||||||
|
**Critical rules:**
|
||||||
|
|
||||||
|
1. **Manifest location**: The `plugin.json` manifest MUST be in `.claude-plugin/` directory
|
||||||
|
2. **Component locations**: All component directories (commands, agents, skills, hooks) MUST be at plugin root level, NOT nested inside `.claude-plugin/`
|
||||||
|
3. **Optional components**: Only create directories for components the plugin actually uses
|
||||||
|
4. **Naming convention**: Use kebab-case for all directory and file names
|
||||||
|
|
||||||
|
## Plugin Manifest (plugin.json)
|
||||||
|
|
||||||
|
The manifest defines plugin metadata and configuration. Located at `.claude-plugin/plugin.json`:
|
||||||
|
|
||||||
|
### Required Fields
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"name": "plugin-name"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Name requirements:**
|
||||||
|
- Use kebab-case format (lowercase with hyphens)
|
||||||
|
- Must be unique across installed plugins
|
||||||
|
- No spaces or special characters
|
||||||
|
- Example: `code-review-assistant`, `test-runner`, `api-docs`
|
||||||
|
|
||||||
|
### Recommended Metadata
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"name": "plugin-name",
|
||||||
|
"version": "1.0.0",
|
||||||
|
"description": "Brief explanation of plugin purpose",
|
||||||
|
"author": {
|
||||||
|
"name": "Author Name",
|
||||||
|
"email": "author@example.com",
|
||||||
|
"url": "https://example.com"
|
||||||
|
},
|
||||||
|
"homepage": "https://docs.example.com",
|
||||||
|
"repository": "https://github.com/user/plugin-name",
|
||||||
|
"license": "MIT",
|
||||||
|
"keywords": ["testing", "automation", "ci-cd"]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Version format**: Follow semantic versioning (MAJOR.MINOR.PATCH)
|
||||||
|
**Keywords**: Use for plugin discovery and categorization
|
||||||
|
|
||||||
|
### Component Path Configuration
|
||||||
|
|
||||||
|
Specify custom paths for components (supplements default directories):
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"name": "plugin-name",
|
||||||
|
"commands": "./custom-commands",
|
||||||
|
"agents": ["./agents", "./specialized-agents"],
|
||||||
|
"hooks": "./config/hooks.json",
|
||||||
|
"mcpServers": "./.mcp.json"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Important**: Custom paths supplement defaults—they don't replace them. Components in both default directories and custom paths will load.
|
||||||
|
|
||||||
|
**Path rules:**
|
||||||
|
- Must be relative to plugin root
|
||||||
|
- Must start with `./`
|
||||||
|
- Cannot use absolute paths
|
||||||
|
- Support arrays for multiple locations
|
||||||
|
|
||||||
|
## Component Organization
|
||||||
|
|
||||||
|
### Commands
|
||||||
|
|
||||||
|
**Location**: `commands/` directory
|
||||||
|
**Format**: Markdown files with YAML frontmatter
|
||||||
|
**Auto-discovery**: All `.md` files in `commands/` load automatically
|
||||||
|
|
||||||
|
**Example structure**:
|
||||||
|
```
|
||||||
|
commands/
|
||||||
|
├── review.md # /review command
|
||||||
|
├── test.md # /test command
|
||||||
|
└── deploy.md # /deploy command
|
||||||
|
```
|
||||||
|
|
||||||
|
**File format**:
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
name: command-name
|
||||||
|
description: Command description
|
||||||
|
---
|
||||||
|
|
||||||
|
Command implementation instructions...
|
||||||
|
```
|
||||||
|
|
||||||
|
**Usage**: Commands integrate as native slash commands in Claude Code
|
||||||
|
|
||||||
|
### Agents
|
||||||
|
|
||||||
|
**Location**: `agents/` directory
|
||||||
|
**Format**: Markdown files with YAML frontmatter
|
||||||
|
**Auto-discovery**: All `.md` files in `agents/` load automatically
|
||||||
|
|
||||||
|
**Example structure**:
|
||||||
|
```
|
||||||
|
agents/
|
||||||
|
├── code-reviewer.md
|
||||||
|
├── test-generator.md
|
||||||
|
└── refactorer.md
|
||||||
|
```
|
||||||
|
|
||||||
|
**File format**:
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Agent role and expertise
|
||||||
|
capabilities:
|
||||||
|
- Specific task 1
|
||||||
|
- Specific task 2
|
||||||
|
---
|
||||||
|
|
||||||
|
Detailed agent instructions and knowledge...
|
||||||
|
```
|
||||||
|
|
||||||
|
**Usage**: Users can invoke agents manually, or Claude Code selects them automatically based on task context
|
||||||
|
|
||||||
|
### Skills
|
||||||
|
|
||||||
|
**Location**: `skills/` directory with subdirectories per skill
|
||||||
|
**Format**: Each skill in its own directory with `SKILL.md` file
|
||||||
|
**Auto-discovery**: All `SKILL.md` files in skill subdirectories load automatically
|
||||||
|
|
||||||
|
**Example structure**:
|
||||||
|
```
|
||||||
|
skills/
|
||||||
|
├── api-testing/
|
||||||
|
│ ├── SKILL.md
|
||||||
|
│ ├── scripts/
|
||||||
|
│ │ └── test-runner.py
|
||||||
|
│ └── references/
|
||||||
|
│ └── api-spec.md
|
||||||
|
└── database-migrations/
|
||||||
|
├── SKILL.md
|
||||||
|
└── examples/
|
||||||
|
└── migration-template.sql
|
||||||
|
```
|
||||||
|
|
||||||
|
**SKILL.md format**:
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
name: Skill Name
|
||||||
|
description: When to use this skill
|
||||||
|
version: 1.0.0
|
||||||
|
---
|
||||||
|
|
||||||
|
Skill instructions and guidance...
|
||||||
|
```
|
||||||
|
|
||||||
|
**Supporting files**: Skills can include scripts, references, examples, or assets in subdirectories
|
||||||
|
|
||||||
|
**Usage**: Claude Code autonomously activates skills based on task context matching the description
|
||||||
|
|
||||||
|
### Hooks
|
||||||
|
|
||||||
|
**Location**: `hooks/hooks.json` or inline in `plugin.json`
|
||||||
|
**Format**: JSON configuration defining event handlers
|
||||||
|
**Registration**: Hooks register automatically when plugin enables
|
||||||
|
|
||||||
|
**Example structure**:
|
||||||
|
```
|
||||||
|
hooks/
|
||||||
|
├── hooks.json # Hook configuration
|
||||||
|
└── scripts/
|
||||||
|
├── validate.sh # Hook script
|
||||||
|
└── check-style.sh # Hook script
|
||||||
|
```
|
||||||
|
|
||||||
|
**Configuration format**:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"PreToolUse": [{
|
||||||
|
"matcher": "Write|Edit",
|
||||||
|
"hooks": [{
|
||||||
|
"type": "command",
|
||||||
|
"command": "bash ${CLAUDE_PLUGIN_ROOT}/hooks/scripts/validate.sh",
|
||||||
|
"timeout": 30
|
||||||
|
}]
|
||||||
|
}]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Available events**: PreToolUse, PostToolUse, Stop, SubagentStop, SessionStart, SessionEnd, UserPromptSubmit, PreCompact, Notification
|
||||||
|
|
||||||
|
**Usage**: Hooks execute automatically in response to Claude Code events
|
||||||
|
|
||||||
|
### MCP Servers
|
||||||
|
|
||||||
|
**Location**: `.mcp.json` at plugin root or inline in `plugin.json`
|
||||||
|
**Format**: JSON configuration for MCP server definitions
|
||||||
|
**Auto-start**: Servers start automatically when plugin enables
|
||||||
|
|
||||||
|
**Example format**:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"mcpServers": {
|
||||||
|
"server-name": {
|
||||||
|
"command": "node",
|
||||||
|
"args": ["${CLAUDE_PLUGIN_ROOT}/servers/server.js"],
|
||||||
|
"env": {
|
||||||
|
"API_KEY": "${API_KEY}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Usage**: MCP servers integrate seamlessly with Claude Code's tool system
|
||||||
|
|
||||||
|
## Portable Path References
|
||||||
|
|
||||||
|
### ${CLAUDE_PLUGIN_ROOT}
|
||||||
|
|
||||||
|
Use `${CLAUDE_PLUGIN_ROOT}` environment variable for all intra-plugin path references:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"command": "bash ${CLAUDE_PLUGIN_ROOT}/scripts/run.sh"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Why it matters**: Plugins install in different locations depending on:
|
||||||
|
- User installation method (marketplace, local, npm)
|
||||||
|
- Operating system conventions
|
||||||
|
- User preferences
|
||||||
|
|
||||||
|
**Where to use it**:
|
||||||
|
- Hook command paths
|
||||||
|
- MCP server command arguments
|
||||||
|
- Script execution references
|
||||||
|
- Resource file paths
|
||||||
|
|
||||||
|
**Never use**:
|
||||||
|
- Hardcoded absolute paths (`/Users/name/plugins/...`)
|
||||||
|
- Relative paths from working directory (`./scripts/...` in commands)
|
||||||
|
- Home directory shortcuts (`~/plugins/...`)
|
||||||
|
|
||||||
|
### Path Resolution Rules
|
||||||
|
|
||||||
|
**In manifest JSON fields** (hooks, MCP servers):
|
||||||
|
```json
|
||||||
|
"command": "${CLAUDE_PLUGIN_ROOT}/scripts/tool.sh"
|
||||||
|
```
|
||||||
|
|
||||||
|
**In component files** (commands, agents, skills):
|
||||||
|
```markdown
|
||||||
|
Reference scripts at: ${CLAUDE_PLUGIN_ROOT}/scripts/helper.py
|
||||||
|
```
|
||||||
|
|
||||||
|
**In executed scripts**:
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
# ${CLAUDE_PLUGIN_ROOT} available as environment variable
|
||||||
|
source "${CLAUDE_PLUGIN_ROOT}/lib/common.sh"
|
||||||
|
```
|
||||||
|
|
||||||
|
## File Naming Conventions
|
||||||
|
|
||||||
|
### Component Files
|
||||||
|
|
||||||
|
**Commands**: Use kebab-case `.md` files
|
||||||
|
- `code-review.md` → `/code-review`
|
||||||
|
- `run-tests.md` → `/run-tests`
|
||||||
|
- `api-docs.md` → `/api-docs`
|
||||||
|
|
||||||
|
**Agents**: Use kebab-case `.md` files describing role
|
||||||
|
- `test-generator.md`
|
||||||
|
- `code-reviewer.md`
|
||||||
|
- `performance-analyzer.md`
|
||||||
|
|
||||||
|
**Skills**: Use kebab-case directory names
|
||||||
|
- `api-testing/`
|
||||||
|
- `database-migrations/`
|
||||||
|
- `error-handling/`
|
||||||
|
|
||||||
|
### Supporting Files
|
||||||
|
|
||||||
|
**Scripts**: Use descriptive kebab-case names with appropriate extensions
|
||||||
|
- `validate-input.sh`
|
||||||
|
- `generate-report.py`
|
||||||
|
- `process-data.js`
|
||||||
|
|
||||||
|
**Documentation**: Use kebab-case markdown files
|
||||||
|
- `api-reference.md`
|
||||||
|
- `migration-guide.md`
|
||||||
|
- `best-practices.md`
|
||||||
|
|
||||||
|
**Configuration**: Use standard names
|
||||||
|
- `hooks.json`
|
||||||
|
- `.mcp.json`
|
||||||
|
- `plugin.json`
|
||||||
|
|
||||||
|
## Auto-Discovery Mechanism
|
||||||
|
|
||||||
|
Claude Code automatically discovers and loads components:
|
||||||
|
|
||||||
|
1. **Plugin manifest**: Reads `.claude-plugin/plugin.json` when plugin enables
|
||||||
|
2. **Commands**: Scans `commands/` directory for `.md` files
|
||||||
|
3. **Agents**: Scans `agents/` directory for `.md` files
|
||||||
|
4. **Skills**: Scans `skills/` for subdirectories containing `SKILL.md`
|
||||||
|
5. **Hooks**: Loads configuration from `hooks/hooks.json` or manifest
|
||||||
|
6. **MCP servers**: Loads configuration from `.mcp.json` or manifest
|
||||||
|
|
||||||
|
**Discovery timing**:
|
||||||
|
- Plugin installation: Components register with Claude Code
|
||||||
|
- Plugin enable: Components become available for use
|
||||||
|
- No restart required: Changes take effect on next Claude Code session
|
||||||
|
|
||||||
|
**Override behavior**: Custom paths in `plugin.json` supplement (not replace) default directories
|
||||||
|
|
||||||
|
## Best Practices
|
||||||
|
|
||||||
|
### Organization
|
||||||
|
|
||||||
|
1. **Logical grouping**: Group related components together
|
||||||
|
- Put test-related commands, agents, and skills together
|
||||||
|
- Create subdirectories in `scripts/` for different purposes
|
||||||
|
|
||||||
|
2. **Minimal manifest**: Keep `plugin.json` lean
|
||||||
|
- Only specify custom paths when necessary
|
||||||
|
- Rely on auto-discovery for standard layouts
|
||||||
|
- Use inline configuration only for simple cases
|
||||||
|
|
||||||
|
3. **Documentation**: Include README files
|
||||||
|
- Plugin root: Overall purpose and usage
|
||||||
|
- Component directories: Specific guidance
|
||||||
|
- Script directories: Usage and requirements
|
||||||
|
|
||||||
|
### Naming
|
||||||
|
|
||||||
|
1. **Consistency**: Use consistent naming across components
|
||||||
|
- If command is `test-runner`, name related agent `test-runner-agent`
|
||||||
|
- Match skill directory names to their purpose
|
||||||
|
|
||||||
|
2. **Clarity**: Use descriptive names that indicate purpose
|
||||||
|
- Good: `api-integration-testing/`, `code-quality-checker.md`
|
||||||
|
- Avoid: `utils/`, `misc.md`, `temp.sh`
|
||||||
|
|
||||||
|
3. **Length**: Balance brevity with clarity
|
||||||
|
- Commands: 2-3 words (`review-pr`, `run-ci`)
|
||||||
|
- Agents: Describe role clearly (`code-reviewer`, `test-generator`)
|
||||||
|
- Skills: Topic-focused (`error-handling`, `api-design`)
|
||||||
|
|
||||||
|
### Portability
|
||||||
|
|
||||||
|
1. **Always use ${CLAUDE_PLUGIN_ROOT}**: Never hardcode paths
|
||||||
|
2. **Test on multiple systems**: Verify on macOS, Linux, Windows
|
||||||
|
3. **Document dependencies**: List required tools and versions
|
||||||
|
4. **Avoid system-specific features**: Use portable bash/Python constructs
|
||||||
|
|
||||||
|
### Maintenance
|
||||||
|
|
||||||
|
1. **Version consistently**: Update version in plugin.json for releases
|
||||||
|
2. **Deprecate gracefully**: Mark old components clearly before removal
|
||||||
|
3. **Document breaking changes**: Note changes affecting existing users
|
||||||
|
4. **Test thoroughly**: Verify all components work after changes
|
||||||
|
|
||||||
|
## Common Patterns
|
||||||
|
|
||||||
|
### Minimal Plugin
|
||||||
|
|
||||||
|
Single command with no dependencies:
|
||||||
|
```
|
||||||
|
my-plugin/
|
||||||
|
├── .claude-plugin/
|
||||||
|
│ └── plugin.json # Just name field
|
||||||
|
└── commands/
|
||||||
|
└── hello.md # Single command
|
||||||
|
```
|
||||||
|
|
||||||
|
### Full-Featured Plugin
|
||||||
|
|
||||||
|
Complete plugin with all component types:
|
||||||
|
```
|
||||||
|
my-plugin/
|
||||||
|
├── .claude-plugin/
|
||||||
|
│ └── plugin.json
|
||||||
|
├── commands/ # User-facing commands
|
||||||
|
├── agents/ # Specialized subagents
|
||||||
|
├── skills/ # Auto-activating skills
|
||||||
|
├── hooks/ # Event handlers
|
||||||
|
│ ├── hooks.json
|
||||||
|
│ └── scripts/
|
||||||
|
├── .mcp.json # External integrations
|
||||||
|
└── scripts/ # Shared utilities
|
||||||
|
```
|
||||||
|
|
||||||
|
### Skill-Focused Plugin
|
||||||
|
|
||||||
|
Plugin providing only skills:
|
||||||
|
```
|
||||||
|
my-plugin/
|
||||||
|
├── .claude-plugin/
|
||||||
|
│ └── plugin.json
|
||||||
|
└── skills/
|
||||||
|
├── skill-one/
|
||||||
|
│ └── SKILL.md
|
||||||
|
└── skill-two/
|
||||||
|
└── SKILL.md
|
||||||
|
```
|
||||||
|
|
||||||
|
## Troubleshooting
|
||||||
|
|
||||||
|
**Component not loading**:
|
||||||
|
- Verify file is in correct directory with correct extension
|
||||||
|
- Check YAML frontmatter syntax (commands, agents, skills)
|
||||||
|
- Ensure skill has `SKILL.md` (not `README.md` or other name)
|
||||||
|
- Confirm plugin is enabled in Claude Code settings
|
||||||
|
|
||||||
|
**Path resolution errors**:
|
||||||
|
- Replace all hardcoded paths with `${CLAUDE_PLUGIN_ROOT}`
|
||||||
|
- Verify paths are relative and start with `./` in manifest
|
||||||
|
- Check that referenced files exist at specified paths
|
||||||
|
- Test with `echo $CLAUDE_PLUGIN_ROOT` in hook scripts
|
||||||
|
|
||||||
|
**Auto-discovery not working**:
|
||||||
|
- Confirm directories are at plugin root (not in `.claude-plugin/`)
|
||||||
|
- Check file naming follows conventions (kebab-case, correct extensions)
|
||||||
|
- Verify custom paths in manifest are correct
|
||||||
|
- Restart Claude Code to reload plugin configuration
|
||||||
|
|
||||||
|
**Conflicts between plugins**:
|
||||||
|
- Use unique, descriptive component names
|
||||||
|
- Namespace commands with plugin name if needed
|
||||||
|
- Document potential conflicts in plugin README
|
||||||
|
- Consider command prefixes for related functionality
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
For detailed examples and advanced patterns, see files in `references/` and `examples/` directories.
|
||||||
765
skills/plugin-structure/examples/advanced-plugin.md
Normal file
765
skills/plugin-structure/examples/advanced-plugin.md
Normal file
@@ -0,0 +1,765 @@
|
|||||||
|
# Advanced Plugin Example
|
||||||
|
|
||||||
|
A complex, enterprise-grade plugin with MCP integration and advanced organization.
|
||||||
|
|
||||||
|
## Directory Structure
|
||||||
|
|
||||||
|
```
|
||||||
|
enterprise-devops/
|
||||||
|
├── .claude-plugin/
|
||||||
|
│ └── plugin.json
|
||||||
|
├── commands/
|
||||||
|
│ ├── ci/
|
||||||
|
│ │ ├── build.md
|
||||||
|
│ │ ├── test.md
|
||||||
|
│ │ └── deploy.md
|
||||||
|
│ ├── monitoring/
|
||||||
|
│ │ ├── status.md
|
||||||
|
│ │ └── logs.md
|
||||||
|
│ └── admin/
|
||||||
|
│ ├── configure.md
|
||||||
|
│ └── manage.md
|
||||||
|
├── agents/
|
||||||
|
│ ├── orchestration/
|
||||||
|
│ │ ├── deployment-orchestrator.md
|
||||||
|
│ │ └── rollback-manager.md
|
||||||
|
│ └── specialized/
|
||||||
|
│ ├── kubernetes-expert.md
|
||||||
|
│ ├── terraform-expert.md
|
||||||
|
│ └── security-auditor.md
|
||||||
|
├── skills/
|
||||||
|
│ ├── kubernetes-ops/
|
||||||
|
│ │ ├── SKILL.md
|
||||||
|
│ │ ├── references/
|
||||||
|
│ │ │ ├── deployment-patterns.md
|
||||||
|
│ │ │ ├── troubleshooting.md
|
||||||
|
│ │ │ └── security.md
|
||||||
|
│ │ ├── examples/
|
||||||
|
│ │ │ ├── basic-deployment.yaml
|
||||||
|
│ │ │ ├── stateful-set.yaml
|
||||||
|
│ │ │ └── ingress-config.yaml
|
||||||
|
│ │ └── scripts/
|
||||||
|
│ │ ├── validate-manifest.sh
|
||||||
|
│ │ └── health-check.sh
|
||||||
|
│ ├── terraform-iac/
|
||||||
|
│ │ ├── SKILL.md
|
||||||
|
│ │ ├── references/
|
||||||
|
│ │ │ └── best-practices.md
|
||||||
|
│ │ └── examples/
|
||||||
|
│ │ └── module-template/
|
||||||
|
│ └── ci-cd-pipelines/
|
||||||
|
│ ├── SKILL.md
|
||||||
|
│ └── references/
|
||||||
|
│ └── pipeline-patterns.md
|
||||||
|
├── hooks/
|
||||||
|
│ ├── hooks.json
|
||||||
|
│ └── scripts/
|
||||||
|
│ ├── security/
|
||||||
|
│ │ ├── scan-secrets.sh
|
||||||
|
│ │ ├── validate-permissions.sh
|
||||||
|
│ │ └── audit-changes.sh
|
||||||
|
│ ├── quality/
|
||||||
|
│ │ ├── check-config.sh
|
||||||
|
│ │ └── verify-tests.sh
|
||||||
|
│ └── workflow/
|
||||||
|
│ ├── notify-team.sh
|
||||||
|
│ └── update-status.sh
|
||||||
|
├── .mcp.json
|
||||||
|
├── servers/
|
||||||
|
│ ├── kubernetes-mcp/
|
||||||
|
│ │ ├── index.js
|
||||||
|
│ │ ├── package.json
|
||||||
|
│ │ └── lib/
|
||||||
|
│ ├── terraform-mcp/
|
||||||
|
│ │ ├── main.py
|
||||||
|
│ │ └── requirements.txt
|
||||||
|
│ └── github-actions-mcp/
|
||||||
|
│ ├── server.js
|
||||||
|
│ └── package.json
|
||||||
|
├── lib/
|
||||||
|
│ ├── core/
|
||||||
|
│ │ ├── logger.js
|
||||||
|
│ │ ├── config.js
|
||||||
|
│ │ └── auth.js
|
||||||
|
│ ├── integrations/
|
||||||
|
│ │ ├── slack.js
|
||||||
|
│ │ ├── pagerduty.js
|
||||||
|
│ │ └── datadog.js
|
||||||
|
│ └── utils/
|
||||||
|
│ ├── retry.js
|
||||||
|
│ └── validation.js
|
||||||
|
└── config/
|
||||||
|
├── environments/
|
||||||
|
│ ├── production.json
|
||||||
|
│ ├── staging.json
|
||||||
|
│ └── development.json
|
||||||
|
└── templates/
|
||||||
|
├── deployment.yaml
|
||||||
|
└── service.yaml
|
||||||
|
```
|
||||||
|
|
||||||
|
## File Contents
|
||||||
|
|
||||||
|
### .claude-plugin/plugin.json
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"name": "enterprise-devops",
|
||||||
|
"version": "2.3.1",
|
||||||
|
"description": "Comprehensive DevOps automation for enterprise CI/CD pipelines, infrastructure management, and monitoring",
|
||||||
|
"author": {
|
||||||
|
"name": "DevOps Platform Team",
|
||||||
|
"email": "devops-platform@company.com",
|
||||||
|
"url": "https://company.com/teams/devops"
|
||||||
|
},
|
||||||
|
"homepage": "https://docs.company.com/plugins/devops",
|
||||||
|
"repository": {
|
||||||
|
"type": "git",
|
||||||
|
"url": "https://github.com/company/devops-plugin.git"
|
||||||
|
},
|
||||||
|
"license": "Apache-2.0",
|
||||||
|
"keywords": [
|
||||||
|
"devops",
|
||||||
|
"ci-cd",
|
||||||
|
"kubernetes",
|
||||||
|
"terraform",
|
||||||
|
"automation",
|
||||||
|
"infrastructure",
|
||||||
|
"deployment",
|
||||||
|
"monitoring"
|
||||||
|
],
|
||||||
|
"commands": [
|
||||||
|
"./commands/ci",
|
||||||
|
"./commands/monitoring",
|
||||||
|
"./commands/admin"
|
||||||
|
],
|
||||||
|
"agents": [
|
||||||
|
"./agents/orchestration",
|
||||||
|
"./agents/specialized"
|
||||||
|
],
|
||||||
|
"hooks": "./hooks/hooks.json",
|
||||||
|
"mcpServers": "./.mcp.json"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### .mcp.json
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"mcpServers": {
|
||||||
|
"kubernetes": {
|
||||||
|
"command": "node",
|
||||||
|
"args": ["${CLAUDE_PLUGIN_ROOT}/servers/kubernetes-mcp/index.js"],
|
||||||
|
"env": {
|
||||||
|
"KUBECONFIG": "${KUBECONFIG}",
|
||||||
|
"K8S_NAMESPACE": "${K8S_NAMESPACE:-default}"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"terraform": {
|
||||||
|
"command": "python",
|
||||||
|
"args": ["${CLAUDE_PLUGIN_ROOT}/servers/terraform-mcp/main.py"],
|
||||||
|
"env": {
|
||||||
|
"TF_STATE_BUCKET": "${TF_STATE_BUCKET}",
|
||||||
|
"AWS_REGION": "${AWS_REGION}"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"github-actions": {
|
||||||
|
"command": "node",
|
||||||
|
"args": ["${CLAUDE_PLUGIN_ROOT}/servers/github-actions-mcp/server.js"],
|
||||||
|
"env": {
|
||||||
|
"GITHUB_TOKEN": "${GITHUB_TOKEN}",
|
||||||
|
"GITHUB_ORG": "${GITHUB_ORG}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### commands/ci/build.md
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
name: build
|
||||||
|
description: Trigger and monitor CI build pipeline
|
||||||
|
---
|
||||||
|
|
||||||
|
# Build Command
|
||||||
|
|
||||||
|
Trigger CI/CD build pipeline and monitor progress in real-time.
|
||||||
|
|
||||||
|
## Process
|
||||||
|
|
||||||
|
1. **Validation**: Check prerequisites
|
||||||
|
- Verify branch status
|
||||||
|
- Check for uncommitted changes
|
||||||
|
- Validate configuration files
|
||||||
|
|
||||||
|
2. **Trigger**: Start build via MCP server
|
||||||
|
\`\`\`javascript
|
||||||
|
// Uses github-actions MCP server
|
||||||
|
const build = await tools.github_actions_trigger_workflow({
|
||||||
|
workflow: 'build.yml',
|
||||||
|
ref: currentBranch
|
||||||
|
})
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
3. **Monitor**: Track build progress
|
||||||
|
- Display real-time logs
|
||||||
|
- Show test results as they complete
|
||||||
|
- Alert on failures
|
||||||
|
|
||||||
|
4. **Report**: Summarize results
|
||||||
|
- Build status
|
||||||
|
- Test coverage
|
||||||
|
- Performance metrics
|
||||||
|
- Deploy readiness
|
||||||
|
|
||||||
|
## Integration
|
||||||
|
|
||||||
|
After successful build:
|
||||||
|
- Offer to deploy to staging
|
||||||
|
- Suggest performance optimizations
|
||||||
|
- Generate deployment checklist
|
||||||
|
```
|
||||||
|
|
||||||
|
### agents/orchestration/deployment-orchestrator.md
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Orchestrates complex multi-environment deployments with rollback capabilities and health monitoring
|
||||||
|
capabilities:
|
||||||
|
- Plan and execute multi-stage deployments
|
||||||
|
- Coordinate service dependencies
|
||||||
|
- Monitor deployment health
|
||||||
|
- Execute automated rollbacks
|
||||||
|
- Manage deployment approvals
|
||||||
|
---
|
||||||
|
|
||||||
|
# Deployment Orchestrator Agent
|
||||||
|
|
||||||
|
Specialized agent for orchestrating complex deployments across multiple environments.
|
||||||
|
|
||||||
|
## Expertise
|
||||||
|
|
||||||
|
- **Deployment strategies**: Blue-green, canary, rolling updates
|
||||||
|
- **Dependency management**: Service startup ordering, dependency injection
|
||||||
|
- **Health monitoring**: Service health checks, metric validation
|
||||||
|
- **Rollback automation**: Automatic rollback on failure detection
|
||||||
|
- **Approval workflows**: Multi-stage approval processes
|
||||||
|
|
||||||
|
## Orchestration Process
|
||||||
|
|
||||||
|
1. **Planning Phase**
|
||||||
|
- Analyze deployment requirements
|
||||||
|
- Identify service dependencies
|
||||||
|
- Generate deployment plan
|
||||||
|
- Calculate rollback strategy
|
||||||
|
|
||||||
|
2. **Validation Phase**
|
||||||
|
- Verify environment readiness
|
||||||
|
- Check resource availability
|
||||||
|
- Validate configurations
|
||||||
|
- Run pre-deployment tests
|
||||||
|
|
||||||
|
3. **Execution Phase**
|
||||||
|
- Deploy services in dependency order
|
||||||
|
- Monitor health after each stage
|
||||||
|
- Validate metrics and logs
|
||||||
|
- Proceed to next stage on success
|
||||||
|
|
||||||
|
4. **Verification Phase**
|
||||||
|
- Run smoke tests
|
||||||
|
- Validate service integration
|
||||||
|
- Check performance metrics
|
||||||
|
- Confirm deployment success
|
||||||
|
|
||||||
|
5. **Rollback Phase** (if needed)
|
||||||
|
- Detect failure conditions
|
||||||
|
- Execute rollback plan
|
||||||
|
- Restore previous state
|
||||||
|
- Notify stakeholders
|
||||||
|
|
||||||
|
## MCP Integration
|
||||||
|
|
||||||
|
Uses multiple MCP servers:
|
||||||
|
- `kubernetes`: Deploy and manage containers
|
||||||
|
- `terraform`: Provision infrastructure
|
||||||
|
- `github-actions`: Trigger deployment pipelines
|
||||||
|
|
||||||
|
## Monitoring Integration
|
||||||
|
|
||||||
|
Integrates with monitoring tools via lib:
|
||||||
|
\`\`\`javascript
|
||||||
|
const { DatadogClient } = require('${CLAUDE_PLUGIN_ROOT}/lib/integrations/datadog')
|
||||||
|
const metrics = await DatadogClient.getMetrics(service, timeRange)
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
## Notification Integration
|
||||||
|
|
||||||
|
Sends updates via Slack and PagerDuty:
|
||||||
|
\`\`\`javascript
|
||||||
|
const { SlackClient } = require('${CLAUDE_PLUGIN_ROOT}/lib/integrations/slack')
|
||||||
|
await SlackClient.notify({
|
||||||
|
channel: '#deployments',
|
||||||
|
message: 'Deployment started',
|
||||||
|
metadata: deploymentPlan
|
||||||
|
})
|
||||||
|
\`\`\`
|
||||||
|
```
|
||||||
|
|
||||||
|
### skills/kubernetes-ops/SKILL.md
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
name: Kubernetes Operations
|
||||||
|
description: This skill should be used when deploying to Kubernetes, managing K8s resources, troubleshooting cluster issues, configuring ingress/services, scaling deployments, or working with Kubernetes manifests. Provides comprehensive Kubernetes operational knowledge and best practices.
|
||||||
|
version: 2.0.0
|
||||||
|
---
|
||||||
|
|
||||||
|
# Kubernetes Operations
|
||||||
|
|
||||||
|
Comprehensive operational knowledge for managing Kubernetes clusters and workloads.
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
|
||||||
|
Manage Kubernetes infrastructure effectively through:
|
||||||
|
- Deployment strategies and patterns
|
||||||
|
- Resource configuration and optimization
|
||||||
|
- Troubleshooting and debugging
|
||||||
|
- Security best practices
|
||||||
|
- Performance tuning
|
||||||
|
|
||||||
|
## Core Concepts
|
||||||
|
|
||||||
|
### Resource Management
|
||||||
|
|
||||||
|
**Deployments**: Use for stateless applications
|
||||||
|
- Rolling updates for zero-downtime deployments
|
||||||
|
- Rollback capabilities for failed deployments
|
||||||
|
- Replica management for scaling
|
||||||
|
|
||||||
|
**StatefulSets**: Use for stateful applications
|
||||||
|
- Stable network identities
|
||||||
|
- Persistent storage
|
||||||
|
- Ordered deployment and scaling
|
||||||
|
|
||||||
|
**DaemonSets**: Use for node-level services
|
||||||
|
- Log collectors
|
||||||
|
- Monitoring agents
|
||||||
|
- Network plugins
|
||||||
|
|
||||||
|
### Configuration
|
||||||
|
|
||||||
|
**ConfigMaps**: Store non-sensitive configuration
|
||||||
|
- Environment-specific settings
|
||||||
|
- Application configuration files
|
||||||
|
- Feature flags
|
||||||
|
|
||||||
|
**Secrets**: Store sensitive data
|
||||||
|
- API keys and tokens
|
||||||
|
- Database credentials
|
||||||
|
- TLS certificates
|
||||||
|
|
||||||
|
Use external secret management (Vault, AWS Secrets Manager) for production.
|
||||||
|
|
||||||
|
### Networking
|
||||||
|
|
||||||
|
**Services**: Expose applications internally
|
||||||
|
- ClusterIP for internal communication
|
||||||
|
- NodePort for external access (non-production)
|
||||||
|
- LoadBalancer for external access (production)
|
||||||
|
|
||||||
|
**Ingress**: HTTP/HTTPS routing
|
||||||
|
- Path-based routing
|
||||||
|
- Host-based routing
|
||||||
|
- TLS termination
|
||||||
|
- Load balancing
|
||||||
|
|
||||||
|
## Deployment Strategies
|
||||||
|
|
||||||
|
### Rolling Update
|
||||||
|
|
||||||
|
Default strategy, gradual replacement:
|
||||||
|
\`\`\`yaml
|
||||||
|
strategy:
|
||||||
|
type: RollingUpdate
|
||||||
|
rollingUpdate:
|
||||||
|
maxSurge: 1
|
||||||
|
maxUnavailable: 0
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
**When to use**: Standard deployments, minor updates
|
||||||
|
|
||||||
|
### Recreate
|
||||||
|
|
||||||
|
Stop all pods, then create new ones:
|
||||||
|
\`\`\`yaml
|
||||||
|
strategy:
|
||||||
|
type: Recreate
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
**When to use**: Stateful apps that can't run multiple versions
|
||||||
|
|
||||||
|
### Blue-Green
|
||||||
|
|
||||||
|
Run two complete environments, switch traffic:
|
||||||
|
1. Deploy new version (green)
|
||||||
|
2. Test green environment
|
||||||
|
3. Switch traffic to green
|
||||||
|
4. Keep blue for quick rollback
|
||||||
|
|
||||||
|
**When to use**: Critical services, need instant rollback
|
||||||
|
|
||||||
|
### Canary
|
||||||
|
|
||||||
|
Gradually roll out to subset of users:
|
||||||
|
1. Deploy canary version (10% traffic)
|
||||||
|
2. Monitor metrics and errors
|
||||||
|
3. Increase traffic gradually
|
||||||
|
4. Complete rollout or rollback
|
||||||
|
|
||||||
|
**When to use**: High-risk changes, want gradual validation
|
||||||
|
|
||||||
|
## Resource Configuration
|
||||||
|
|
||||||
|
### Resource Requests and Limits
|
||||||
|
|
||||||
|
Always set for production workloads:
|
||||||
|
\`\`\`yaml
|
||||||
|
resources:
|
||||||
|
requests:
|
||||||
|
memory: "256Mi"
|
||||||
|
cpu: "250m"
|
||||||
|
limits:
|
||||||
|
memory: "512Mi"
|
||||||
|
cpu: "500m"
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
**Requests**: Guaranteed resources
|
||||||
|
**Limits**: Maximum allowed resources
|
||||||
|
|
||||||
|
### Health Checks
|
||||||
|
|
||||||
|
Essential for reliability:
|
||||||
|
\`\`\`yaml
|
||||||
|
livenessProbe:
|
||||||
|
httpGet:
|
||||||
|
path: /health
|
||||||
|
port: 8080
|
||||||
|
initialDelaySeconds: 30
|
||||||
|
periodSeconds: 10
|
||||||
|
|
||||||
|
readinessProbe:
|
||||||
|
httpGet:
|
||||||
|
path: /ready
|
||||||
|
port: 8080
|
||||||
|
initialDelaySeconds: 5
|
||||||
|
periodSeconds: 5
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
**Liveness**: Restart unhealthy pods
|
||||||
|
**Readiness**: Remove unready pods from service
|
||||||
|
|
||||||
|
## Troubleshooting
|
||||||
|
|
||||||
|
### Common Issues
|
||||||
|
|
||||||
|
1. **Pods not starting**
|
||||||
|
- Check: `kubectl describe pod <name>`
|
||||||
|
- Look for: Image pull errors, resource constraints
|
||||||
|
- Fix: Verify image name, increase resources
|
||||||
|
|
||||||
|
2. **Service not reachable**
|
||||||
|
- Check: `kubectl get svc`, `kubectl get endpoints`
|
||||||
|
- Look for: No endpoints, wrong selector
|
||||||
|
- Fix: Verify pod labels match service selector
|
||||||
|
|
||||||
|
3. **High memory usage**
|
||||||
|
- Check: `kubectl top pods`
|
||||||
|
- Look for: Pods near memory limit
|
||||||
|
- Fix: Increase limits, optimize application
|
||||||
|
|
||||||
|
4. **Frequent restarts**
|
||||||
|
- Check: `kubectl get pods`, `kubectl logs <name>`
|
||||||
|
- Look for: Liveness probe failures, OOMKilled
|
||||||
|
- Fix: Adjust health checks, increase memory
|
||||||
|
|
||||||
|
### Debugging Commands
|
||||||
|
|
||||||
|
Get pod details:
|
||||||
|
\`\`\`bash
|
||||||
|
kubectl describe pod <name>
|
||||||
|
kubectl logs <name>
|
||||||
|
kubectl logs <name> --previous # logs from crashed container
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
Execute commands in pod:
|
||||||
|
\`\`\`bash
|
||||||
|
kubectl exec -it <name> -- /bin/sh
|
||||||
|
kubectl exec <name> -- env
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
Check resource usage:
|
||||||
|
\`\`\`bash
|
||||||
|
kubectl top nodes
|
||||||
|
kubectl top pods
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
## Security Best Practices
|
||||||
|
|
||||||
|
### Pod Security
|
||||||
|
|
||||||
|
- Run as non-root user
|
||||||
|
- Use read-only root filesystem
|
||||||
|
- Drop unnecessary capabilities
|
||||||
|
- Use security contexts
|
||||||
|
|
||||||
|
Example:
|
||||||
|
\`\`\`yaml
|
||||||
|
securityContext:
|
||||||
|
runAsNonRoot: true
|
||||||
|
runAsUser: 1000
|
||||||
|
readOnlyRootFilesystem: true
|
||||||
|
capabilities:
|
||||||
|
drop:
|
||||||
|
- ALL
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
### Network Policies
|
||||||
|
|
||||||
|
Restrict pod communication:
|
||||||
|
\`\`\`yaml
|
||||||
|
apiVersion: networking.k8s.io/v1
|
||||||
|
kind: NetworkPolicy
|
||||||
|
metadata:
|
||||||
|
name: api-allow
|
||||||
|
spec:
|
||||||
|
podSelector:
|
||||||
|
matchLabels:
|
||||||
|
app: api
|
||||||
|
ingress:
|
||||||
|
- from:
|
||||||
|
- podSelector:
|
||||||
|
matchLabels:
|
||||||
|
app: frontend
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
### Secrets Management
|
||||||
|
|
||||||
|
- Never commit secrets to git
|
||||||
|
- Use external secret managers
|
||||||
|
- Rotate secrets regularly
|
||||||
|
- Limit secret access with RBAC
|
||||||
|
|
||||||
|
## Performance Optimization
|
||||||
|
|
||||||
|
### Resource Tuning
|
||||||
|
|
||||||
|
1. **Start conservative**: Set low limits initially
|
||||||
|
2. **Monitor usage**: Track actual resource consumption
|
||||||
|
3. **Adjust gradually**: Increase based on metrics
|
||||||
|
4. **Set appropriate requests**: Match typical usage
|
||||||
|
5. **Set safe limits**: 2x requests for headroom
|
||||||
|
|
||||||
|
### Horizontal Pod Autoscaling
|
||||||
|
|
||||||
|
Automatically scale based on metrics:
|
||||||
|
\`\`\`yaml
|
||||||
|
apiVersion: autoscaling/v2
|
||||||
|
kind: HorizontalPodAutoscaler
|
||||||
|
metadata:
|
||||||
|
name: api-hpa
|
||||||
|
spec:
|
||||||
|
scaleTargetRef:
|
||||||
|
apiVersion: apps/v1
|
||||||
|
kind: Deployment
|
||||||
|
name: api
|
||||||
|
minReplicas: 2
|
||||||
|
maxReplicas: 10
|
||||||
|
metrics:
|
||||||
|
- type: Resource
|
||||||
|
resource:
|
||||||
|
name: cpu
|
||||||
|
target:
|
||||||
|
type: Utilization
|
||||||
|
averageUtilization: 70
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
## MCP Server Integration
|
||||||
|
|
||||||
|
This skill works with the kubernetes MCP server for operations:
|
||||||
|
|
||||||
|
**List pods**:
|
||||||
|
\`\`\`javascript
|
||||||
|
const pods = await tools.k8s_list_pods({ namespace: 'default' })
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
**Get pod logs**:
|
||||||
|
\`\`\`javascript
|
||||||
|
const logs = await tools.k8s_get_logs({ pod: 'api-xyz', container: 'app' })
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
**Apply manifests**:
|
||||||
|
\`\`\`javascript
|
||||||
|
const result = await tools.k8s_apply_manifest({ file: 'deployment.yaml' })
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
## Detailed References
|
||||||
|
|
||||||
|
For in-depth information:
|
||||||
|
- **Deployment patterns**: `references/deployment-patterns.md`
|
||||||
|
- **Troubleshooting guide**: `references/troubleshooting.md`
|
||||||
|
- **Security hardening**: `references/security.md`
|
||||||
|
|
||||||
|
## Example Manifests
|
||||||
|
|
||||||
|
For copy-paste examples:
|
||||||
|
- **Basic deployment**: `examples/basic-deployment.yaml`
|
||||||
|
- **StatefulSet**: `examples/stateful-set.yaml`
|
||||||
|
- **Ingress config**: `examples/ingress-config.yaml`
|
||||||
|
|
||||||
|
## Validation Scripts
|
||||||
|
|
||||||
|
For manifest validation:
|
||||||
|
\`\`\`bash
|
||||||
|
bash ${CLAUDE_PLUGIN_ROOT}/skills/kubernetes-ops/scripts/validate-manifest.sh deployment.yaml
|
||||||
|
\`\`\`
|
||||||
|
```
|
||||||
|
|
||||||
|
### hooks/hooks.json
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"PreToolUse": [
|
||||||
|
{
|
||||||
|
"matcher": "Write|Edit",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"command": "bash ${CLAUDE_PLUGIN_ROOT}/hooks/scripts/security/scan-secrets.sh",
|
||||||
|
"timeout": 30
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"matcher": "Bash",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "prompt",
|
||||||
|
"prompt": "Evaluate if this bash command is safe for production environment. Check for destructive operations, missing safeguards, and potential security issues. Commands should be idempotent and reversible.",
|
||||||
|
"timeout": 20
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"PostToolUse": [
|
||||||
|
{
|
||||||
|
"matcher": "Bash",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"command": "bash ${CLAUDE_PLUGIN_ROOT}/hooks/scripts/workflow/update-status.sh",
|
||||||
|
"timeout": 15
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"Stop": [
|
||||||
|
{
|
||||||
|
"matcher": ".*",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"command": "bash ${CLAUDE_PLUGIN_ROOT}/hooks/scripts/quality/check-config.sh",
|
||||||
|
"timeout": 45
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"command": "bash ${CLAUDE_PLUGIN_ROOT}/hooks/scripts/workflow/notify-team.sh",
|
||||||
|
"timeout": 30
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"SessionStart": [
|
||||||
|
{
|
||||||
|
"matcher": ".*",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"command": "bash ${CLAUDE_PLUGIN_ROOT}/hooks/scripts/security/validate-permissions.sh",
|
||||||
|
"timeout": 20
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## Key Features
|
||||||
|
|
||||||
|
### Multi-Level Organization
|
||||||
|
|
||||||
|
**Commands**: Organized by function (CI, monitoring, admin)
|
||||||
|
**Agents**: Separated by role (orchestration vs. specialized)
|
||||||
|
**Skills**: Rich resources (references, examples, scripts)
|
||||||
|
|
||||||
|
### MCP Integration
|
||||||
|
|
||||||
|
Three custom MCP servers:
|
||||||
|
- **Kubernetes**: Cluster operations
|
||||||
|
- **Terraform**: Infrastructure provisioning
|
||||||
|
- **GitHub Actions**: CI/CD automation
|
||||||
|
|
||||||
|
### Shared Libraries
|
||||||
|
|
||||||
|
Reusable code in `lib/`:
|
||||||
|
- **Core**: Common utilities (logging, config, auth)
|
||||||
|
- **Integrations**: External services (Slack, Datadog)
|
||||||
|
- **Utils**: Helper functions (retry, validation)
|
||||||
|
|
||||||
|
### Configuration Management
|
||||||
|
|
||||||
|
Environment-specific configs in `config/`:
|
||||||
|
- **Environments**: Per-environment settings
|
||||||
|
- **Templates**: Reusable deployment templates
|
||||||
|
|
||||||
|
### Security Automation
|
||||||
|
|
||||||
|
Multiple security hooks:
|
||||||
|
- Secret scanning before writes
|
||||||
|
- Permission validation on session start
|
||||||
|
- Configuration auditing on completion
|
||||||
|
|
||||||
|
### Monitoring Integration
|
||||||
|
|
||||||
|
Built-in monitoring via lib integrations:
|
||||||
|
- Datadog for metrics
|
||||||
|
- PagerDuty for alerts
|
||||||
|
- Slack for notifications
|
||||||
|
|
||||||
|
## Use Cases
|
||||||
|
|
||||||
|
1. **Multi-environment deployments**: Orchestrated rollouts across dev/staging/prod
|
||||||
|
2. **Infrastructure as code**: Terraform automation with state management
|
||||||
|
3. **CI/CD automation**: Build, test, deploy pipelines
|
||||||
|
4. **Monitoring and observability**: Integrated metrics and alerting
|
||||||
|
5. **Security enforcement**: Automated security scanning and validation
|
||||||
|
6. **Team collaboration**: Slack notifications and status updates
|
||||||
|
|
||||||
|
## When to Use This Pattern
|
||||||
|
|
||||||
|
- Large-scale enterprise deployments
|
||||||
|
- Multiple environment management
|
||||||
|
- Complex CI/CD workflows
|
||||||
|
- Integrated monitoring requirements
|
||||||
|
- Security-critical infrastructure
|
||||||
|
- Team collaboration needs
|
||||||
|
|
||||||
|
## Scaling Considerations
|
||||||
|
|
||||||
|
- **Performance**: Separate MCP servers for parallel operations
|
||||||
|
- **Organization**: Multi-level directories for scalability
|
||||||
|
- **Maintainability**: Shared libraries reduce duplication
|
||||||
|
- **Flexibility**: Environment configs enable customization
|
||||||
|
- **Security**: Layered security hooks and validation
|
||||||
83
skills/plugin-structure/examples/minimal-plugin.md
Normal file
83
skills/plugin-structure/examples/minimal-plugin.md
Normal file
@@ -0,0 +1,83 @@
|
|||||||
|
# Minimal Plugin Example
|
||||||
|
|
||||||
|
A bare-bones plugin with a single command.
|
||||||
|
|
||||||
|
## Directory Structure
|
||||||
|
|
||||||
|
```
|
||||||
|
hello-world/
|
||||||
|
├── .claude-plugin/
|
||||||
|
│ └── plugin.json
|
||||||
|
└── commands/
|
||||||
|
└── hello.md
|
||||||
|
```
|
||||||
|
|
||||||
|
## File Contents
|
||||||
|
|
||||||
|
### .claude-plugin/plugin.json
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"name": "hello-world"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### commands/hello.md
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
name: hello
|
||||||
|
description: Prints a friendly greeting message
|
||||||
|
---
|
||||||
|
|
||||||
|
# Hello Command
|
||||||
|
|
||||||
|
Print a friendly greeting to the user.
|
||||||
|
|
||||||
|
## Implementation
|
||||||
|
|
||||||
|
Output the following message to the user:
|
||||||
|
|
||||||
|
> Hello! This is a simple command from the hello-world plugin.
|
||||||
|
>
|
||||||
|
> Use this as a starting point for building more complex plugins.
|
||||||
|
|
||||||
|
Include the current timestamp in the greeting to show the command executed successfully.
|
||||||
|
```
|
||||||
|
|
||||||
|
## Usage
|
||||||
|
|
||||||
|
After installing the plugin:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ claude
|
||||||
|
> /hello
|
||||||
|
Hello! This is a simple command from the hello-world plugin.
|
||||||
|
|
||||||
|
Use this as a starting point for building more complex plugins.
|
||||||
|
|
||||||
|
Executed at: 2025-01-15 14:30:22 UTC
|
||||||
|
```
|
||||||
|
|
||||||
|
## Key Points
|
||||||
|
|
||||||
|
1. **Minimal manifest**: Only the required `name` field
|
||||||
|
2. **Single command**: One markdown file in `commands/` directory
|
||||||
|
3. **Auto-discovery**: Claude Code finds the command automatically
|
||||||
|
4. **No dependencies**: No scripts, hooks, or external resources
|
||||||
|
|
||||||
|
## When to Use This Pattern
|
||||||
|
|
||||||
|
- Quick prototypes
|
||||||
|
- Single-purpose utilities
|
||||||
|
- Learning plugin development
|
||||||
|
- Internal team tools with one specific function
|
||||||
|
|
||||||
|
## Extending This Plugin
|
||||||
|
|
||||||
|
To add more functionality:
|
||||||
|
|
||||||
|
1. **Add commands**: Create more `.md` files in `commands/`
|
||||||
|
2. **Add metadata**: Update `plugin.json` with version, description, author
|
||||||
|
3. **Add agents**: Create `agents/` directory with agent definitions
|
||||||
|
4. **Add hooks**: Create `hooks/hooks.json` for event handling
|
||||||
587
skills/plugin-structure/examples/standard-plugin.md
Normal file
587
skills/plugin-structure/examples/standard-plugin.md
Normal file
@@ -0,0 +1,587 @@
|
|||||||
|
# Standard Plugin Example
|
||||||
|
|
||||||
|
A well-structured plugin with commands, agents, and skills.
|
||||||
|
|
||||||
|
## Directory Structure
|
||||||
|
|
||||||
|
```
|
||||||
|
code-quality/
|
||||||
|
├── .claude-plugin/
|
||||||
|
│ └── plugin.json
|
||||||
|
├── commands/
|
||||||
|
│ ├── lint.md
|
||||||
|
│ ├── test.md
|
||||||
|
│ └── review.md
|
||||||
|
├── agents/
|
||||||
|
│ ├── code-reviewer.md
|
||||||
|
│ └── test-generator.md
|
||||||
|
├── skills/
|
||||||
|
│ ├── code-standards/
|
||||||
|
│ │ ├── SKILL.md
|
||||||
|
│ │ └── references/
|
||||||
|
│ │ └── style-guide.md
|
||||||
|
│ └── testing-patterns/
|
||||||
|
│ ├── SKILL.md
|
||||||
|
│ └── examples/
|
||||||
|
│ ├── unit-test.js
|
||||||
|
│ └── integration-test.js
|
||||||
|
├── hooks/
|
||||||
|
│ ├── hooks.json
|
||||||
|
│ └── scripts/
|
||||||
|
│ └── validate-commit.sh
|
||||||
|
└── scripts/
|
||||||
|
├── run-linter.sh
|
||||||
|
└── generate-report.py
|
||||||
|
```
|
||||||
|
|
||||||
|
## File Contents
|
||||||
|
|
||||||
|
### .claude-plugin/plugin.json
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"name": "code-quality",
|
||||||
|
"version": "1.0.0",
|
||||||
|
"description": "Comprehensive code quality tools including linting, testing, and review automation",
|
||||||
|
"author": {
|
||||||
|
"name": "Quality Team",
|
||||||
|
"email": "quality@example.com"
|
||||||
|
},
|
||||||
|
"homepage": "https://docs.example.com/plugins/code-quality",
|
||||||
|
"repository": "https://github.com/example/code-quality-plugin",
|
||||||
|
"license": "MIT",
|
||||||
|
"keywords": ["code-quality", "linting", "testing", "code-review", "automation"]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### commands/lint.md
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
name: lint
|
||||||
|
description: Run linting checks on the codebase
|
||||||
|
---
|
||||||
|
|
||||||
|
# Lint Command
|
||||||
|
|
||||||
|
Run comprehensive linting checks on the project codebase.
|
||||||
|
|
||||||
|
## Process
|
||||||
|
|
||||||
|
1. Detect project type and installed linters
|
||||||
|
2. Run appropriate linters (ESLint, Pylint, RuboCop, etc.)
|
||||||
|
3. Collect and format results
|
||||||
|
4. Report issues with file locations and severity
|
||||||
|
|
||||||
|
## Implementation
|
||||||
|
|
||||||
|
Execute the linting script:
|
||||||
|
|
||||||
|
\`\`\`bash
|
||||||
|
bash ${CLAUDE_PLUGIN_ROOT}/scripts/run-linter.sh
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
Parse the output and present issues organized by:
|
||||||
|
- Critical issues (must fix)
|
||||||
|
- Warnings (should fix)
|
||||||
|
- Style suggestions (optional)
|
||||||
|
|
||||||
|
For each issue, show:
|
||||||
|
- File path and line number
|
||||||
|
- Issue description
|
||||||
|
- Suggested fix (if available)
|
||||||
|
```
|
||||||
|
|
||||||
|
### commands/test.md
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
name: test
|
||||||
|
description: Run test suite with coverage reporting
|
||||||
|
---
|
||||||
|
|
||||||
|
# Test Command
|
||||||
|
|
||||||
|
Execute the project test suite and generate coverage reports.
|
||||||
|
|
||||||
|
## Process
|
||||||
|
|
||||||
|
1. Identify test framework (Jest, pytest, RSpec, etc.)
|
||||||
|
2. Run all tests
|
||||||
|
3. Generate coverage report
|
||||||
|
4. Identify untested code
|
||||||
|
|
||||||
|
## Output
|
||||||
|
|
||||||
|
Present results in structured format:
|
||||||
|
- Test summary (passed/failed/skipped)
|
||||||
|
- Coverage percentage by file
|
||||||
|
- Critical untested areas
|
||||||
|
- Failed test details
|
||||||
|
|
||||||
|
## Integration
|
||||||
|
|
||||||
|
After test completion, offer to:
|
||||||
|
- Fix failing tests
|
||||||
|
- Generate tests for untested code (using test-generator agent)
|
||||||
|
- Update documentation based on test changes
|
||||||
|
```
|
||||||
|
|
||||||
|
### agents/code-reviewer.md
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Expert code reviewer specializing in identifying bugs, security issues, and improvement opportunities
|
||||||
|
capabilities:
|
||||||
|
- Analyze code for potential bugs and logic errors
|
||||||
|
- Identify security vulnerabilities
|
||||||
|
- Suggest performance improvements
|
||||||
|
- Ensure code follows project standards
|
||||||
|
- Review test coverage adequacy
|
||||||
|
---
|
||||||
|
|
||||||
|
# Code Reviewer Agent
|
||||||
|
|
||||||
|
Specialized agent for comprehensive code review.
|
||||||
|
|
||||||
|
## Expertise
|
||||||
|
|
||||||
|
- **Bug detection**: Logic errors, edge cases, error handling
|
||||||
|
- **Security analysis**: Injection vulnerabilities, authentication issues, data exposure
|
||||||
|
- **Performance**: Algorithm efficiency, resource usage, optimization opportunities
|
||||||
|
- **Standards compliance**: Style guide adherence, naming conventions, documentation
|
||||||
|
- **Test coverage**: Adequacy of test cases, missing scenarios
|
||||||
|
|
||||||
|
## Review Process
|
||||||
|
|
||||||
|
1. **Initial scan**: Quick pass for obvious issues
|
||||||
|
2. **Deep analysis**: Line-by-line review of changed code
|
||||||
|
3. **Context evaluation**: Check impact on related code
|
||||||
|
4. **Best practices**: Compare against project and language standards
|
||||||
|
5. **Recommendations**: Prioritized list of improvements
|
||||||
|
|
||||||
|
## Integration with Skills
|
||||||
|
|
||||||
|
Automatically loads `code-standards` skill for project-specific guidelines.
|
||||||
|
|
||||||
|
## Output Format
|
||||||
|
|
||||||
|
For each file reviewed:
|
||||||
|
- Overall assessment
|
||||||
|
- Critical issues (must fix before merge)
|
||||||
|
- Important issues (should fix)
|
||||||
|
- Suggestions (nice to have)
|
||||||
|
- Positive feedback (what was done well)
|
||||||
|
```
|
||||||
|
|
||||||
|
### agents/test-generator.md
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
description: Generates comprehensive test suites from code analysis
|
||||||
|
capabilities:
|
||||||
|
- Analyze code structure and logic flow
|
||||||
|
- Generate unit tests for functions and methods
|
||||||
|
- Create integration tests for modules
|
||||||
|
- Design edge case and error condition tests
|
||||||
|
- Suggest test fixtures and mocks
|
||||||
|
---
|
||||||
|
|
||||||
|
# Test Generator Agent
|
||||||
|
|
||||||
|
Specialized agent for generating comprehensive test suites.
|
||||||
|
|
||||||
|
## Expertise
|
||||||
|
|
||||||
|
- **Unit testing**: Individual function/method tests
|
||||||
|
- **Integration testing**: Module interaction tests
|
||||||
|
- **Edge cases**: Boundary conditions, error paths
|
||||||
|
- **Test organization**: Proper test structure and naming
|
||||||
|
- **Mocking**: Appropriate use of mocks and stubs
|
||||||
|
|
||||||
|
## Generation Process
|
||||||
|
|
||||||
|
1. **Code analysis**: Understand function purpose and logic
|
||||||
|
2. **Path identification**: Map all execution paths
|
||||||
|
3. **Input design**: Create test inputs covering all paths
|
||||||
|
4. **Assertion design**: Define expected outputs
|
||||||
|
5. **Test generation**: Write tests in project's framework
|
||||||
|
|
||||||
|
## Integration with Skills
|
||||||
|
|
||||||
|
Automatically loads `testing-patterns` skill for project-specific test conventions.
|
||||||
|
|
||||||
|
## Test Quality
|
||||||
|
|
||||||
|
Generated tests include:
|
||||||
|
- Happy path scenarios
|
||||||
|
- Edge cases and boundary conditions
|
||||||
|
- Error handling verification
|
||||||
|
- Mock data for external dependencies
|
||||||
|
- Clear test descriptions
|
||||||
|
```
|
||||||
|
|
||||||
|
### skills/code-standards/SKILL.md
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
---
|
||||||
|
name: Code Standards
|
||||||
|
description: This skill should be used when reviewing code, enforcing style guidelines, checking naming conventions, or ensuring code quality standards. Provides project-specific coding standards and best practices.
|
||||||
|
version: 1.0.0
|
||||||
|
---
|
||||||
|
|
||||||
|
# Code Standards
|
||||||
|
|
||||||
|
Comprehensive coding standards and best practices for maintaining code quality.
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
|
||||||
|
Enforce consistent code quality through standardized conventions for:
|
||||||
|
- Code style and formatting
|
||||||
|
- Naming conventions
|
||||||
|
- Documentation requirements
|
||||||
|
- Error handling patterns
|
||||||
|
- Security practices
|
||||||
|
|
||||||
|
## Style Guidelines
|
||||||
|
|
||||||
|
### Formatting
|
||||||
|
|
||||||
|
- **Indentation**: 2 spaces (JavaScript/TypeScript), 4 spaces (Python)
|
||||||
|
- **Line length**: Maximum 100 characters
|
||||||
|
- **Braces**: Same line for opening brace (K&R style)
|
||||||
|
- **Whitespace**: Space after commas, around operators
|
||||||
|
|
||||||
|
### Naming Conventions
|
||||||
|
|
||||||
|
- **Variables**: camelCase for JavaScript, snake_case for Python
|
||||||
|
- **Functions**: camelCase, descriptive verb-noun pairs
|
||||||
|
- **Classes**: PascalCase
|
||||||
|
- **Constants**: UPPER_SNAKE_CASE
|
||||||
|
- **Files**: kebab-case for modules
|
||||||
|
|
||||||
|
## Documentation Requirements
|
||||||
|
|
||||||
|
### Function Documentation
|
||||||
|
|
||||||
|
Every function must include:
|
||||||
|
- Purpose description
|
||||||
|
- Parameter descriptions with types
|
||||||
|
- Return value description with type
|
||||||
|
- Example usage (for public functions)
|
||||||
|
|
||||||
|
### Module Documentation
|
||||||
|
|
||||||
|
Every module must include:
|
||||||
|
- Module purpose
|
||||||
|
- Public API overview
|
||||||
|
- Usage examples
|
||||||
|
- Dependencies
|
||||||
|
|
||||||
|
## Error Handling
|
||||||
|
|
||||||
|
### Required Practices
|
||||||
|
|
||||||
|
- Never swallow errors silently
|
||||||
|
- Always log errors with context
|
||||||
|
- Use specific error types
|
||||||
|
- Provide actionable error messages
|
||||||
|
- Clean up resources in finally blocks
|
||||||
|
|
||||||
|
### Example Pattern
|
||||||
|
|
||||||
|
\`\`\`javascript
|
||||||
|
async function processData(data) {
|
||||||
|
try {
|
||||||
|
const result = await transform(data)
|
||||||
|
return result
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('Data processing failed', {
|
||||||
|
data: sanitize(data),
|
||||||
|
error: error.message,
|
||||||
|
stack: error.stack
|
||||||
|
})
|
||||||
|
throw new DataProcessingError('Failed to process data', { cause: error })
|
||||||
|
}
|
||||||
|
}
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
## Security Practices
|
||||||
|
|
||||||
|
- Validate all external input
|
||||||
|
- Sanitize data before output
|
||||||
|
- Use parameterized queries
|
||||||
|
- Never log sensitive information
|
||||||
|
- Keep dependencies updated
|
||||||
|
|
||||||
|
## Detailed Guidelines
|
||||||
|
|
||||||
|
For comprehensive style guides by language, see:
|
||||||
|
- `references/style-guide.md`
|
||||||
|
```
|
||||||
|
|
||||||
|
### skills/code-standards/references/style-guide.md
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
# Comprehensive Style Guide
|
||||||
|
|
||||||
|
Detailed style guidelines for all supported languages.
|
||||||
|
|
||||||
|
## JavaScript/TypeScript
|
||||||
|
|
||||||
|
### Variable Declarations
|
||||||
|
|
||||||
|
Use `const` by default, `let` when reassignment needed, never `var`:
|
||||||
|
|
||||||
|
\`\`\`javascript
|
||||||
|
// Good
|
||||||
|
const MAX_RETRIES = 3
|
||||||
|
let currentTry = 0
|
||||||
|
|
||||||
|
// Bad
|
||||||
|
var MAX_RETRIES = 3
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
### Function Declarations
|
||||||
|
|
||||||
|
Use function expressions for consistency:
|
||||||
|
|
||||||
|
\`\`\`javascript
|
||||||
|
// Good
|
||||||
|
const calculateTotal = (items) => {
|
||||||
|
return items.reduce((sum, item) => sum + item.price, 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Bad (inconsistent style)
|
||||||
|
function calculateTotal(items) {
|
||||||
|
return items.reduce((sum, item) => sum + item.price, 0)
|
||||||
|
}
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
### Async/Await
|
||||||
|
|
||||||
|
Prefer async/await over promise chains:
|
||||||
|
|
||||||
|
\`\`\`javascript
|
||||||
|
// Good
|
||||||
|
async function fetchUserData(userId) {
|
||||||
|
const user = await db.getUser(userId)
|
||||||
|
const orders = await db.getOrders(user.id)
|
||||||
|
return { user, orders }
|
||||||
|
}
|
||||||
|
|
||||||
|
// Bad
|
||||||
|
function fetchUserData(userId) {
|
||||||
|
return db.getUser(userId)
|
||||||
|
.then(user => db.getOrders(user.id)
|
||||||
|
.then(orders => ({ user, orders })))
|
||||||
|
}
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
## Python
|
||||||
|
|
||||||
|
### Import Organization
|
||||||
|
|
||||||
|
Order imports: standard library, third-party, local:
|
||||||
|
|
||||||
|
\`\`\`python
|
||||||
|
# Good
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
|
|
||||||
|
import numpy as np
|
||||||
|
import pandas as pd
|
||||||
|
|
||||||
|
from app.models import User
|
||||||
|
from app.utils import helper
|
||||||
|
|
||||||
|
# Bad - mixed order
|
||||||
|
from app.models import User
|
||||||
|
import numpy as np
|
||||||
|
import os
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
### Type Hints
|
||||||
|
|
||||||
|
Use type hints for all function signatures:
|
||||||
|
|
||||||
|
\`\`\`python
|
||||||
|
# Good
|
||||||
|
def calculate_average(numbers: list[float]) -> float:
|
||||||
|
return sum(numbers) / len(numbers)
|
||||||
|
|
||||||
|
# Bad
|
||||||
|
def calculate_average(numbers):
|
||||||
|
return sum(numbers) / len(numbers)
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
## Additional Languages
|
||||||
|
|
||||||
|
See language-specific guides for:
|
||||||
|
- Go: `references/go-style.md`
|
||||||
|
- Rust: `references/rust-style.md`
|
||||||
|
- Ruby: `references/ruby-style.md`
|
||||||
|
```
|
||||||
|
|
||||||
|
### hooks/hooks.json
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"PreToolUse": [
|
||||||
|
{
|
||||||
|
"matcher": "Write|Edit",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "prompt",
|
||||||
|
"prompt": "Before modifying code, verify it meets our coding standards from the code-standards skill. Check formatting, naming conventions, and documentation. If standards aren't met, suggest improvements.",
|
||||||
|
"timeout": 30
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"Stop": [
|
||||||
|
{
|
||||||
|
"matcher": ".*",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"command": "bash ${CLAUDE_PLUGIN_ROOT}/hooks/scripts/validate-commit.sh",
|
||||||
|
"timeout": 45
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### hooks/scripts/validate-commit.sh
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
# Validate code quality before task completion
|
||||||
|
|
||||||
|
set -e
|
||||||
|
|
||||||
|
# Check if there are any uncommitted changes
|
||||||
|
if [[ -z $(git status -s) ]]; then
|
||||||
|
echo '{"systemMessage": "No changes to validate. Task complete."}'
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Run linter on changed files
|
||||||
|
CHANGED_FILES=$(git diff --name-only --cached | grep -E '\.(js|ts|py)$' || true)
|
||||||
|
|
||||||
|
if [[ -z "$CHANGED_FILES" ]]; then
|
||||||
|
echo '{"systemMessage": "No code files changed. Validation passed."}'
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Run appropriate linters
|
||||||
|
ISSUES=0
|
||||||
|
|
||||||
|
for file in $CHANGED_FILES; do
|
||||||
|
case "$file" in
|
||||||
|
*.js|*.ts)
|
||||||
|
if ! npx eslint "$file" --quiet; then
|
||||||
|
ISSUES=$((ISSUES + 1))
|
||||||
|
fi
|
||||||
|
;;
|
||||||
|
*.py)
|
||||||
|
if ! python -m pylint "$file" --errors-only; then
|
||||||
|
ISSUES=$((ISSUES + 1))
|
||||||
|
fi
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
done
|
||||||
|
|
||||||
|
if [[ $ISSUES -gt 0 ]]; then
|
||||||
|
echo "{\"systemMessage\": \"Found $ISSUES code quality issues. Please fix before completing.\"}"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo '{"systemMessage": "Code quality checks passed. Ready to commit."}'
|
||||||
|
exit 0
|
||||||
|
```
|
||||||
|
|
||||||
|
## Usage Examples
|
||||||
|
|
||||||
|
### Running Commands
|
||||||
|
|
||||||
|
```
|
||||||
|
$ claude
|
||||||
|
> /lint
|
||||||
|
Running linter checks...
|
||||||
|
|
||||||
|
Critical Issues (2):
|
||||||
|
src/api/users.js:45 - SQL injection vulnerability
|
||||||
|
src/utils/helpers.js:12 - Unhandled promise rejection
|
||||||
|
|
||||||
|
Warnings (5):
|
||||||
|
src/components/Button.tsx:23 - Missing PropTypes
|
||||||
|
...
|
||||||
|
|
||||||
|
Style Suggestions (8):
|
||||||
|
src/index.js:1 - Use const instead of let
|
||||||
|
...
|
||||||
|
|
||||||
|
> /test
|
||||||
|
Running test suite...
|
||||||
|
|
||||||
|
Test Results:
|
||||||
|
✓ 245 passed
|
||||||
|
✗ 3 failed
|
||||||
|
○ 2 skipped
|
||||||
|
|
||||||
|
Coverage: 87.3%
|
||||||
|
|
||||||
|
Untested Files:
|
||||||
|
src/utils/cache.js - 0% coverage
|
||||||
|
src/api/webhooks.js - 23% coverage
|
||||||
|
|
||||||
|
Failed Tests:
|
||||||
|
1. User API › GET /users › should handle pagination
|
||||||
|
Expected 200, received 500
|
||||||
|
...
|
||||||
|
```
|
||||||
|
|
||||||
|
### Using Agents
|
||||||
|
|
||||||
|
```
|
||||||
|
> Review the changes in src/api/users.js
|
||||||
|
|
||||||
|
[code-reviewer agent selected automatically]
|
||||||
|
|
||||||
|
Code Review: src/api/users.js
|
||||||
|
|
||||||
|
Critical Issues:
|
||||||
|
1. Line 45: SQL injection vulnerability
|
||||||
|
- Using string concatenation for SQL query
|
||||||
|
- Replace with parameterized query
|
||||||
|
- Priority: CRITICAL
|
||||||
|
|
||||||
|
2. Line 67: Missing error handling
|
||||||
|
- Database query without try/catch
|
||||||
|
- Could crash server on DB error
|
||||||
|
- Priority: HIGH
|
||||||
|
|
||||||
|
Suggestions:
|
||||||
|
1. Line 23: Consider caching user data
|
||||||
|
- Frequent DB queries for same users
|
||||||
|
- Add Redis caching layer
|
||||||
|
- Priority: MEDIUM
|
||||||
|
```
|
||||||
|
|
||||||
|
## Key Points
|
||||||
|
|
||||||
|
1. **Complete manifest**: All recommended metadata fields
|
||||||
|
2. **Multiple components**: Commands, agents, skills, hooks
|
||||||
|
3. **Rich skills**: References and examples for detailed information
|
||||||
|
4. **Automation**: Hooks enforce standards automatically
|
||||||
|
5. **Integration**: Components work together cohesively
|
||||||
|
|
||||||
|
## When to Use This Pattern
|
||||||
|
|
||||||
|
- Production plugins for distribution
|
||||||
|
- Team collaboration tools
|
||||||
|
- Plugins requiring consistency enforcement
|
||||||
|
- Complex workflows with multiple entry points
|
||||||
567
skills/plugin-structure/references/component-patterns.md
Normal file
567
skills/plugin-structure/references/component-patterns.md
Normal file
@@ -0,0 +1,567 @@
|
|||||||
|
# Component Organization Patterns
|
||||||
|
|
||||||
|
Advanced patterns for organizing plugin components effectively.
|
||||||
|
|
||||||
|
## Component Lifecycle
|
||||||
|
|
||||||
|
### Discovery Phase
|
||||||
|
|
||||||
|
When Claude Code starts:
|
||||||
|
|
||||||
|
1. **Scan enabled plugins**: Read `.claude-plugin/plugin.json` for each
|
||||||
|
2. **Discover components**: Look in default and custom paths
|
||||||
|
3. **Parse definitions**: Read YAML frontmatter and configurations
|
||||||
|
4. **Register components**: Make available to Claude Code
|
||||||
|
5. **Initialize**: Start MCP servers, register hooks
|
||||||
|
|
||||||
|
**Timing**: Component registration happens during Claude Code initialization, not continuously.
|
||||||
|
|
||||||
|
### Activation Phase
|
||||||
|
|
||||||
|
When components are used:
|
||||||
|
|
||||||
|
**Commands**: User types slash command → Claude Code looks up → Executes
|
||||||
|
**Agents**: Task arrives → Claude Code evaluates capabilities → Selects agent
|
||||||
|
**Skills**: Task context matches description → Claude Code loads skill
|
||||||
|
**Hooks**: Event occurs → Claude Code calls matching hooks
|
||||||
|
**MCP Servers**: Tool call matches server capability → Forwards to server
|
||||||
|
|
||||||
|
## Command Organization Patterns
|
||||||
|
|
||||||
|
### Flat Structure
|
||||||
|
|
||||||
|
Single directory with all commands:
|
||||||
|
|
||||||
|
```
|
||||||
|
commands/
|
||||||
|
├── build.md
|
||||||
|
├── test.md
|
||||||
|
├── deploy.md
|
||||||
|
├── review.md
|
||||||
|
└── docs.md
|
||||||
|
```
|
||||||
|
|
||||||
|
**When to use**:
|
||||||
|
- 5-15 commands total
|
||||||
|
- All commands at same abstraction level
|
||||||
|
- No clear categorization
|
||||||
|
|
||||||
|
**Advantages**:
|
||||||
|
- Simple, easy to navigate
|
||||||
|
- No configuration needed
|
||||||
|
- Fast discovery
|
||||||
|
|
||||||
|
### Categorized Structure
|
||||||
|
|
||||||
|
Multiple directories for different command types:
|
||||||
|
|
||||||
|
```
|
||||||
|
commands/ # Core commands
|
||||||
|
├── build.md
|
||||||
|
└── test.md
|
||||||
|
|
||||||
|
admin-commands/ # Administrative
|
||||||
|
├── configure.md
|
||||||
|
└── manage.md
|
||||||
|
|
||||||
|
workflow-commands/ # Workflow automation
|
||||||
|
├── review.md
|
||||||
|
└── deploy.md
|
||||||
|
```
|
||||||
|
|
||||||
|
**Manifest configuration**:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"commands": [
|
||||||
|
"./commands",
|
||||||
|
"./admin-commands",
|
||||||
|
"./workflow-commands"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**When to use**:
|
||||||
|
- 15+ commands
|
||||||
|
- Clear functional categories
|
||||||
|
- Different permission levels
|
||||||
|
|
||||||
|
**Advantages**:
|
||||||
|
- Organized by purpose
|
||||||
|
- Easier to maintain
|
||||||
|
- Can restrict access by directory
|
||||||
|
|
||||||
|
### Hierarchical Structure
|
||||||
|
|
||||||
|
Nested organization for complex plugins:
|
||||||
|
|
||||||
|
```
|
||||||
|
commands/
|
||||||
|
├── ci/
|
||||||
|
│ ├── build.md
|
||||||
|
│ ├── test.md
|
||||||
|
│ └── lint.md
|
||||||
|
├── deployment/
|
||||||
|
│ ├── staging.md
|
||||||
|
│ └── production.md
|
||||||
|
└── management/
|
||||||
|
├── config.md
|
||||||
|
└── status.md
|
||||||
|
```
|
||||||
|
|
||||||
|
**Note**: Claude Code doesn't support nested command discovery automatically. Use custom paths:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"commands": [
|
||||||
|
"./commands/ci",
|
||||||
|
"./commands/deployment",
|
||||||
|
"./commands/management"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**When to use**:
|
||||||
|
- 20+ commands
|
||||||
|
- Multi-level categorization
|
||||||
|
- Complex workflows
|
||||||
|
|
||||||
|
**Advantages**:
|
||||||
|
- Maximum organization
|
||||||
|
- Clear boundaries
|
||||||
|
- Scalable structure
|
||||||
|
|
||||||
|
## Agent Organization Patterns
|
||||||
|
|
||||||
|
### Role-Based Organization
|
||||||
|
|
||||||
|
Organize agents by their primary role:
|
||||||
|
|
||||||
|
```
|
||||||
|
agents/
|
||||||
|
├── code-reviewer.md # Reviews code
|
||||||
|
├── test-generator.md # Generates tests
|
||||||
|
├── documentation-writer.md # Writes docs
|
||||||
|
└── refactorer.md # Refactors code
|
||||||
|
```
|
||||||
|
|
||||||
|
**When to use**:
|
||||||
|
- Agents have distinct, non-overlapping roles
|
||||||
|
- Users invoke agents manually
|
||||||
|
- Clear agent responsibilities
|
||||||
|
|
||||||
|
### Capability-Based Organization
|
||||||
|
|
||||||
|
Organize by specific capabilities:
|
||||||
|
|
||||||
|
```
|
||||||
|
agents/
|
||||||
|
├── python-expert.md # Python-specific
|
||||||
|
├── typescript-expert.md # TypeScript-specific
|
||||||
|
├── api-specialist.md # API design
|
||||||
|
└── database-specialist.md # Database work
|
||||||
|
```
|
||||||
|
|
||||||
|
**When to use**:
|
||||||
|
- Technology-specific agents
|
||||||
|
- Domain expertise focus
|
||||||
|
- Automatic agent selection
|
||||||
|
|
||||||
|
### Workflow-Based Organization
|
||||||
|
|
||||||
|
Organize by workflow stage:
|
||||||
|
|
||||||
|
```
|
||||||
|
agents/
|
||||||
|
├── planning-agent.md # Planning phase
|
||||||
|
├── implementation-agent.md # Coding phase
|
||||||
|
├── testing-agent.md # Testing phase
|
||||||
|
└── deployment-agent.md # Deployment phase
|
||||||
|
```
|
||||||
|
|
||||||
|
**When to use**:
|
||||||
|
- Sequential workflows
|
||||||
|
- Stage-specific expertise
|
||||||
|
- Pipeline automation
|
||||||
|
|
||||||
|
## Skill Organization Patterns
|
||||||
|
|
||||||
|
### Topic-Based Organization
|
||||||
|
|
||||||
|
Each skill covers a specific topic:
|
||||||
|
|
||||||
|
```
|
||||||
|
skills/
|
||||||
|
├── api-design/
|
||||||
|
│ └── SKILL.md
|
||||||
|
├── error-handling/
|
||||||
|
│ └── SKILL.md
|
||||||
|
├── testing-strategies/
|
||||||
|
│ └── SKILL.md
|
||||||
|
└── performance-optimization/
|
||||||
|
└── SKILL.md
|
||||||
|
```
|
||||||
|
|
||||||
|
**When to use**:
|
||||||
|
- Knowledge-based skills
|
||||||
|
- Educational or reference content
|
||||||
|
- Broad applicability
|
||||||
|
|
||||||
|
### Tool-Based Organization
|
||||||
|
|
||||||
|
Skills for specific tools or technologies:
|
||||||
|
|
||||||
|
```
|
||||||
|
skills/
|
||||||
|
├── docker/
|
||||||
|
│ ├── SKILL.md
|
||||||
|
│ └── references/
|
||||||
|
│ └── dockerfile-best-practices.md
|
||||||
|
├── kubernetes/
|
||||||
|
│ ├── SKILL.md
|
||||||
|
│ └── examples/
|
||||||
|
│ └── deployment.yaml
|
||||||
|
└── terraform/
|
||||||
|
├── SKILL.md
|
||||||
|
└── scripts/
|
||||||
|
└── validate-config.sh
|
||||||
|
```
|
||||||
|
|
||||||
|
**When to use**:
|
||||||
|
- Tool-specific expertise
|
||||||
|
- Complex tool configurations
|
||||||
|
- Tool best practices
|
||||||
|
|
||||||
|
### Workflow-Based Organization
|
||||||
|
|
||||||
|
Skills for complete workflows:
|
||||||
|
|
||||||
|
```
|
||||||
|
skills/
|
||||||
|
├── code-review-workflow/
|
||||||
|
│ ├── SKILL.md
|
||||||
|
│ └── references/
|
||||||
|
│ ├── checklist.md
|
||||||
|
│ └── standards.md
|
||||||
|
├── deployment-workflow/
|
||||||
|
│ ├── SKILL.md
|
||||||
|
│ └── scripts/
|
||||||
|
│ ├── pre-deploy.sh
|
||||||
|
│ └── post-deploy.sh
|
||||||
|
└── testing-workflow/
|
||||||
|
├── SKILL.md
|
||||||
|
└── examples/
|
||||||
|
└── test-structure.md
|
||||||
|
```
|
||||||
|
|
||||||
|
**When to use**:
|
||||||
|
- Multi-step processes
|
||||||
|
- Company-specific workflows
|
||||||
|
- Process automation
|
||||||
|
|
||||||
|
### Skill with Rich Resources
|
||||||
|
|
||||||
|
Comprehensive skill with all resource types:
|
||||||
|
|
||||||
|
```
|
||||||
|
skills/
|
||||||
|
└── api-testing/
|
||||||
|
├── SKILL.md # Core skill (1500 words)
|
||||||
|
├── references/
|
||||||
|
│ ├── rest-api-guide.md
|
||||||
|
│ ├── graphql-guide.md
|
||||||
|
│ └── authentication.md
|
||||||
|
├── examples/
|
||||||
|
│ ├── basic-test.js
|
||||||
|
│ ├── authenticated-test.js
|
||||||
|
│ └── integration-test.js
|
||||||
|
├── scripts/
|
||||||
|
│ ├── run-tests.sh
|
||||||
|
│ └── generate-report.py
|
||||||
|
└── assets/
|
||||||
|
└── test-template.json
|
||||||
|
```
|
||||||
|
|
||||||
|
**Resource usage**:
|
||||||
|
- **SKILL.md**: Overview and when to use resources
|
||||||
|
- **references/**: Detailed guides (loaded as needed)
|
||||||
|
- **examples/**: Copy-paste code samples
|
||||||
|
- **scripts/**: Executable test runners
|
||||||
|
- **assets/**: Templates and configurations
|
||||||
|
|
||||||
|
## Hook Organization Patterns
|
||||||
|
|
||||||
|
### Monolithic Configuration
|
||||||
|
|
||||||
|
Single hooks.json with all hooks:
|
||||||
|
|
||||||
|
```
|
||||||
|
hooks/
|
||||||
|
├── hooks.json # All hook definitions
|
||||||
|
└── scripts/
|
||||||
|
├── validate-write.sh
|
||||||
|
├── validate-bash.sh
|
||||||
|
└── load-context.sh
|
||||||
|
```
|
||||||
|
|
||||||
|
**hooks.json**:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"PreToolUse": [...],
|
||||||
|
"PostToolUse": [...],
|
||||||
|
"Stop": [...],
|
||||||
|
"SessionStart": [...]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**When to use**:
|
||||||
|
- 5-10 hooks total
|
||||||
|
- Simple hook logic
|
||||||
|
- Centralized configuration
|
||||||
|
|
||||||
|
### Event-Based Organization
|
||||||
|
|
||||||
|
Separate files per event type:
|
||||||
|
|
||||||
|
```
|
||||||
|
hooks/
|
||||||
|
├── hooks.json # Combines all
|
||||||
|
├── pre-tool-use.json # PreToolUse hooks
|
||||||
|
├── post-tool-use.json # PostToolUse hooks
|
||||||
|
├── stop.json # Stop hooks
|
||||||
|
└── scripts/
|
||||||
|
├── validate/
|
||||||
|
│ ├── write.sh
|
||||||
|
│ └── bash.sh
|
||||||
|
└── context/
|
||||||
|
└── load.sh
|
||||||
|
```
|
||||||
|
|
||||||
|
**hooks.json** (combines):
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"PreToolUse": ${file:./pre-tool-use.json},
|
||||||
|
"PostToolUse": ${file:./post-tool-use.json},
|
||||||
|
"Stop": ${file:./stop.json}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Note**: Use build script to combine files, Claude Code doesn't support file references.
|
||||||
|
|
||||||
|
**When to use**:
|
||||||
|
- 10+ hooks
|
||||||
|
- Different teams managing different events
|
||||||
|
- Complex hook configurations
|
||||||
|
|
||||||
|
### Purpose-Based Organization
|
||||||
|
|
||||||
|
Group by functional purpose:
|
||||||
|
|
||||||
|
```
|
||||||
|
hooks/
|
||||||
|
├── hooks.json
|
||||||
|
└── scripts/
|
||||||
|
├── security/
|
||||||
|
│ ├── validate-paths.sh
|
||||||
|
│ ├── check-credentials.sh
|
||||||
|
│ └── scan-malware.sh
|
||||||
|
├── quality/
|
||||||
|
│ ├── lint-code.sh
|
||||||
|
│ ├── check-tests.sh
|
||||||
|
│ └── verify-docs.sh
|
||||||
|
└── workflow/
|
||||||
|
├── notify-team.sh
|
||||||
|
└── update-status.sh
|
||||||
|
```
|
||||||
|
|
||||||
|
**When to use**:
|
||||||
|
- Many hook scripts
|
||||||
|
- Clear functional boundaries
|
||||||
|
- Team specialization
|
||||||
|
|
||||||
|
## Script Organization Patterns
|
||||||
|
|
||||||
|
### Flat Scripts
|
||||||
|
|
||||||
|
All scripts in single directory:
|
||||||
|
|
||||||
|
```
|
||||||
|
scripts/
|
||||||
|
├── build.sh
|
||||||
|
├── test.py
|
||||||
|
├── deploy.sh
|
||||||
|
├── validate.js
|
||||||
|
└── report.py
|
||||||
|
```
|
||||||
|
|
||||||
|
**When to use**:
|
||||||
|
- 5-10 scripts
|
||||||
|
- All scripts related
|
||||||
|
- Simple plugin
|
||||||
|
|
||||||
|
### Categorized Scripts
|
||||||
|
|
||||||
|
Group by purpose:
|
||||||
|
|
||||||
|
```
|
||||||
|
scripts/
|
||||||
|
├── build/
|
||||||
|
│ ├── compile.sh
|
||||||
|
│ └── package.sh
|
||||||
|
├── test/
|
||||||
|
│ ├── run-unit.sh
|
||||||
|
│ └── run-integration.sh
|
||||||
|
├── deploy/
|
||||||
|
│ ├── staging.sh
|
||||||
|
│ └── production.sh
|
||||||
|
└── utils/
|
||||||
|
├── log.sh
|
||||||
|
└── notify.sh
|
||||||
|
```
|
||||||
|
|
||||||
|
**When to use**:
|
||||||
|
- 10+ scripts
|
||||||
|
- Clear categories
|
||||||
|
- Reusable utilities
|
||||||
|
|
||||||
|
### Language-Based Organization
|
||||||
|
|
||||||
|
Group by programming language:
|
||||||
|
|
||||||
|
```
|
||||||
|
scripts/
|
||||||
|
├── bash/
|
||||||
|
│ ├── build.sh
|
||||||
|
│ └── deploy.sh
|
||||||
|
├── python/
|
||||||
|
│ ├── analyze.py
|
||||||
|
│ └── report.py
|
||||||
|
└── javascript/
|
||||||
|
├── bundle.js
|
||||||
|
└── optimize.js
|
||||||
|
```
|
||||||
|
|
||||||
|
**When to use**:
|
||||||
|
- Multi-language scripts
|
||||||
|
- Different runtime requirements
|
||||||
|
- Language-specific dependencies
|
||||||
|
|
||||||
|
## Cross-Component Patterns
|
||||||
|
|
||||||
|
### Shared Resources
|
||||||
|
|
||||||
|
Components sharing common resources:
|
||||||
|
|
||||||
|
```
|
||||||
|
plugin/
|
||||||
|
├── commands/
|
||||||
|
│ ├── test.md # Uses lib/test-utils.sh
|
||||||
|
│ └── deploy.md # Uses lib/deploy-utils.sh
|
||||||
|
├── agents/
|
||||||
|
│ └── tester.md # References lib/test-utils.sh
|
||||||
|
├── hooks/
|
||||||
|
│ └── scripts/
|
||||||
|
│ └── pre-test.sh # Sources lib/test-utils.sh
|
||||||
|
└── lib/
|
||||||
|
├── test-utils.sh
|
||||||
|
└── deploy-utils.sh
|
||||||
|
```
|
||||||
|
|
||||||
|
**Usage in components**:
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
source "${CLAUDE_PLUGIN_ROOT}/lib/test-utils.sh"
|
||||||
|
run_tests
|
||||||
|
```
|
||||||
|
|
||||||
|
**Benefits**:
|
||||||
|
- Code reuse
|
||||||
|
- Consistent behavior
|
||||||
|
- Easier maintenance
|
||||||
|
|
||||||
|
### Layered Architecture
|
||||||
|
|
||||||
|
Separate concerns into layers:
|
||||||
|
|
||||||
|
```
|
||||||
|
plugin/
|
||||||
|
├── commands/ # User interface layer
|
||||||
|
├── agents/ # Orchestration layer
|
||||||
|
├── skills/ # Knowledge layer
|
||||||
|
└── lib/
|
||||||
|
├── core/ # Core business logic
|
||||||
|
├── integrations/ # External services
|
||||||
|
└── utils/ # Helper functions
|
||||||
|
```
|
||||||
|
|
||||||
|
**When to use**:
|
||||||
|
- Large plugins (100+ files)
|
||||||
|
- Multiple developers
|
||||||
|
- Clear separation of concerns
|
||||||
|
|
||||||
|
### Plugin Within Plugin
|
||||||
|
|
||||||
|
Nested plugin structure:
|
||||||
|
|
||||||
|
```
|
||||||
|
plugin/
|
||||||
|
├── .claude-plugin/
|
||||||
|
│ └── plugin.json
|
||||||
|
├── core/ # Core functionality
|
||||||
|
│ ├── commands/
|
||||||
|
│ └── agents/
|
||||||
|
└── extensions/ # Optional extensions
|
||||||
|
├── extension-a/
|
||||||
|
│ ├── commands/
|
||||||
|
│ └── agents/
|
||||||
|
└── extension-b/
|
||||||
|
├── commands/
|
||||||
|
└── agents/
|
||||||
|
```
|
||||||
|
|
||||||
|
**Manifest**:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"commands": [
|
||||||
|
"./core/commands",
|
||||||
|
"./extensions/extension-a/commands",
|
||||||
|
"./extensions/extension-b/commands"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**When to use**:
|
||||||
|
- Modular functionality
|
||||||
|
- Optional features
|
||||||
|
- Plugin families
|
||||||
|
|
||||||
|
## Best Practices
|
||||||
|
|
||||||
|
### Naming
|
||||||
|
|
||||||
|
1. **Consistent naming**: Match file names to component purpose
|
||||||
|
2. **Descriptive names**: Indicate what component does
|
||||||
|
3. **Avoid abbreviations**: Use full words for clarity
|
||||||
|
|
||||||
|
### Organization
|
||||||
|
|
||||||
|
1. **Start simple**: Use flat structure, reorganize when needed
|
||||||
|
2. **Group related items**: Keep related components together
|
||||||
|
3. **Separate concerns**: Don't mix unrelated functionality
|
||||||
|
|
||||||
|
### Scalability
|
||||||
|
|
||||||
|
1. **Plan for growth**: Choose structure that scales
|
||||||
|
2. **Refactor early**: Reorganize before it becomes painful
|
||||||
|
3. **Document structure**: Explain organization in README
|
||||||
|
|
||||||
|
### Maintainability
|
||||||
|
|
||||||
|
1. **Consistent patterns**: Use same structure throughout
|
||||||
|
2. **Minimize nesting**: Keep directory depth manageable
|
||||||
|
3. **Use conventions**: Follow community standards
|
||||||
|
|
||||||
|
### Performance
|
||||||
|
|
||||||
|
1. **Avoid deep nesting**: Impacts discovery time
|
||||||
|
2. **Minimize custom paths**: Use defaults when possible
|
||||||
|
3. **Keep configurations small**: Large configs slow loading
|
||||||
552
skills/plugin-structure/references/manifest-reference.md
Normal file
552
skills/plugin-structure/references/manifest-reference.md
Normal file
@@ -0,0 +1,552 @@
|
|||||||
|
# Plugin Manifest Reference
|
||||||
|
|
||||||
|
Complete reference for `plugin.json` configuration.
|
||||||
|
|
||||||
|
## File Location
|
||||||
|
|
||||||
|
**Required path**: `.claude-plugin/plugin.json`
|
||||||
|
|
||||||
|
The manifest MUST be in the `.claude-plugin/` directory at the plugin root. Claude Code will not recognize plugins without this file in the correct location.
|
||||||
|
|
||||||
|
## Complete Field Reference
|
||||||
|
|
||||||
|
### Core Fields
|
||||||
|
|
||||||
|
#### name (required)
|
||||||
|
|
||||||
|
**Type**: String
|
||||||
|
**Format**: kebab-case
|
||||||
|
**Example**: `"test-automation-suite"`
|
||||||
|
|
||||||
|
The unique identifier for the plugin. Used for:
|
||||||
|
- Plugin identification in Claude Code
|
||||||
|
- Conflict detection with other plugins
|
||||||
|
- Command namespacing (optional)
|
||||||
|
|
||||||
|
**Requirements**:
|
||||||
|
- Must be unique across all installed plugins
|
||||||
|
- Use only lowercase letters, numbers, and hyphens
|
||||||
|
- No spaces or special characters
|
||||||
|
- Start with a letter
|
||||||
|
- End with a letter or number
|
||||||
|
|
||||||
|
**Validation**:
|
||||||
|
```javascript
|
||||||
|
/^[a-z][a-z0-9]*(-[a-z0-9]+)*$/
|
||||||
|
```
|
||||||
|
|
||||||
|
**Examples**:
|
||||||
|
- ✅ Good: `api-tester`, `code-review`, `git-workflow-automation`
|
||||||
|
- ❌ Bad: `API Tester`, `code_review`, `-git-workflow`, `test-`
|
||||||
|
|
||||||
|
#### version
|
||||||
|
|
||||||
|
**Type**: String
|
||||||
|
**Format**: Semantic versioning (MAJOR.MINOR.PATCH)
|
||||||
|
**Example**: `"2.1.0"`
|
||||||
|
**Default**: `"0.1.0"` if not specified
|
||||||
|
|
||||||
|
Semantic versioning guidelines:
|
||||||
|
- **MAJOR**: Incompatible API changes, breaking changes
|
||||||
|
- **MINOR**: New functionality, backward-compatible
|
||||||
|
- **PATCH**: Bug fixes, backward-compatible
|
||||||
|
|
||||||
|
**Pre-release versions**:
|
||||||
|
- `"1.0.0-alpha.1"` - Alpha release
|
||||||
|
- `"1.0.0-beta.2"` - Beta release
|
||||||
|
- `"1.0.0-rc.1"` - Release candidate
|
||||||
|
|
||||||
|
**Examples**:
|
||||||
|
- `"0.1.0"` - Initial development
|
||||||
|
- `"1.0.0"` - First stable release
|
||||||
|
- `"1.2.3"` - Patch update to 1.2
|
||||||
|
- `"2.0.0"` - Major version with breaking changes
|
||||||
|
|
||||||
|
#### description
|
||||||
|
|
||||||
|
**Type**: String
|
||||||
|
**Length**: 50-200 characters recommended
|
||||||
|
**Example**: `"Automates code review workflows with style checks and automated feedback"`
|
||||||
|
|
||||||
|
Brief explanation of plugin purpose and functionality.
|
||||||
|
|
||||||
|
**Best practices**:
|
||||||
|
- Focus on what the plugin does, not how
|
||||||
|
- Use active voice
|
||||||
|
- Mention key features or benefits
|
||||||
|
- Keep under 200 characters for marketplace display
|
||||||
|
|
||||||
|
**Examples**:
|
||||||
|
- ✅ "Generates comprehensive test suites from code analysis and coverage reports"
|
||||||
|
- ✅ "Integrates with Jira for automatic issue tracking and sprint management"
|
||||||
|
- ❌ "A plugin that helps you do testing stuff"
|
||||||
|
- ❌ "This is a very long description that goes on and on about every single feature..."
|
||||||
|
|
||||||
|
### Metadata Fields
|
||||||
|
|
||||||
|
#### author
|
||||||
|
|
||||||
|
**Type**: Object
|
||||||
|
**Fields**: name (required), email (optional), url (optional)
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"author": {
|
||||||
|
"name": "Jane Developer",
|
||||||
|
"email": "jane@example.com",
|
||||||
|
"url": "https://janedeveloper.com"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Alternative format** (string only):
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"author": "Jane Developer <jane@example.com> (https://janedeveloper.com)"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use cases**:
|
||||||
|
- Credit and attribution
|
||||||
|
- Contact for support or questions
|
||||||
|
- Marketplace display
|
||||||
|
- Community recognition
|
||||||
|
|
||||||
|
#### homepage
|
||||||
|
|
||||||
|
**Type**: String (URL)
|
||||||
|
**Example**: `"https://docs.example.com/plugins/my-plugin"`
|
||||||
|
|
||||||
|
Link to plugin documentation or landing page.
|
||||||
|
|
||||||
|
**Should point to**:
|
||||||
|
- Plugin documentation site
|
||||||
|
- Project homepage
|
||||||
|
- Detailed usage guide
|
||||||
|
- Installation instructions
|
||||||
|
|
||||||
|
**Not for**:
|
||||||
|
- Source code (use `repository` field)
|
||||||
|
- Issue tracker (include in documentation)
|
||||||
|
- Personal websites (use `author.url`)
|
||||||
|
|
||||||
|
#### repository
|
||||||
|
|
||||||
|
**Type**: String (URL) or Object
|
||||||
|
**Example**: `"https://github.com/user/plugin-name"`
|
||||||
|
|
||||||
|
Source code repository location.
|
||||||
|
|
||||||
|
**String format**:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"repository": "https://github.com/user/plugin-name"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Object format** (detailed):
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"repository": {
|
||||||
|
"type": "git",
|
||||||
|
"url": "https://github.com/user/plugin-name.git",
|
||||||
|
"directory": "packages/plugin-name"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use cases**:
|
||||||
|
- Source code access
|
||||||
|
- Issue reporting
|
||||||
|
- Community contributions
|
||||||
|
- Transparency and trust
|
||||||
|
|
||||||
|
#### license
|
||||||
|
|
||||||
|
**Type**: String
|
||||||
|
**Format**: SPDX identifier
|
||||||
|
**Example**: `"MIT"`
|
||||||
|
|
||||||
|
Software license identifier.
|
||||||
|
|
||||||
|
**Common licenses**:
|
||||||
|
- `"MIT"` - Permissive, popular choice
|
||||||
|
- `"Apache-2.0"` - Permissive with patent grant
|
||||||
|
- `"GPL-3.0"` - Copyleft
|
||||||
|
- `"BSD-3-Clause"` - Permissive
|
||||||
|
- `"ISC"` - Permissive, similar to MIT
|
||||||
|
- `"UNLICENSED"` - Proprietary, not open source
|
||||||
|
|
||||||
|
**Full list**: https://spdx.org/licenses/
|
||||||
|
|
||||||
|
**Multiple licenses**:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"license": "(MIT OR Apache-2.0)"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
#### keywords
|
||||||
|
|
||||||
|
**Type**: Array of strings
|
||||||
|
**Example**: `["testing", "automation", "ci-cd", "quality-assurance"]`
|
||||||
|
|
||||||
|
Tags for plugin discovery and categorization.
|
||||||
|
|
||||||
|
**Best practices**:
|
||||||
|
- Use 5-10 keywords
|
||||||
|
- Include functionality categories
|
||||||
|
- Add technology names
|
||||||
|
- Use common search terms
|
||||||
|
- Avoid duplicating plugin name
|
||||||
|
|
||||||
|
**Categories to consider**:
|
||||||
|
- Functionality: `testing`, `debugging`, `documentation`, `deployment`
|
||||||
|
- Technologies: `typescript`, `python`, `docker`, `aws`
|
||||||
|
- Workflows: `ci-cd`, `code-review`, `git-workflow`
|
||||||
|
- Domains: `web-development`, `data-science`, `devops`
|
||||||
|
|
||||||
|
### Component Path Fields
|
||||||
|
|
||||||
|
#### commands
|
||||||
|
|
||||||
|
**Type**: String or Array of strings
|
||||||
|
**Default**: `["./commands"]`
|
||||||
|
**Example**: `"./cli-commands"`
|
||||||
|
|
||||||
|
Additional directories or files containing command definitions.
|
||||||
|
|
||||||
|
**Single path**:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"commands": "./custom-commands"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Multiple paths**:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"commands": [
|
||||||
|
"./commands",
|
||||||
|
"./admin-commands",
|
||||||
|
"./experimental-commands"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Behavior**: Supplements default `commands/` directory (does not replace)
|
||||||
|
|
||||||
|
**Use cases**:
|
||||||
|
- Organizing commands by category
|
||||||
|
- Separating stable from experimental commands
|
||||||
|
- Loading commands from shared locations
|
||||||
|
|
||||||
|
#### agents
|
||||||
|
|
||||||
|
**Type**: String or Array of strings
|
||||||
|
**Default**: `["./agents"]`
|
||||||
|
**Example**: `"./specialized-agents"`
|
||||||
|
|
||||||
|
Additional directories or files containing agent definitions.
|
||||||
|
|
||||||
|
**Format**: Same as `commands` field
|
||||||
|
|
||||||
|
**Use cases**:
|
||||||
|
- Grouping agents by specialization
|
||||||
|
- Separating general-purpose from task-specific agents
|
||||||
|
- Loading agents from plugin dependencies
|
||||||
|
|
||||||
|
#### hooks
|
||||||
|
|
||||||
|
**Type**: String (path to JSON file) or Object (inline configuration)
|
||||||
|
**Default**: `"./hooks/hooks.json"`
|
||||||
|
|
||||||
|
Hook configuration location or inline definition.
|
||||||
|
|
||||||
|
**File path**:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"hooks": "./config/hooks.json"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Inline configuration**:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"hooks": {
|
||||||
|
"PreToolUse": [
|
||||||
|
{
|
||||||
|
"matcher": "Write",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"command": "bash ${CLAUDE_PLUGIN_ROOT}/scripts/validate.sh",
|
||||||
|
"timeout": 30
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use cases**:
|
||||||
|
- Simple plugins: Inline configuration (< 50 lines)
|
||||||
|
- Complex plugins: External JSON file
|
||||||
|
- Multiple hook sets: Separate files for different contexts
|
||||||
|
|
||||||
|
#### mcpServers
|
||||||
|
|
||||||
|
**Type**: String (path to JSON file) or Object (inline configuration)
|
||||||
|
**Default**: `./.mcp.json`
|
||||||
|
|
||||||
|
MCP server configuration location or inline definition.
|
||||||
|
|
||||||
|
**File path**:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"mcpServers": "./.mcp.json"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Inline configuration**:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"mcpServers": {
|
||||||
|
"github": {
|
||||||
|
"command": "node",
|
||||||
|
"args": ["${CLAUDE_PLUGIN_ROOT}/servers/github-mcp.js"],
|
||||||
|
"env": {
|
||||||
|
"GITHUB_TOKEN": "${GITHUB_TOKEN}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Use cases**:
|
||||||
|
- Simple plugins: Single inline server (< 20 lines)
|
||||||
|
- Complex plugins: External `.mcp.json` file
|
||||||
|
- Multiple servers: Always use external file
|
||||||
|
|
||||||
|
## Path Resolution
|
||||||
|
|
||||||
|
### Relative Path Rules
|
||||||
|
|
||||||
|
All paths in component fields must follow these rules:
|
||||||
|
|
||||||
|
1. **Must be relative**: No absolute paths
|
||||||
|
2. **Must start with `./`**: Indicates relative to plugin root
|
||||||
|
3. **Cannot use `../`**: No parent directory navigation
|
||||||
|
4. **Forward slashes only**: Even on Windows
|
||||||
|
|
||||||
|
**Examples**:
|
||||||
|
- ✅ `"./commands"`
|
||||||
|
- ✅ `"./src/commands"`
|
||||||
|
- ✅ `"./configs/hooks.json"`
|
||||||
|
- ❌ `"/Users/name/plugin/commands"`
|
||||||
|
- ❌ `"commands"` (missing `./`)
|
||||||
|
- ❌ `"../shared/commands"`
|
||||||
|
- ❌ `".\\commands"` (backslash)
|
||||||
|
|
||||||
|
### Resolution Order
|
||||||
|
|
||||||
|
When Claude Code loads components:
|
||||||
|
|
||||||
|
1. **Default directories**: Scans standard locations first
|
||||||
|
- `./commands/`
|
||||||
|
- `./agents/`
|
||||||
|
- `./skills/`
|
||||||
|
- `./hooks/hooks.json`
|
||||||
|
- `./.mcp.json`
|
||||||
|
|
||||||
|
2. **Custom paths**: Scans paths specified in manifest
|
||||||
|
- Paths from `commands` field
|
||||||
|
- Paths from `agents` field
|
||||||
|
- Files from `hooks` and `mcpServers` fields
|
||||||
|
|
||||||
|
3. **Merge behavior**: Components from all locations load
|
||||||
|
- No overwriting
|
||||||
|
- All discovered components register
|
||||||
|
- Name conflicts cause errors
|
||||||
|
|
||||||
|
## Validation
|
||||||
|
|
||||||
|
### Manifest Validation
|
||||||
|
|
||||||
|
Claude Code validates the manifest on plugin load:
|
||||||
|
|
||||||
|
**Syntax validation**:
|
||||||
|
- Valid JSON format
|
||||||
|
- No syntax errors
|
||||||
|
- Correct field types
|
||||||
|
|
||||||
|
**Field validation**:
|
||||||
|
- `name` field present and valid format
|
||||||
|
- `version` follows semantic versioning (if present)
|
||||||
|
- Paths are relative with `./` prefix
|
||||||
|
- URLs are valid (if present)
|
||||||
|
|
||||||
|
**Component validation**:
|
||||||
|
- Referenced paths exist
|
||||||
|
- Hook and MCP configurations are valid
|
||||||
|
- No circular dependencies
|
||||||
|
|
||||||
|
### Common Validation Errors
|
||||||
|
|
||||||
|
**Invalid name format**:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"name": "My Plugin" // ❌ Contains spaces
|
||||||
|
}
|
||||||
|
```
|
||||||
|
Fix: Use kebab-case
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"name": "my-plugin" // ✅
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Absolute path**:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"commands": "/Users/name/commands" // ❌ Absolute path
|
||||||
|
}
|
||||||
|
```
|
||||||
|
Fix: Use relative path
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"commands": "./commands" // ✅
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Missing ./ prefix**:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"hooks": "hooks/hooks.json" // ❌ No ./
|
||||||
|
}
|
||||||
|
```
|
||||||
|
Fix: Add ./ prefix
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"hooks": "./hooks/hooks.json" // ✅
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Invalid version**:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"version": "1.0" // ❌ Not semantic versioning
|
||||||
|
}
|
||||||
|
```
|
||||||
|
Fix: Use MAJOR.MINOR.PATCH
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"version": "1.0.0" // ✅
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## Minimal vs. Complete Examples
|
||||||
|
|
||||||
|
### Minimal Plugin
|
||||||
|
|
||||||
|
Bare minimum for a working plugin:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"name": "hello-world"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Relies entirely on default directory discovery.
|
||||||
|
|
||||||
|
### Recommended Plugin
|
||||||
|
|
||||||
|
Good metadata for distribution:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"name": "code-review-assistant",
|
||||||
|
"version": "1.0.0",
|
||||||
|
"description": "Automates code review with style checks and suggestions",
|
||||||
|
"author": {
|
||||||
|
"name": "Jane Developer",
|
||||||
|
"email": "jane@example.com"
|
||||||
|
},
|
||||||
|
"homepage": "https://docs.example.com/code-review",
|
||||||
|
"repository": "https://github.com/janedev/code-review-assistant",
|
||||||
|
"license": "MIT",
|
||||||
|
"keywords": ["code-review", "automation", "quality", "ci-cd"]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Complete Plugin
|
||||||
|
|
||||||
|
Full configuration with all features:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"name": "enterprise-devops",
|
||||||
|
"version": "2.3.1",
|
||||||
|
"description": "Comprehensive DevOps automation for enterprise CI/CD pipelines",
|
||||||
|
"author": {
|
||||||
|
"name": "DevOps Team",
|
||||||
|
"email": "devops@company.com",
|
||||||
|
"url": "https://company.com/devops"
|
||||||
|
},
|
||||||
|
"homepage": "https://docs.company.com/plugins/devops",
|
||||||
|
"repository": {
|
||||||
|
"type": "git",
|
||||||
|
"url": "https://github.com/company/devops-plugin.git"
|
||||||
|
},
|
||||||
|
"license": "Apache-2.0",
|
||||||
|
"keywords": [
|
||||||
|
"devops",
|
||||||
|
"ci-cd",
|
||||||
|
"automation",
|
||||||
|
"kubernetes",
|
||||||
|
"docker",
|
||||||
|
"deployment"
|
||||||
|
],
|
||||||
|
"commands": [
|
||||||
|
"./commands",
|
||||||
|
"./admin-commands"
|
||||||
|
],
|
||||||
|
"agents": "./specialized-agents",
|
||||||
|
"hooks": "./config/hooks.json",
|
||||||
|
"mcpServers": "./.mcp.json"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## Best Practices
|
||||||
|
|
||||||
|
### Metadata
|
||||||
|
|
||||||
|
1. **Always include version**: Track changes and updates
|
||||||
|
2. **Write clear descriptions**: Help users understand plugin purpose
|
||||||
|
3. **Provide contact information**: Enable user support
|
||||||
|
4. **Link to documentation**: Reduce support burden
|
||||||
|
5. **Choose appropriate license**: Match project goals
|
||||||
|
|
||||||
|
### Paths
|
||||||
|
|
||||||
|
1. **Use defaults when possible**: Minimize configuration
|
||||||
|
2. **Organize logically**: Group related components
|
||||||
|
3. **Document custom paths**: Explain why non-standard layout used
|
||||||
|
4. **Test path resolution**: Verify on multiple systems
|
||||||
|
|
||||||
|
### Maintenance
|
||||||
|
|
||||||
|
1. **Bump version on changes**: Follow semantic versioning
|
||||||
|
2. **Update keywords**: Reflect new functionality
|
||||||
|
3. **Keep description current**: Match actual capabilities
|
||||||
|
4. **Maintain changelog**: Track version history
|
||||||
|
5. **Update repository links**: Keep URLs current
|
||||||
|
|
||||||
|
### Distribution
|
||||||
|
|
||||||
|
1. **Complete metadata before publishing**: All fields filled
|
||||||
|
2. **Test on clean install**: Verify plugin works without dev environment
|
||||||
|
3. **Validate manifest**: Use validation tools
|
||||||
|
4. **Include README**: Document installation and usage
|
||||||
|
5. **Specify license file**: Include LICENSE file in plugin root
|
||||||
693
skills/skill-development/SKILL.md
Normal file
693
skills/skill-development/SKILL.md
Normal file
@@ -0,0 +1,693 @@
|
|||||||
|
---
|
||||||
|
name: skill-development
|
||||||
|
description: This skill should be used when the user wants to "create a skill", "add a skill to plugin", "write a new skill", "improve skill description", "organize skill content", or needs guidance on skill structure, progressive disclosure, or skill development best practices for Claude Code plugins.
|
||||||
|
---
|
||||||
|
|
||||||
|
# Skill Development for Claude Code Plugins
|
||||||
|
|
||||||
|
This skill provides guidance for creating effective skills for Claude Code plugins.
|
||||||
|
|
||||||
|
## About Skills
|
||||||
|
|
||||||
|
Skills are modular, self-contained packages that extend Claude's capabilities by providing
|
||||||
|
specialized knowledge, workflows, and tools. Think of them as "onboarding guides" for specific
|
||||||
|
domains or tasks—they transform Claude from a general-purpose agent into a specialized agent
|
||||||
|
equipped with procedural knowledge that no model can fully possess.
|
||||||
|
|
||||||
|
### What Skills Provide
|
||||||
|
|
||||||
|
1. Specialized workflows - Multi-step procedures for specific domains
|
||||||
|
2. Tool integrations - Instructions for working with specific file formats or APIs
|
||||||
|
3. Domain expertise - Company-specific knowledge, schemas, business logic
|
||||||
|
4. Bundled resources - Scripts, references, and assets for complex and repetitive tasks
|
||||||
|
|
||||||
|
### Anatomy of a Skill
|
||||||
|
|
||||||
|
Every skill consists of a required SKILL.md file and optional bundled resources:
|
||||||
|
|
||||||
|
```
|
||||||
|
skill-name/
|
||||||
|
├── SKILL.md (required)
|
||||||
|
│ ├── YAML frontmatter metadata (required)
|
||||||
|
│ │ ├── name: (required)
|
||||||
|
│ │ └── description: (required)
|
||||||
|
│ └── Markdown instructions (required)
|
||||||
|
└── Bundled Resources (optional)
|
||||||
|
├── scripts/ - Executable code (Python/Bash/etc.)
|
||||||
|
├── references/ - Documentation intended to be loaded into context as needed
|
||||||
|
└── assets/ - Files used in output (templates, icons, fonts, etc.)
|
||||||
|
```
|
||||||
|
|
||||||
|
#### SKILL.md (required)
|
||||||
|
|
||||||
|
**Metadata Quality:** The `name` and `description` in YAML frontmatter determine when Claude will use the skill. Be specific about what the skill does and when to use it. Use the third-person (e.g. "This skill should be used when..." instead of "Use this skill when...").
|
||||||
|
|
||||||
|
#### Bundled Resources (optional)
|
||||||
|
|
||||||
|
##### Scripts (`scripts/`)
|
||||||
|
|
||||||
|
Executable code (Python/Bash/etc.) for tasks that require deterministic reliability or are repeatedly rewritten.
|
||||||
|
|
||||||
|
- **When to include**: When the same code is being rewritten repeatedly or deterministic reliability is needed
|
||||||
|
- **Example**: `scripts/rotate_pdf.py` for PDF rotation tasks
|
||||||
|
- **Benefits**: Token efficient, deterministic, may be executed without loading into context
|
||||||
|
- **Note**: Scripts may still need to be read by Claude for patching or environment-specific adjustments
|
||||||
|
|
||||||
|
##### References (`references/`)
|
||||||
|
|
||||||
|
Documentation and reference material intended to be loaded as needed into context to inform Claude's process and thinking.
|
||||||
|
|
||||||
|
- **When to include**: For documentation that Claude should reference while working
|
||||||
|
- **Examples**: `references/finance.md` for financial schemas, `references/mnda.md` for company NDA template, `references/policies.md` for company policies, `references/api_docs.md` for API specifications
|
||||||
|
- **Use cases**: Database schemas, API documentation, domain knowledge, company policies, detailed workflow guides
|
||||||
|
- **Benefits**: Keeps SKILL.md lean, loaded only when Claude determines it's needed
|
||||||
|
- **Best practice**: If files are large (>10k words), include grep search patterns in SKILL.md
|
||||||
|
- **Avoid duplication**: Information should live in either SKILL.md or references files, not both. Prefer references files for detailed information unless it's truly core to the skill—this keeps SKILL.md lean while making information discoverable without hogging the context window. Keep only essential procedural instructions and workflow guidance in SKILL.md; move detailed reference material, schemas, and examples to references files.
|
||||||
|
|
||||||
|
##### Assets (`assets/`)
|
||||||
|
|
||||||
|
Files not intended to be loaded into context, but rather used within the output Claude produces.
|
||||||
|
|
||||||
|
- **When to include**: When the skill needs files that will be used in the final output
|
||||||
|
- **Examples**: `assets/logo.png` for brand assets, `assets/slides.pptx` for PowerPoint templates, `assets/frontend-template/` for HTML/React boilerplate, `assets/font.ttf` for typography
|
||||||
|
- **Use cases**: Templates, images, icons, boilerplate code, fonts, sample documents that get copied or modified
|
||||||
|
- **Benefits**: Separates output resources from documentation, enables Claude to use files without loading them into context
|
||||||
|
|
||||||
|
### Progressive Disclosure Design Principle
|
||||||
|
|
||||||
|
Skills use a three-level loading system to manage context efficiently:
|
||||||
|
|
||||||
|
1. **Metadata (name + description)** - Always in context (~100 words)
|
||||||
|
2. **SKILL.md body** - When skill triggers (<5k words)
|
||||||
|
3. **Bundled resources** - As needed by Claude (Unlimited\*)
|
||||||
|
|
||||||
|
\*Unlimited because scripts can be executed without reading into context window.
|
||||||
|
|
||||||
|
## Skill Creation Process
|
||||||
|
|
||||||
|
To create a skill, follow the "Skill Creation Process" in order, skipping steps only if there is a clear reason why they are not applicable.
|
||||||
|
|
||||||
|
### Step 1: Understanding the Skill with Concrete Examples
|
||||||
|
|
||||||
|
Skip this step only when the skill's usage patterns are already clearly understood. It remains valuable even when working with an existing skill.
|
||||||
|
|
||||||
|
To create an effective skill, clearly understand concrete examples of how the skill will be used. This understanding can come from either direct user examples or generated examples that are validated with user feedback.
|
||||||
|
|
||||||
|
For example, when building an image-editor skill, relevant questions include:
|
||||||
|
|
||||||
|
- "What functionality should the image-editor skill support? Editing, rotating, anything else?"
|
||||||
|
- "Can you give some examples of how this skill would be used?"
|
||||||
|
- "I can imagine users asking for things like 'Remove the red-eye from this image' or 'Rotate this image'. Are there other ways you imagine this skill being used?"
|
||||||
|
- "What would a user say that should trigger this skill?"
|
||||||
|
|
||||||
|
To avoid overwhelming users, avoid asking too many questions in a single message. Start with the most important questions and follow up as needed for better effectiveness.
|
||||||
|
|
||||||
|
Conclude this step when there is a clear sense of the functionality the skill should support.
|
||||||
|
|
||||||
|
### Step 2: Planning the Reusable Skill Contents
|
||||||
|
|
||||||
|
To turn concrete examples into an effective skill, analyze each example by:
|
||||||
|
|
||||||
|
1. Considering how to execute on the example from scratch
|
||||||
|
2. Identifying what scripts, references, and assets would be helpful when executing these workflows repeatedly
|
||||||
|
|
||||||
|
Example: When building a `pdf-editor` skill to handle queries like "Help me rotate this PDF," the analysis shows:
|
||||||
|
|
||||||
|
1. Rotating a PDF requires re-writing the same code each time
|
||||||
|
2. A `scripts/rotate_pdf.py` script would be helpful to store in the skill
|
||||||
|
|
||||||
|
Example: When designing a `frontend-webapp-builder` skill for queries like "Build me a todo app" or "Build me a dashboard to track my steps," the analysis shows:
|
||||||
|
|
||||||
|
1. Writing a frontend webapp requires the same boilerplate HTML/React each time
|
||||||
|
2. An `assets/hello-world/` template containing the boilerplate HTML/React project files would be helpful to store in the skill
|
||||||
|
|
||||||
|
Example: When building a `big-query` skill to handle queries like "How many users have logged in today?" the analysis shows:
|
||||||
|
|
||||||
|
1. Querying BigQuery requires re-discovering the table schemas and relationships each time
|
||||||
|
2. A `references/schema.md` file documenting the table schemas would be helpful to store in the skill
|
||||||
|
|
||||||
|
**For Claude Code plugins:** When building a hooks skill, the analysis shows:
|
||||||
|
|
||||||
|
1. Developers repeatedly need to validate hooks.json and test hook scripts
|
||||||
|
2. `scripts/validate-hook-schema.sh` and `scripts/test-hook.sh` utilities would be helpful
|
||||||
|
3. `references/patterns.md` for detailed hook patterns to avoid bloating SKILL.md
|
||||||
|
|
||||||
|
To establish the skill's contents, analyze each concrete example to create a list of the reusable resources to include: scripts, references, and assets.
|
||||||
|
|
||||||
|
### Step 3: Create Skill Structure
|
||||||
|
|
||||||
|
For Claude Code plugins, create the skill directory structure:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
mkdir -p plugin-name/skills/skill-name/{references,examples,scripts}
|
||||||
|
touch plugin-name/skills/skill-name/SKILL.md
|
||||||
|
```
|
||||||
|
|
||||||
|
**Note:** Unlike the generic skill-creator which uses `init_skill.py`, plugin skills are created directly in the plugin's `skills/` directory with a simpler manual structure.
|
||||||
|
|
||||||
|
### Step 4: Edit the Skill
|
||||||
|
|
||||||
|
When editing the (newly-created or existing) skill, remember that the skill is being created for another instance of Claude to use. Focus on including information that would be beneficial and non-obvious to Claude. Consider what procedural knowledge, domain-specific details, or reusable assets would help another Claude instance execute these tasks more effectively.
|
||||||
|
|
||||||
|
#### Start with Reusable Skill Contents
|
||||||
|
|
||||||
|
To begin implementation, start with the reusable resources identified above: `scripts/`, `references/`, and `assets/` files. Note that this step may require user input. For example, when implementing a `brand-guidelines` skill, the user may need to provide brand assets or templates to store in `assets/`, or documentation to store in `references/`.
|
||||||
|
|
||||||
|
Also, delete any example files and directories not needed for the skill. Create only the directories you actually need (references/, examples/, scripts/).
|
||||||
|
|
||||||
|
#### Update SKILL.md
|
||||||
|
|
||||||
|
**Writing Style:** Write the entire skill using **imperative/infinitive form** (verb-first instructions), not second person. Use objective, instructional language (e.g., "To accomplish X, do Y" rather than "You should do X" or "If you need to do X"). This maintains consistency and clarity for AI consumption.
|
||||||
|
|
||||||
|
**Description (Frontmatter):** Use third-person format with specific trigger phrases. Max 300 characters:
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
---
|
||||||
|
name: skill-name
|
||||||
|
description: This skill should be used when the user asks to "specific phrase 1", "specific phrase 2", "specific phrase 3". Include exact phrases users would say that should trigger this skill. Be concrete and specific.
|
||||||
|
version: 0.1.0
|
||||||
|
---
|
||||||
|
```
|
||||||
|
|
||||||
|
**Good description examples:**
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
description: This skill should be used when the user asks to "create a hook", "add a PreToolUse hook", "validate tool use", "implement prompt-based hooks", or mentions hook events (PreToolUse, PostToolUse, Stop).
|
||||||
|
```
|
||||||
|
|
||||||
|
**Bad description examples:**
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
description: Use this skill when working with hooks. # Wrong person, vague
|
||||||
|
description: Load when user needs hook help. # Not third person
|
||||||
|
description: Provides hook guidance. # No trigger phrases
|
||||||
|
```
|
||||||
|
|
||||||
|
To complete SKILL.md body, answer the following questions:
|
||||||
|
|
||||||
|
1. What is the purpose of the skill, in a few sentences?
|
||||||
|
2. When should the skill be used? (Include this in frontmatter description with specific triggers)
|
||||||
|
3. In practice, how should Claude use the skill? All reusable skill contents developed above should be referenced so that Claude knows how to use them.
|
||||||
|
|
||||||
|
**Keep SKILL.md lean:** Target 1,500-2,000 words for the body. Move detailed content to references/:
|
||||||
|
|
||||||
|
- Detailed patterns → `references/patterns.md`
|
||||||
|
- Advanced techniques → `references/advanced.md`
|
||||||
|
- Migration guides → `references/migration.md`
|
||||||
|
- API references → `references/api-reference.md`
|
||||||
|
|
||||||
|
**Reference resources in SKILL.md:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
## Additional Resources
|
||||||
|
|
||||||
|
### Reference Files
|
||||||
|
|
||||||
|
For detailed patterns and techniques, consult:
|
||||||
|
|
||||||
|
- **`references/patterns.md`** - Common patterns
|
||||||
|
- **`references/advanced.md`** - Advanced use cases
|
||||||
|
|
||||||
|
### Example Files
|
||||||
|
|
||||||
|
Working examples in `examples/`:
|
||||||
|
|
||||||
|
- **`example-script.sh`** - Working example
|
||||||
|
```
|
||||||
|
|
||||||
|
### Step 5: Validate and Test
|
||||||
|
|
||||||
|
**For plugin skills, validation is different from generic skills:**
|
||||||
|
|
||||||
|
1. **Check structure**: Skill directory in `plugin-name/skills/skill-name/`
|
||||||
|
2. **Validate SKILL.md**: Has frontmatter with name and description
|
||||||
|
3. **Check trigger phrases**: Description includes specific user queries
|
||||||
|
4. **Verify writing style**: Body uses imperative/infinitive form, not second person
|
||||||
|
5. **Test progressive disclosure**: SKILL.md is lean (~1,500-2,000 words), detailed content in references/
|
||||||
|
6. **Check references**: All referenced files exist
|
||||||
|
7. **Validate examples**: Examples are complete and correct
|
||||||
|
8. **Test scripts**: Scripts are executable and work correctly
|
||||||
|
|
||||||
|
**Use the skill-reviewer agent:**
|
||||||
|
|
||||||
|
```
|
||||||
|
Ask: "Review my skill and check if it follows best practices"
|
||||||
|
```
|
||||||
|
|
||||||
|
The skill-reviewer agent will check description quality, content organization, and progressive disclosure.
|
||||||
|
|
||||||
|
### Step 6: Iterate
|
||||||
|
|
||||||
|
After testing the skill, users may request improvements. Often this happens right after using the skill, with fresh context of how the skill performed.
|
||||||
|
|
||||||
|
**Iteration workflow:**
|
||||||
|
|
||||||
|
1. Use the skill on real tasks
|
||||||
|
2. Notice struggles or inefficiencies
|
||||||
|
3. Identify how SKILL.md or bundled resources should be updated
|
||||||
|
4. Implement changes and test again
|
||||||
|
|
||||||
|
**Common improvements:**
|
||||||
|
|
||||||
|
- Strengthen trigger phrases in description
|
||||||
|
- Move long sections from SKILL.md to references/
|
||||||
|
- Add missing examples or scripts
|
||||||
|
- Clarify ambiguous instructions
|
||||||
|
- Add edge case handling
|
||||||
|
|
||||||
|
## Plugin-Specific Considerations
|
||||||
|
|
||||||
|
### Skill Location in Plugins
|
||||||
|
|
||||||
|
Plugin skills live in the plugin's `skills/` directory:
|
||||||
|
|
||||||
|
```
|
||||||
|
my-plugin/
|
||||||
|
├── .claude-plugin/
|
||||||
|
│ └── plugin.json
|
||||||
|
├── commands/
|
||||||
|
├── agents/
|
||||||
|
└── skills/
|
||||||
|
└── my-skill/
|
||||||
|
├── SKILL.md
|
||||||
|
├── references/
|
||||||
|
├── examples/
|
||||||
|
└── scripts/
|
||||||
|
```
|
||||||
|
|
||||||
|
### Auto-Discovery
|
||||||
|
|
||||||
|
Claude Code automatically discovers skills:
|
||||||
|
|
||||||
|
- Scans `skills/` directory
|
||||||
|
- Finds subdirectories containing `SKILL.md`
|
||||||
|
- Loads skill metadata (name + description) always
|
||||||
|
- Loads SKILL.md body when skill triggers
|
||||||
|
- Loads references/examples when needed
|
||||||
|
|
||||||
|
### No Packaging Needed
|
||||||
|
|
||||||
|
Plugin skills are distributed as part of the plugin, not as separate ZIP files. Users get skills when they install the plugin.
|
||||||
|
|
||||||
|
### Testing in Plugins
|
||||||
|
|
||||||
|
Test skills by installing plugin locally:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Test with --plugin-dir
|
||||||
|
cc --plugin-dir /path/to/plugin
|
||||||
|
|
||||||
|
# Ask questions that should trigger the skill
|
||||||
|
# Verify skill loads correctly
|
||||||
|
```
|
||||||
|
|
||||||
|
## Examples from Plugin-Dev
|
||||||
|
|
||||||
|
Study the skills in this plugin as examples of best practices:
|
||||||
|
|
||||||
|
**hook-development skill:**
|
||||||
|
|
||||||
|
- Excellent trigger phrases: "create a hook", "add a PreToolUse hook", etc.
|
||||||
|
- Lean SKILL.md (1,651 words)
|
||||||
|
- 3 references/ files for detailed content
|
||||||
|
- 3 examples/ of working hooks
|
||||||
|
- 3 scripts/ utilities
|
||||||
|
|
||||||
|
**agent-development skill:**
|
||||||
|
|
||||||
|
- Strong triggers: "create an agent", "agent frontmatter", etc.
|
||||||
|
- Focused SKILL.md (1,438 words)
|
||||||
|
- References include the AI generation prompt from Claude Code
|
||||||
|
- Complete agent examples
|
||||||
|
|
||||||
|
**plugin-settings skill:**
|
||||||
|
|
||||||
|
- Specific triggers: "plugin settings", ".local.md files", "YAML frontmatter"
|
||||||
|
- References show real implementations (multi-agent-swarm, ralph-wiggum)
|
||||||
|
- Working parsing scripts
|
||||||
|
|
||||||
|
Each demonstrates progressive disclosure and strong triggering.
|
||||||
|
|
||||||
|
## Progressive Disclosure in Practice
|
||||||
|
|
||||||
|
### What Goes in SKILL.md
|
||||||
|
|
||||||
|
**Include (always loaded when skill triggers):**
|
||||||
|
|
||||||
|
- Core concepts and overview
|
||||||
|
- Essential procedures and workflows
|
||||||
|
- Quick reference tables
|
||||||
|
- Pointers to references/examples/scripts
|
||||||
|
- Most common use cases
|
||||||
|
|
||||||
|
**Keep under 3,000 words, ideally 1,500-2,000 words**
|
||||||
|
|
||||||
|
### What Goes in references/
|
||||||
|
|
||||||
|
**Move to references/ (loaded as needed):**
|
||||||
|
|
||||||
|
- Detailed patterns and advanced techniques
|
||||||
|
- Comprehensive API documentation
|
||||||
|
- Migration guides
|
||||||
|
- Edge cases and troubleshooting
|
||||||
|
- Extensive examples and walkthroughs
|
||||||
|
|
||||||
|
**Each reference file can be large (2,000-5,000+ words)**
|
||||||
|
|
||||||
|
### What Goes in examples/
|
||||||
|
|
||||||
|
**Working code examples:**
|
||||||
|
|
||||||
|
- Complete, runnable scripts
|
||||||
|
- Configuration files
|
||||||
|
- Template files
|
||||||
|
- Real-world usage examples
|
||||||
|
|
||||||
|
**Users can copy and adapt these directly**
|
||||||
|
|
||||||
|
### What Goes in scripts/
|
||||||
|
|
||||||
|
**Utility scripts:**
|
||||||
|
|
||||||
|
- Validation tools
|
||||||
|
- Testing helpers
|
||||||
|
- Parsing utilities
|
||||||
|
- Automation scripts
|
||||||
|
|
||||||
|
**Should be executable and documented**
|
||||||
|
|
||||||
|
## Writing Style Requirements
|
||||||
|
|
||||||
|
### Imperative/Infinitive Form
|
||||||
|
|
||||||
|
Write using verb-first instructions, not second person:
|
||||||
|
|
||||||
|
**Correct (imperative):**
|
||||||
|
|
||||||
|
```
|
||||||
|
To create a hook, define the event type.
|
||||||
|
Configure the MCP server with authentication.
|
||||||
|
Validate settings before use.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Incorrect (second person):**
|
||||||
|
|
||||||
|
```
|
||||||
|
You should create a hook by defining the event type.
|
||||||
|
You need to configure the MCP server.
|
||||||
|
You must validate settings before use.
|
||||||
|
```
|
||||||
|
|
||||||
|
### Third-Person in Description
|
||||||
|
|
||||||
|
The frontmatter description must use third person:
|
||||||
|
|
||||||
|
**Correct:**
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
description: This skill should be used when the user asks to "create X", "configure Y"...
|
||||||
|
```
|
||||||
|
|
||||||
|
**Incorrect:**
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
description: Use this skill when you want to create X...
|
||||||
|
description: Load this skill when user asks...
|
||||||
|
```
|
||||||
|
|
||||||
|
### Objective, Instructional Language
|
||||||
|
|
||||||
|
Focus on what to do, not who should do it:
|
||||||
|
|
||||||
|
**Correct:**
|
||||||
|
|
||||||
|
```
|
||||||
|
Parse the frontmatter using sed.
|
||||||
|
Extract fields with grep.
|
||||||
|
Validate values before use.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Incorrect:**
|
||||||
|
|
||||||
|
```
|
||||||
|
You can parse the frontmatter...
|
||||||
|
Claude should extract fields...
|
||||||
|
The user might validate values...
|
||||||
|
```
|
||||||
|
|
||||||
|
## Validation Checklist
|
||||||
|
|
||||||
|
Before finalizing a skill:
|
||||||
|
|
||||||
|
**Structure:**
|
||||||
|
|
||||||
|
- [ ] SKILL.md file exists with valid YAML frontmatter
|
||||||
|
- [ ] Frontmatter has `name` and `description` fields
|
||||||
|
- [ ] Markdown body is present and substantial
|
||||||
|
- [ ] Referenced files actually exist
|
||||||
|
|
||||||
|
**Description Quality:**
|
||||||
|
|
||||||
|
- [ ] Uses third person ("This skill should be used when...")
|
||||||
|
- [ ] Includes specific trigger phrases users would say
|
||||||
|
- [ ] Lists concrete scenarios ("create X", "configure Y")
|
||||||
|
- [ ] Not vague or generic
|
||||||
|
- [ ] Description is max 300 characters
|
||||||
|
|
||||||
|
**Content Quality:**
|
||||||
|
|
||||||
|
- [ ] SKILL.md body uses imperative/infinitive form
|
||||||
|
- [ ] Body is focused and lean (1,500-2,000 words ideal, <5k max)
|
||||||
|
- [ ] Detailed content moved to references/
|
||||||
|
- [ ] Examples are complete and working
|
||||||
|
- [ ] Scripts are executable and documented
|
||||||
|
|
||||||
|
**Progressive Disclosure:**
|
||||||
|
|
||||||
|
- [ ] Core concepts in SKILL.md
|
||||||
|
- [ ] Detailed docs in references/
|
||||||
|
- [ ] Working code in examples/
|
||||||
|
- [ ] Utilities in scripts/
|
||||||
|
- [ ] SKILL.md references these resources
|
||||||
|
|
||||||
|
**Testing:**
|
||||||
|
|
||||||
|
- [ ] Skill triggers on expected user queries
|
||||||
|
- [ ] Content is helpful for intended tasks
|
||||||
|
- [ ] No duplicated information across files
|
||||||
|
- [ ] References load when needed
|
||||||
|
|
||||||
|
## Common Mistakes to Avoid
|
||||||
|
|
||||||
|
### Mistake 1: Weak Trigger Description
|
||||||
|
|
||||||
|
❌ **Bad:**
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
description: Provides guidance for working with hooks.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Why bad:** Vague, no specific trigger phrases, not third person
|
||||||
|
|
||||||
|
✅ **Good:**
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
description: This skill should be used when the user asks to "create a hook", "add a PreToolUse hook", "validate tool use", or mentions hook events. Provides comprehensive hooks API guidance.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Why good:** Third person, specific phrases, concrete scenarios
|
||||||
|
|
||||||
|
### Mistake 2: Too Much in SKILL.md
|
||||||
|
|
||||||
|
❌ **Bad:**
|
||||||
|
|
||||||
|
```
|
||||||
|
skill-name/
|
||||||
|
└── SKILL.md (8,000 words - everything in one file)
|
||||||
|
```
|
||||||
|
|
||||||
|
**Why bad:** Bloats context when skill loads, detailed content always loaded
|
||||||
|
|
||||||
|
✅ **Good:**
|
||||||
|
|
||||||
|
```
|
||||||
|
skill-name/
|
||||||
|
├── SKILL.md (1,800 words - core essentials)
|
||||||
|
└── references/
|
||||||
|
├── patterns.md (2,500 words)
|
||||||
|
└── advanced.md (3,700 words)
|
||||||
|
```
|
||||||
|
|
||||||
|
**Why good:** Progressive disclosure, detailed content loaded only when needed
|
||||||
|
|
||||||
|
### Mistake 3: Second Person Writing
|
||||||
|
|
||||||
|
❌ **Bad:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
You should start by reading the configuration file.
|
||||||
|
You need to validate the input.
|
||||||
|
You can use the grep tool to search.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Why bad:** Second person, not imperative form
|
||||||
|
|
||||||
|
✅ **Good:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
Start by reading the configuration file.
|
||||||
|
Validate the input before processing.
|
||||||
|
Use the grep tool to search for patterns.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Why good:** Imperative form, direct instructions
|
||||||
|
|
||||||
|
### Mistake 4: Missing Resource References
|
||||||
|
|
||||||
|
❌ **Bad:**
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
# SKILL.md
|
||||||
|
|
||||||
|
[Core content]
|
||||||
|
|
||||||
|
[No mention of references/ or examples/]
|
||||||
|
```
|
||||||
|
|
||||||
|
**Why bad:** Claude doesn't know references exist
|
||||||
|
|
||||||
|
### Mistake 5: Incorrect Name Metadata Format
|
||||||
|
|
||||||
|
❌ **Bad:**
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
name: Skill Name
|
||||||
|
```
|
||||||
|
|
||||||
|
**Why bad:** Name should be lowercase with hyphens
|
||||||
|
|
||||||
|
✅ **Good:**
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
name: skill-name
|
||||||
|
```
|
||||||
|
|
||||||
|
## Additional Resources
|
||||||
|
|
||||||
|
### Reference Files
|
||||||
|
|
||||||
|
- **`references/patterns.md`** - Detailed patterns
|
||||||
|
- **`references/advanced.md`** - Advanced techniques
|
||||||
|
|
||||||
|
### Examples
|
||||||
|
|
||||||
|
- **`examples/script.sh`** - Working example
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
**Why good:** Claude knows where to find additional information
|
||||||
|
|
||||||
|
## Quick Reference
|
||||||
|
|
||||||
|
### Minimal Skill
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
skill-name/
|
||||||
|
└── SKILL.md
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
Good for: Simple knowledge, no complex resources needed
|
||||||
|
|
||||||
|
### Standard Skill (Recommended)
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
skill-name/
|
||||||
|
├── SKILL.md
|
||||||
|
├── references/
|
||||||
|
│ └── detailed-guide.md
|
||||||
|
└── examples/
|
||||||
|
└── working-example.sh
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
Good for: Most plugin skills with detailed documentation
|
||||||
|
|
||||||
|
### Complete Skill
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
skill-name/
|
||||||
|
├── SKILL.md
|
||||||
|
├── references/
|
||||||
|
│ ├── patterns.md
|
||||||
|
│ └── advanced.md
|
||||||
|
├── examples/
|
||||||
|
│ ├── example1.sh
|
||||||
|
│ └── example2.json
|
||||||
|
└── scripts/
|
||||||
|
└── validate.sh
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
Good for: Complex domains with validation utilities
|
||||||
|
|
||||||
|
## Best Practices Summary
|
||||||
|
|
||||||
|
✅ **DO:**
|
||||||
|
- Use third-person in description ("This skill should be used when...")
|
||||||
|
- Include specific trigger phrases ("create X", "configure Y")
|
||||||
|
- Keep SKILL.md lean (1,500-2,000 words)
|
||||||
|
- Use progressive disclosure (move details to references/)
|
||||||
|
- Write in imperative/infinitive form
|
||||||
|
- Reference supporting files clearly
|
||||||
|
- Provide working examples
|
||||||
|
- Create utility scripts for common operations
|
||||||
|
- Study plugin-dev's skills as templates
|
||||||
|
|
||||||
|
❌ **DON'T:**
|
||||||
|
- Use second person anywhere
|
||||||
|
- Have vague trigger conditions
|
||||||
|
- Put everything in SKILL.md (>3,000 words without references/)
|
||||||
|
- Write in second person ("You should...")
|
||||||
|
- Leave resources unreferenced
|
||||||
|
- Include broken or incomplete examples
|
||||||
|
- Skip validation
|
||||||
|
|
||||||
|
## Additional Resources
|
||||||
|
|
||||||
|
### Study These Skills
|
||||||
|
|
||||||
|
Plugin-dev's skills demonstrate best practices:
|
||||||
|
- `../hook-development/` - Progressive disclosure, utilities
|
||||||
|
- `../agent-development/` - AI-assisted creation, references
|
||||||
|
- `../mcp-integration/` - Comprehensive references
|
||||||
|
- `../plugin-settings/` - Real-world examples
|
||||||
|
- `../command-development/` - Clear critical concepts
|
||||||
|
- `../plugin-structure/` - Good organization
|
||||||
|
|
||||||
|
### Reference Files
|
||||||
|
|
||||||
|
For complete skill-creator methodology:
|
||||||
|
- **`references/skill-creator-original.md`** - Full original skill-creator content
|
||||||
|
|
||||||
|
## Implementation Workflow
|
||||||
|
|
||||||
|
To create a skill for your plugin:
|
||||||
|
|
||||||
|
1. **Understand use cases**: Identify concrete examples of skill usage
|
||||||
|
2. **Plan resources**: Determine what scripts/references/examples needed
|
||||||
|
3. **Create structure**: `mkdir -p skills/skill-name/{references,examples,scripts}`
|
||||||
|
4. **Write SKILL.md**:
|
||||||
|
- Frontmatter with third-person description and trigger phrases
|
||||||
|
- Lean body (1,500-2,000 words) in imperative form
|
||||||
|
- Reference supporting files
|
||||||
|
5. **Add resources**: Create references/, examples/, scripts/ as needed
|
||||||
|
6. **Validate**: Check description, writing style, organization
|
||||||
|
7. **Test**: Verify skill loads on expected triggers
|
||||||
|
8. **Iterate**: Improve based on usage
|
||||||
|
|
||||||
|
Focus on strong trigger descriptions, progressive disclosure, and imperative writing style for effective skills that load when needed and provide targeted guidance.
|
||||||
|
```
|
||||||
209
skills/skill-development/references/skill-creator-original.md
Normal file
209
skills/skill-development/references/skill-creator-original.md
Normal file
@@ -0,0 +1,209 @@
|
|||||||
|
---
|
||||||
|
name: skill-creator
|
||||||
|
description: Guide for creating effective skills. This skill should be used when users want to create a new skill (or update an existing skill) that extends Claude's capabilities with specialized knowledge, workflows, or tool integrations.
|
||||||
|
license: Complete terms in LICENSE.txt
|
||||||
|
---
|
||||||
|
|
||||||
|
# Skill Creator
|
||||||
|
|
||||||
|
This skill provides guidance for creating effective skills.
|
||||||
|
|
||||||
|
## About Skills
|
||||||
|
|
||||||
|
Skills are modular, self-contained packages that extend Claude's capabilities by providing
|
||||||
|
specialized knowledge, workflows, and tools. Think of them as "onboarding guides" for specific
|
||||||
|
domains or tasks—they transform Claude from a general-purpose agent into a specialized agent
|
||||||
|
equipped with procedural knowledge that no model can fully possess.
|
||||||
|
|
||||||
|
### What Skills Provide
|
||||||
|
|
||||||
|
1. Specialized workflows - Multi-step procedures for specific domains
|
||||||
|
2. Tool integrations - Instructions for working with specific file formats or APIs
|
||||||
|
3. Domain expertise - Company-specific knowledge, schemas, business logic
|
||||||
|
4. Bundled resources - Scripts, references, and assets for complex and repetitive tasks
|
||||||
|
|
||||||
|
### Anatomy of a Skill
|
||||||
|
|
||||||
|
Every skill consists of a required SKILL.md file and optional bundled resources:
|
||||||
|
|
||||||
|
```
|
||||||
|
skill-name/
|
||||||
|
├── SKILL.md (required)
|
||||||
|
│ ├── YAML frontmatter metadata (required)
|
||||||
|
│ │ ├── name: (required)
|
||||||
|
│ │ └── description: (required)
|
||||||
|
│ └── Markdown instructions (required)
|
||||||
|
└── Bundled Resources (optional)
|
||||||
|
├── scripts/ - Executable code (Python/Bash/etc.)
|
||||||
|
├── references/ - Documentation intended to be loaded into context as needed
|
||||||
|
└── assets/ - Files used in output (templates, icons, fonts, etc.)
|
||||||
|
```
|
||||||
|
|
||||||
|
#### SKILL.md (required)
|
||||||
|
|
||||||
|
**Metadata Quality:** The `name` and `description` in YAML frontmatter determine when Claude will use the skill. Be specific about what the skill does and when to use it. Use the third-person (e.g. "This skill should be used when..." instead of "Use this skill when...").
|
||||||
|
|
||||||
|
#### Bundled Resources (optional)
|
||||||
|
|
||||||
|
##### Scripts (`scripts/`)
|
||||||
|
|
||||||
|
Executable code (Python/Bash/etc.) for tasks that require deterministic reliability or are repeatedly rewritten.
|
||||||
|
|
||||||
|
- **When to include**: When the same code is being rewritten repeatedly or deterministic reliability is needed
|
||||||
|
- **Example**: `scripts/rotate_pdf.py` for PDF rotation tasks
|
||||||
|
- **Benefits**: Token efficient, deterministic, may be executed without loading into context
|
||||||
|
- **Note**: Scripts may still need to be read by Claude for patching or environment-specific adjustments
|
||||||
|
|
||||||
|
##### References (`references/`)
|
||||||
|
|
||||||
|
Documentation and reference material intended to be loaded as needed into context to inform Claude's process and thinking.
|
||||||
|
|
||||||
|
- **When to include**: For documentation that Claude should reference while working
|
||||||
|
- **Examples**: `references/finance.md` for financial schemas, `references/mnda.md` for company NDA template, `references/policies.md` for company policies, `references/api_docs.md` for API specifications
|
||||||
|
- **Use cases**: Database schemas, API documentation, domain knowledge, company policies, detailed workflow guides
|
||||||
|
- **Benefits**: Keeps SKILL.md lean, loaded only when Claude determines it's needed
|
||||||
|
- **Best practice**: If files are large (>10k words), include grep search patterns in SKILL.md
|
||||||
|
- **Avoid duplication**: Information should live in either SKILL.md or references files, not both. Prefer references files for detailed information unless it's truly core to the skill—this keeps SKILL.md lean while making information discoverable without hogging the context window. Keep only essential procedural instructions and workflow guidance in SKILL.md; move detailed reference material, schemas, and examples to references files.
|
||||||
|
|
||||||
|
##### Assets (`assets/`)
|
||||||
|
|
||||||
|
Files not intended to be loaded into context, but rather used within the output Claude produces.
|
||||||
|
|
||||||
|
- **When to include**: When the skill needs files that will be used in the final output
|
||||||
|
- **Examples**: `assets/logo.png` for brand assets, `assets/slides.pptx` for PowerPoint templates, `assets/frontend-template/` for HTML/React boilerplate, `assets/font.ttf` for typography
|
||||||
|
- **Use cases**: Templates, images, icons, boilerplate code, fonts, sample documents that get copied or modified
|
||||||
|
- **Benefits**: Separates output resources from documentation, enables Claude to use files without loading them into context
|
||||||
|
|
||||||
|
### Progressive Disclosure Design Principle
|
||||||
|
|
||||||
|
Skills use a three-level loading system to manage context efficiently:
|
||||||
|
|
||||||
|
1. **Metadata (name + description)** - Always in context (~100 words)
|
||||||
|
2. **SKILL.md body** - When skill triggers (<5k words)
|
||||||
|
3. **Bundled resources** - As needed by Claude (Unlimited*)
|
||||||
|
|
||||||
|
*Unlimited because scripts can be executed without reading into context window.
|
||||||
|
|
||||||
|
## Skill Creation Process
|
||||||
|
|
||||||
|
To create a skill, follow the "Skill Creation Process" in order, skipping steps only if there is a clear reason why they are not applicable.
|
||||||
|
|
||||||
|
### Step 1: Understanding the Skill with Concrete Examples
|
||||||
|
|
||||||
|
Skip this step only when the skill's usage patterns are already clearly understood. It remains valuable even when working with an existing skill.
|
||||||
|
|
||||||
|
To create an effective skill, clearly understand concrete examples of how the skill will be used. This understanding can come from either direct user examples or generated examples that are validated with user feedback.
|
||||||
|
|
||||||
|
For example, when building an image-editor skill, relevant questions include:
|
||||||
|
|
||||||
|
- "What functionality should the image-editor skill support? Editing, rotating, anything else?"
|
||||||
|
- "Can you give some examples of how this skill would be used?"
|
||||||
|
- "I can imagine users asking for things like 'Remove the red-eye from this image' or 'Rotate this image'. Are there other ways you imagine this skill being used?"
|
||||||
|
- "What would a user say that should trigger this skill?"
|
||||||
|
|
||||||
|
To avoid overwhelming users, avoid asking too many questions in a single message. Start with the most important questions and follow up as needed for better effectiveness.
|
||||||
|
|
||||||
|
Conclude this step when there is a clear sense of the functionality the skill should support.
|
||||||
|
|
||||||
|
### Step 2: Planning the Reusable Skill Contents
|
||||||
|
|
||||||
|
To turn concrete examples into an effective skill, analyze each example by:
|
||||||
|
|
||||||
|
1. Considering how to execute on the example from scratch
|
||||||
|
2. Identifying what scripts, references, and assets would be helpful when executing these workflows repeatedly
|
||||||
|
|
||||||
|
Example: When building a `pdf-editor` skill to handle queries like "Help me rotate this PDF," the analysis shows:
|
||||||
|
|
||||||
|
1. Rotating a PDF requires re-writing the same code each time
|
||||||
|
2. A `scripts/rotate_pdf.py` script would be helpful to store in the skill
|
||||||
|
|
||||||
|
Example: When designing a `frontend-webapp-builder` skill for queries like "Build me a todo app" or "Build me a dashboard to track my steps," the analysis shows:
|
||||||
|
|
||||||
|
1. Writing a frontend webapp requires the same boilerplate HTML/React each time
|
||||||
|
2. An `assets/hello-world/` template containing the boilerplate HTML/React project files would be helpful to store in the skill
|
||||||
|
|
||||||
|
Example: When building a `big-query` skill to handle queries like "How many users have logged in today?" the analysis shows:
|
||||||
|
|
||||||
|
1. Querying BigQuery requires re-discovering the table schemas and relationships each time
|
||||||
|
2. A `references/schema.md` file documenting the table schemas would be helpful to store in the skill
|
||||||
|
|
||||||
|
To establish the skill's contents, analyze each concrete example to create a list of the reusable resources to include: scripts, references, and assets.
|
||||||
|
|
||||||
|
### Step 3: Initializing the Skill
|
||||||
|
|
||||||
|
At this point, it is time to actually create the skill.
|
||||||
|
|
||||||
|
Skip this step only if the skill being developed already exists, and iteration or packaging is needed. In this case, continue to the next step.
|
||||||
|
|
||||||
|
When creating a new skill from scratch, always run the `init_skill.py` script. The script conveniently generates a new template skill directory that automatically includes everything a skill requires, making the skill creation process much more efficient and reliable.
|
||||||
|
|
||||||
|
Usage:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
scripts/init_skill.py <skill-name> --path <output-directory>
|
||||||
|
```
|
||||||
|
|
||||||
|
The script:
|
||||||
|
|
||||||
|
- Creates the skill directory at the specified path
|
||||||
|
- Generates a SKILL.md template with proper frontmatter and TODO placeholders
|
||||||
|
- Creates example resource directories: `scripts/`, `references/`, and `assets/`
|
||||||
|
- Adds example files in each directory that can be customized or deleted
|
||||||
|
|
||||||
|
After initialization, customize or remove the generated SKILL.md and example files as needed.
|
||||||
|
|
||||||
|
### Step 4: Edit the Skill
|
||||||
|
|
||||||
|
When editing the (newly-generated or existing) skill, remember that the skill is being created for another instance of Claude to use. Focus on including information that would be beneficial and non-obvious to Claude. Consider what procedural knowledge, domain-specific details, or reusable assets would help another Claude instance execute these tasks more effectively.
|
||||||
|
|
||||||
|
#### Start with Reusable Skill Contents
|
||||||
|
|
||||||
|
To begin implementation, start with the reusable resources identified above: `scripts/`, `references/`, and `assets/` files. Note that this step may require user input. For example, when implementing a `brand-guidelines` skill, the user may need to provide brand assets or templates to store in `assets/`, or documentation to store in `references/`.
|
||||||
|
|
||||||
|
Also, delete any example files and directories not needed for the skill. The initialization script creates example files in `scripts/`, `references/`, and `assets/` to demonstrate structure, but most skills won't need all of them.
|
||||||
|
|
||||||
|
#### Update SKILL.md
|
||||||
|
|
||||||
|
**Writing Style:** Write the entire skill using **imperative/infinitive form** (verb-first instructions), not second person. Use objective, instructional language (e.g., "To accomplish X, do Y" rather than "You should do X" or "If you need to do X"). This maintains consistency and clarity for AI consumption.
|
||||||
|
|
||||||
|
To complete SKILL.md, answer the following questions:
|
||||||
|
|
||||||
|
1. What is the purpose of the skill, in a few sentences?
|
||||||
|
2. When should the skill be used?
|
||||||
|
3. In practice, how should Claude use the skill? All reusable skill contents developed above should be referenced so that Claude knows how to use them.
|
||||||
|
|
||||||
|
### Step 5: Packaging a Skill
|
||||||
|
|
||||||
|
Once the skill is ready, it should be packaged into a distributable zip file that gets shared with the user. The packaging process automatically validates the skill first to ensure it meets all requirements:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
scripts/package_skill.py <path/to/skill-folder>
|
||||||
|
```
|
||||||
|
|
||||||
|
Optional output directory specification:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
scripts/package_skill.py <path/to/skill-folder> ./dist
|
||||||
|
```
|
||||||
|
|
||||||
|
The packaging script will:
|
||||||
|
|
||||||
|
1. **Validate** the skill automatically, checking:
|
||||||
|
- YAML frontmatter format and required fields
|
||||||
|
- Skill naming conventions and directory structure
|
||||||
|
- Description completeness and quality
|
||||||
|
- File organization and resource references
|
||||||
|
|
||||||
|
2. **Package** the skill if validation passes, creating a zip file named after the skill (e.g., `my-skill.zip`) that includes all files and maintains the proper directory structure for distribution.
|
||||||
|
|
||||||
|
If validation fails, the script will report the errors and exit without creating a package. Fix any validation errors and run the packaging command again.
|
||||||
|
|
||||||
|
### Step 6: Iterate
|
||||||
|
|
||||||
|
After testing the skill, users may request improvements. Often this happens right after using the skill, with fresh context of how the skill performed.
|
||||||
|
|
||||||
|
**Iteration workflow:**
|
||||||
|
1. Use the skill on real tasks
|
||||||
|
2. Notice struggles or inefficiencies
|
||||||
|
3. Identify how SKILL.md or bundled resources should be updated
|
||||||
|
4. Implement changes and test again
|
||||||
Reference in New Issue
Block a user