.claude-plugin/PLUGIN_SCHEMA_NOTES.md
This document captures undocumented but enforced constraints of the Claude Code plugin manifest validator.
These rules are based on real installation failures, validator behavior, and comparison with known working plugins. They exist to prevent silent breakage and repeated regressions.
If you edit .claude-plugin/plugin.json, read this first.
The Claude plugin manifest validator is strict and opinionated. It enforces rules that are not fully documented in public schema references.
The most common failure mode is:
The manifest looks reasonable, but the validator rejects it with vague errors like
agents: Invalid input
This document explains why.
version (MANDATORY)The version field is required by the validator even if omitted from some examples.
If missing, installation may fail during marketplace install or CLI validation.
Example:
{
"version": "1.1.0"
}
The following fields must always be arrays:
agentscommandsskillshooks (if present)Even if there is only one entry, strings are not accepted.
{
"agents": "./agents"
}
{
"agents": ["./agents/planner.md"]
}
This applies consistently across all component path fields.
The validator does not accept directory paths for agents.
Even the following will fail:
{
"agents": ["./agents/"]
}
Instead, you must enumerate agent files explicitly:
{
"agents": [
"./agents/planner.md",
"./agents/architect.md",
"./agents/code-reviewer.md"
]
}
This is the most common source of validation errors.
commands and skills accept directory paths only when wrapped in arraysclaude plugin validate is stricter than some marketplace previewsInvalid input) and do not indicate root causeAssume the validator is hostile and literal.
hooks Field: DO NOT ADDWARNING: CRITICAL: Do NOT add a
"hooks"field toplugin.json. This is enforced by a regression test.
Claude Code v2.1+ automatically loads hooks/hooks.json from any installed plugin by convention. If you also declare it in plugin.json, you get:
Duplicate hooks file detected: ./hooks/hooks.json resolves to already-loaded file.
The standard hooks/hooks.json is loaded automatically, so manifest.hooks should
only reference additional hook files.
This has caused repeated fix/revert cycles in this repo:
| Commit | Action | Trigger |
|---|---|---|
22ad036 | ADD hooks | Users reported "hooks not loading" |
a7bc5f2 | REMOVE hooks | Users reported "duplicate hooks error" (#52) |
779085e | ADD hooks | Users reported "agents not loading" (#88) |
e3a1306 | REMOVE hooks | Users reported "duplicate hooks error" (#103) |
Root cause: Claude Code CLI changed behavior between versions:
hooks declarationThe test plugin.json does NOT have explicit hooks declaration in tests/hooks/hooks.test.js prevents this from being reintroduced.
If you're adding additional hook files (not hooks/hooks.json), those CAN be declared. But the standard hooks/hooks.json must NOT be declared.
These look correct but are rejected:
agentsversion"hooks": "./hooks/hooks.json" - auto-loaded by convention, causes duplicate errorAvoid cleverness. Be explicit.
{
"version": "1.1.0",
"agents": [
"./agents/planner.md",
"./agents/code-reviewer.md"
],
"commands": ["./commands/"],
"skills": ["./skills/"]
}
This structure has been validated against the Claude plugin validator.
Important: Notice there is NO "hooks" field. The hooks/hooks.json file is loaded automatically by convention. Adding it explicitly causes a duplicate error.
Before submitting changes that touch plugin.json:
versionclaude plugin validate .claude-plugin/plugin.json
If in doubt, choose verbosity over convenience.
This repository is widely forked and used as a reference implementation.
Documenting validator quirks here:
If the validator changes, update this document first.