plugins/agent-plugin-analyzer/skills/audit-plugin/SKILL.md
This skill should be used when the user asks to "audit a plugin", "validate my plugin", "check plugin structure", "verify plugin is correct", "validate .claude-plugin/plugin.json", "check if my plugin is compliant", "review plugin components", or mentions plugin validation or structure compliance. Also trigger proactively after the user creates or modifies any plugin component (commands, agents, skills, hooks, .claude-plugin/plugin.json). Use this skill even when the user says "check my work" or "make sure this is right" in a plugin context. Do NOT use this for auditing individual skills only (use skill-reviewer for that).
npx skillsauth add richfrem/agent-plugins-skills audit-pluginInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
This skill requires Python 3.8+ and standard library only. No external packages needed.
To install this skill's dependencies:
pip-compile ./requirements.in
pip install -r ./requirements.txt
See ../../requirements.txt for the dependency lockfile (currently empty — standard library only).
Performs comprehensive validation of a Claude Code plugin against structure standards,
naming conventions, component requirements, and security best practices. Uses the
plugin-validator agent for deep validation, supported by component-specific scripts.
Establish the plugin root:
../../../../.claude-plugin/plugin.json -- this is the definitive markerNote: The
plugin-validatoragent is defined inagent-scaffolders. If not installed, skip this step and rely on the component scripts in Step 3 and manual checks in Step 4.
Trigger the plugin-validator agent for comprehensive validation:
"Validate the plugin at <path>"
The agent checks all 10 categories automatically:
../../../../.claude-plugin/plugin.json) -- JSON syntax, required name field, kebab-case.claude-plugin/commands/**/*.md) -- frontmatter, description, argument-hint, allowed-toolsagents/**/*.md) -- name, description with <example> blocks, model, colorskills/*/SKILL.md) -- frontmatter, name, description (<= 1024 chars), supporting directorieshooks/hooks.json) -- JSON syntax, valid event names, matcher + hooks array.mcp.json) -- server type, required fields, HTTPS enforcementOutput format from plugin-validator:
## Plugin Validation Report
### Plugin: [name] | Location: [path]
### Summary: [PASS/FAIL with stats]
### Critical Issues ([count]) -- file path + issue + fix
### Warnings ([count]) -- file path + recommendation
### Component Summary -- counts of each type
### Positive Findings
### Overall Assessment: [PASS/FAIL + reasoning]
Before deep validation, run the load-error fixer to catch issues that prevent
Claude Code from loading the plugin at all. These are silent failures — the plugin
simply doesn't load with no useful error until you run /doctor.
python ${CLAUDE_PLUGIN_ROOT}/scripts/fix_plugin_load_errors.py <plugin_root>
What it fixes automatically:
| Issue | Symptom in /doctor | Root cause |
|---|---|---|
| plugin.json has skills/agents/hooks/commands arrays | Invalid input | Validator rejects these — auto-discovery handles them |
| hooks.json is {} (empty object) | expected record, received undefined | Must be {"hooks": {}} |
| hooks.json is [] (array) | expected object received array | Must be an object |
| hooks.json flat format {"EventName":{...}} | expected record, received undefined | Must be nested under "hooks" key |
| hooks.json/lsp.json/.mcp.json has literal \n chars | Unrecognized token '\' | Python json.dump wrote escaped newlines; file must have real newlines |
| SKILL.md has comment lines before --- | skill fails to load | Frontmatter parser requires --- as the very first line |
Correct hooks.json format:
{
"hooks": {
"SessionStart": [
{
"matcher": "",
"hooks": [{ "type": "command", "command": "python ${CLAUDE_PLUGIN_ROOT}/hooks/script.py" }]
}
]
}
}
Empty hooks (no hooks needed):
{ "hooks": {} }
IMPORTANT — Cache coverage: Claude Code scans ALL cached versions under
~/.claude/plugins/cache/, not just the active installPath. Fixing source files
and reinstalling is the only reliable fix. Use uvx plugin-add to reinstall.
Git checks out symlinks as plain-text "stand-in" files when core.symlinks=false (common on
Windows without Developer Mode, or when cloned without the setting). Stand-ins look like real files
but contain only a relative path (e.g. ../../../scripts/execute.py). They are functionally broken
— the bridge installer will copy the path string, not the actual script.
Run the bulk scanner from the link-checker plugin:
python plugins/link-checker/scripts/bulk_symlink_fixer.py plugins/<plugin-name>
The scanner detects both:
< 512 bytes) whose content looks like a relative pathImportant — the fixer has a silent failure mode. symlink_manager.py always exits 0, so
bulk_symlink_fixer.py prints "✓ Fixed" even when the source doesn't exist. Always verify manually:
# Confirm symlinks resolved (count should match expectations)
find plugins/<plugin-name> -type l | wc -l
# Confirm no text-file stand-ins remain for scripts paths (critical)
find plugins/<plugin-name>/skills -path "*/scripts/*" -type f ! -type l
Two categories of stand-ins:
Valid stand-ins (target exists) — the fixer converts these automatically. If it fails silently, convert manually:
# Read the path out of the stand-in, unlink the file, recreate as symlink
content = Path(standin).read_text().strip()
Path(standin).unlink()
Path(standin).symlink_to(content)
Wrong-path stand-ins (target missing) — common cause: an extra subdirectory in the path
(e.g. references/architecture/architecture.md when the file is at references/architecture.md).
Correct the relative path before creating the symlink. Check what actually exists at the plugin
references/ root and recalculate the depth.
Correct symlink pattern (must match ADR manager / all standard skills):
skills/<skill>/scripts/execute.py → ../../../scripts/<canonical_name>.py
skills/<skill>/references/architecture.md → ../../../references/architecture.md
The symlink filename and the target filename may differ (e.g. execute.py → exploration_optimizer_execute.py)
— that is intentional and valid.
6 known missing-source stand-ins in exploration-cycle-plugin (leave as-is until source files
are created at plugins/exploration-cycle-plugin/references/):
agent-loop-patterns.md, exploration-output-standards.md
After plugin-validator, run targeted scripts for detailed checks:
Validate each agent file:
python ${CLAUDE_PLUGIN_ROOT}/scripts/validate_agent.py agents/my-agent.md
Checks: frontmatter structure, required fields (name/description/model/color), name format
(3-50 chars, lowercase + hyphens), description has <example> blocks, system prompt
length (minimum 20 chars, recommended 500-3,000).
Validate hooks.json schema:
python ${CLAUDE_PLUGIN_ROOT}/scripts/validate_hook_schema.py hooks/hooks.json
Checks: JSON syntax, valid event names, each hook has matcher + hooks array,
hook type is command or prompt, command hooks reference existing scripts with
${CLAUDE_PLUGIN_ROOT}.
Test a hook script directly:
python ${CLAUDE_PLUGIN_ROOT}/scripts/test_hook.py \
--hook hooks/scripts/validate.py \
--event PreToolUse \
--input '{"tool_name": "Write", "tool_input": {"file_path": "src/app.py"}}'
Lint hook scripts for common issues:
python ${CLAUDE_PLUGIN_ROOT}/scripts/hook_linter.py hooks/
For issues the scripts may not catch:
Plugin structure check:
# Manifest must be here (not in root)
ls .claude-plugin/plugin.json
# Components must be at root (not in .claude-plugin/)
ls commands/ agents/ skills/ hooks/
# Validate JSON
jq . .claude-plugin/plugin.json
Security scan:
# Check for hardcoded credentials
grep -rn "password\|api_key\|secret\|token" --include="*.md" --include="*.json" --include="*.sh" .
${CLAUDE_PLUGIN_ROOT} portability:
# Ensure no hardcoded paths in hook commands or MCP config
grep -rn "/Users/\|/home/" --include="*.json" --include="*.sh" .
Naming conventions:
my-plugin, not MyPlugin or my_plugin).md.md describing role.sh, .py, .js)Skill quality (run skill-reviewer for each skill):
"Review my skill at skills/skill-name/SKILL.md"
Severity levels:
<example> blocks in agents)Fix critical issues first, then re-validate:
# Re-run validation after fixes
"Validate my plugin at <path>"
Keep running until: 0 critical issues, warnings addressed or documented.
references/*.md. Always consult them (especially ADR 001-006) to verify if the plugin follows our standards for shared scripts, cross-plugin dependencies, symlinking patterns, and loose coupling. A plugin that violates these ADRs (e.g. duplicates shared scripts instead of symlinking) is considered structurally non-compliant..claude-plugin/plugin.json minimal valid:
{ "name": "plugin-name" }
.claude-plugin/plugin.json recommended:
{
"name": "plugin-name",
"version": "0.1.0",
"description": "What the plugin does",
"author": { "name": "Author Name", "email": "email" }
}
Agent description pattern (must have <example> blocks):
description: |
Use this agent when user asks to "do X", "run Y", or mentions Z.
<example>
Context: user just finished creating a plugin
user: "I've set up my plugin"
assistant: "Let me validate the structure."
</example>
Skill description pattern (third-person, anti-undertrigger):
description: >
This skill should be used when the user asks to "X", "Y", or "Z".
Use this skill even when the user doesn't explicitly say "Z" --
mentions of [related concept] should also trigger this.
create-skill, create-command, or create-hook to add missing componentsskill-reviewer on each skill for trigger optimizationaudit-plugin-l5 for advanced red-team structural auditplugin_add.py richfrem/agent-plugins-skillstools
Ingests repository files into the ChromaDB vector store. Builds or updates the vector index from a manifest or directory scan using ingest.py. Use when new files need to be indexed or the vector store is out of date. <example> user: "Index these new plugin files into the vector database" assistant: "I'll use vector-db-ingest to add them to the vector store." </example> <example> user: "The vector store is missing recent files -- update it" assistant: "I'll use vector-db-ingest to re-index the changes." </example>
data-ai
Removes stale and orphaned chunks from the ChromaDB vector store for files that have been deleted or renamed. Use after files are removed or moved to keep the vector index in sync with the filesystem. <example> user: "Clean up the vector store after I deleted some files" assistant: "I'll use vector-db-cleanup to remove orphaned chunks." </example> <example> user: "The vector database has chunks for files that no longer exist" assistant: "I'll run vector-db-cleanup to prune them." </example>
testing
Audit Vector DB coverage -- compares the live filesystem manifest against the ChromaDB index to identify coverage gaps.
development
3-Phase Knowledge Search strategy for the RLM Factory ecosystem. Auto-invoked when tasks involve finding code, documentation, or architecture context in the repository. Enforces the optimal search order: RLM Summary Scan (O(1)) -> Vector DB Semantic Search -> Grep/Exact Match. Never skip phases.