| name | PRPM JSON Best Practices |
| description | Best practices for structuring prpm.json package manifests with required fields, tags, organization, and multi-package management |
| author | PRPM Team |
| version | 1.0.0 |
| tags | prpm, package-management, json, manifest, best-practices, publishing |
PRPM JSON Best Practices
You are an expert at creating and maintaining prpm.json package manifests for PRPM (Prompt Package Manager). You understand the structure, required fields, organization patterns, and best practices for multi-package repositories.
When to Apply This Skill
Use when:
- Creating a new
prpm.jsonmanifest for publishing packages - Maintaining existing
prpm.jsonfiles - Organizing multi-package repositories
- Adding or updating package metadata
- Ensuring package manifest quality and completeness
Don't use for:
- User configuration files (
.prpmrc) - those are for users - Lockfiles (
prpm.lock) - those are auto-generated by PRPM - Regular package installation (users don't need
prpm.json) - Dependencies already tracked in lockfiles
Core Purpose
prpm.json is only needed if you're publishing packages. Regular users installing packages from the registry don't need this file.
Use prpm.json when you're:
- Publishing a package to the PRPM registry
- Creating a collection of packages
- Distributing your own prompts/rules/skills/agents
- Managing multiple related packages in a monorepo
File Structure
Single Package
See examples/single-package.json for complete structure.
Key fields: name, version, description, author, license, format, subtype, files
Multi-Package Repository
See examples/multi-package.json for complete structure.
Use when: Publishing multiple related packages from one repo
Key difference: Top-level packages array with individual package definitions
Collections Repository
See examples/collections-repository.json for complete structure.
Use when: Bundling existing published packages into curated collections Key points:
collectionsarray references packages bypackageId(not files)- Each collection has
id,name,description,packages - Packages can be
required: true(default) orfalse(optional) - Use version ranges (
^1.0.0) orlatest - Add
reasonto explain why package is included
Packages + Collections (Combined)
See examples/packages-with-collections.json for complete structure.
Use when: Publishing packages AND creating collections that bundle them Key points:
- Define packages in
packagesarray with files - Define collections in
collectionsarray referencing those packages - Collections can reference both local packages and external ones
- Publish both individual packages and collection bundles from same repo
Required Fields
Top-Level (Single Package)
| Field | Type | Required | Description |
|---|---|---|---|
name |
string | Yes | Package name (kebab-case, unique in registry) |
version |
string | Yes | Semver version (e.g., 1.0.0) |
description |
string | Yes | Clear description of what the package does |
author |
string | Yes | Author name and optional email |
license |
string | Yes | SPDX license identifier (e.g., MIT, Apache-2.0) |
format |
string | Yes | Target format: claude, cursor, continue, windsurf, etc. |
subtype |
string | Yes | Package type: agent, skill, rule, slash-command, prompt, collection |
files |
string[] | Yes | Array of files to include in package |
Optional Top-Level Fields
| Field | Type | Description |
|---|---|---|
repository |
string | Git repository URL |
organization |
string | Organization name (for scoped packages) |
homepage |
string | Package homepage URL |
documentation |
string | Documentation URL |
tags |
string[] | Searchable tags (kebab-case) |
keywords |
string[] | Additional keywords for search |
category |
string | Package category |
private |
boolean | If true, won't be published to public registry |
dependencies |
object | Package dependencies (name: semver) |
Multi-Package Fields
When using packages array:
| Field | Type | Required | Description |
|---|---|---|---|
name |
string | Yes | Unique package name |
version |
string | Yes | Package version |
description |
string | Yes | Package description |
format |
string | Yes | Package format |
subtype |
string | Yes | Package subtype |
tags |
string[] | Recommended | Searchable tags |
files |
string[] | Yes | Files to include |
private |
boolean | No | Mark as private |
Collection Fields
When using collections array:
Top-level (repository with collections):
name,version,description,author,license- Requiredrepository,organization- Recommended- Note: No
format,subtype, orfilesrequired at top level
Each collection object:
| Field | Type | Required | Description |
|---|---|---|---|
id |
string | Yes | Unique collection identifier (kebab-case, 3-100 chars) |
name |
string | Yes | Display name (3-100 chars) |
description |
string | Yes | What the collection provides (10-500 chars) |
packages |
array | Yes | Array of packages to include (minimum 1) |
version |
string | Recommended | Semantic version of collection |
category |
string | Recommended | Collection category (development, testing, etc.) |
tags |
string[] | Recommended | Searchable tags (kebab-case, 1-10 items) |
icon |
string | Optional | Emoji or icon (max 10 chars) |
Each package within collection:
| Field | Type | Required | Description |
|---|---|---|---|
packageId |
string | Yes | Package to include |
version |
string | Optional | Version range (^1.0.0, ~2.1.0, 1.0.0, latest) |
required |
boolean | Optional | Whether package is required (default: true) |
reason |
string | Optional | Why package is included (max 200 chars) |
Format and Subtype Values
Format (Target AI Tool)
| Format | Description |
|---|---|
claude |
Claude Code (agents, skills) |
cursor |
Cursor IDE (rules, MDC files) |
continue |
Continue.dev extension |
windsurf |
Windsurf IDE |
copilot |
GitHub Copilot |
kiro |
Kiro IDE |
agents.md |
Agents.md format |
generic |
Generic/universal format |
mcp |
Model Context Protocol |
Subtype (Package Type)
| Subtype | Description | Typical Formats |
|---|---|---|
agent |
Autonomous agents | claude, agents.md |
skill |
Specialized capabilities | claude |
rule |
IDE rules and guidelines | cursor, windsurf |
slash-command |
Slash commands | cursor, continue |
prompt |
Prompt templates | generic |
collection |
Package collections | Any |
chatmode |
Chat modes | kiro |
tool |
MCP tools | mcp |
Tags Best Practices
Tag Structure
- Use kebab-case for all tags
- Be specific and searchable
- Include 3-8 tags per package
- Combine technology, domain, and purpose tags
Tag Categories
Technology Tags:
- Languages:
typescript,python,javascript,rust - Frameworks:
react,nextjs,fastify,django - Tools:
aws,docker,kubernetes,postgresql
Domain Tags:
deployment,testing,ci-cd,databaseinfrastructure,cloud,monitoringdocumentation,code-review,security
Purpose Tags:
troubleshooting,debugging,best-practicesautomation,quality-assurance,performancearchitecture,design-patterns
Meta Tags:
meta- For packages about creating packagesprpm-internal- For internal/private packagesprpm-development- For PRPM development itself
Tag Examples
Good Tags:
{
"tags": [
"typescript",
"type-safety",
"code-quality",
"best-practices",
"static-analysis"
]
}
Poor Tags:
{
"tags": [
"code", // Too generic
"stuff", // Meaningless
"TypeScript", // Wrong case
"type_safety" // Wrong format (use kebab-case)
]
}
Organization Best Practices
Multi-Package Organization
Order packages by:
- Privacy - Private packages first
- Format - Group by format (claude, cursor, etc.)
- Subtype - Group by subtype (agent, skill, rule)
Example organization:
{
"packages": [
// Private > Claude > Agents
{ "name": "internal-agent", "private": true, "format": "claude", "subtype": "agent" },
// Private > Claude > Skills
{ "name": "internal-skill", "private": true, "format": "claude", "subtype": "skill" },
// Private > Cursor > Rules
{ "name": "internal-rule", "private": true, "format": "cursor", "subtype": "rule" },
// Public > Claude > Skills
{ "name": "public-skill", "format": "claude", "subtype": "skill" },
// Public > Cursor > Rules
{ "name": "public-rule", "format": "cursor", "subtype": "rule" }
]
}
Naming Conventions
Package Names:
- Use kebab-case:
my-awesome-skill - Be descriptive:
typescript-type-safetynotts-types - Avoid duplicates across formats: use suffixes if needed
format-conversion-agent(Claude agent)format-conversion(Cursor rule)
File Paths:
- Use full paths from project root (where prpm.json lives)
- Agents:
.claude/agents/name.md - Skills:
.claude/skills/name/SKILL.md - Rules:
.cursor/rules/name.mdc - Commands:
.claude/commands/category/name.md
Version Management
Semver Guidelines
Follow semantic versioning:
- Major (1.0.0 → 2.0.0): Breaking changes
- Minor (1.0.0 → 1.1.0): New features, backward compatible
- Patch (1.0.0 → 1.0.1): Bug fixes, backward compatible
Version Bumping
When to bump versions:
- Patch: Bug fixes, typo corrections, minor improvements
- Minor: New sections, additional examples, new features
- Major: Complete rewrites, breaking changes, renamed fields
Keep Versions in Sync
For multi-package repos, keep related packages in sync:
{
"packages": [
{ "name": "pkg-one", "version": "1.2.0" },
{ "name": "pkg-two", "version": "1.2.0" },
{ "name": "pkg-three", "version": "1.2.0" }
]
}
File Management
Files Array
CRITICAL: File paths must be full paths from project root (where prpm.json lives).
Required:
- List all files to include in the package
- Use full paths from project root - not relative to destination directories
- Paths should start with
.claude/,.cursor/, etc. - Include documentation files
Why Full Paths?
File paths in prpm.json are used for:
- Tarball creation - Reads files directly from these paths
- Snippet extraction - Shows file preview before install
- Installation - CLI derives destination from format/subtype
Examples:
Claude agent (single file):
{
"format": "claude",
"subtype": "agent",
"files": [".claude/agents/my-agent.md"]
}
Claude skill (multiple files):
{
"format": "claude",
"subtype": "skill",
"files": [
".claude/skills/my-skill/SKILL.md",
".claude/skills/my-skill/EXAMPLES.md",
".claude/skills/my-skill/README.md"
]
}
Cursor rule:
{
"format": "cursor",
"subtype": "rule",
"files": [".cursor/rules/my-rule.mdc"]
}
Slash command:
{
"format": "claude",
"subtype": "slash-command",
"files": [".claude/commands/category/my-command.md"]
}
Common Mistake:
{
// ❌ WRONG - Relative paths without directory prefix
"files": ["agents/my-agent.md"] // Will fail to find file
// ✅ CORRECT - Full path from project root
"files": [".claude/agents/my-agent.md"]
}
File Verification
Always verify files exist:
# Check all files in prpm.json exist
for file in $(cat prpm.json | jq -r '.packages[].files[]'); do
if [ ! -f "$file" ]; then
echo "Missing: $file"
fi
done
Duplicate Detection
Check for Duplicate Names
Run this check before committing:
# Check for duplicate package names
cat prpm.json | jq -r '.packages[].name' | sort | uniq -d
If output is empty, no duplicates exist. If names appear, you have duplicates to resolve.
Resolving Duplicates
Bad:
{
"packages": [
{ "name": "typescript-safety", "format": "claude" },
{ "name": "typescript-safety", "format": "cursor" }
]
}
Good:
{
"packages": [
{ "name": "typescript-safety", "format": "claude", "subtype": "skill" },
{ "name": "typescript-safety-rule", "format": "cursor", "subtype": "rule" }
]
}
Common Patterns
Private Internal Packages
{
"name": "internal-tool",
"version": "1.0.0",
"description": "Internal development tool",
"private": true,
"format": "claude",
"subtype": "skill",
"tags": ["prpm-internal", "development"],
"files": [".claude/skills/internal-tool/SKILL.md"]
}
Meta Packages (Creating Other Packages)
{
"name": "creating-skills",
"version": "1.0.0",
"description": "Guide for creating effective Claude Code skills",
"format": "claude",
"subtype": "skill",
"tags": ["meta", "claude-code", "skills", "documentation", "best-practices"],
"files": [".claude/skills/creating-skills/SKILL.md"]
}
Cross-Format Packages
When you have the same content for multiple formats:
{
"packages": [
{
"name": "format-conversion-agent",
"format": "claude",
"subtype": "agent",
"description": "Agent for converting between AI prompt formats",
"files": [".claude/agents/format-conversion.md"]
},
{
"name": "format-conversion",
"format": "cursor",
"subtype": "rule",
"description": "Rule for converting between AI prompt formats",
"files": [".cursor/rules/format-conversion.mdc"]
}
]
}
Collections in prpm.json
Collections CAN be defined in prpm.json alongside packages using the collections array. Collections bundle multiple packages together for easier installation.
Example with both packages and collections:
{
"name": "my-prompts-repo",
"author": "Your Name",
"license": "MIT",
"packages": [
{
"name": "typescript-rules",
"version": "1.0.0",
"description": "TypeScript best practices",
"format": "cursor",
"subtype": "rule",
"tags": ["typescript"],
"files": [".cursor/rules/typescript.mdc"]
}
],
"collections": [
{
"id": "my-dev-setup",
"name": "My Development Setup",
"description": "Complete development setup with TypeScript and React",
"version": "1.0.0",
"category": "development",
"tags": ["typescript", "react"],
"packages": [
{
"packageId": "typescript-strict",
"version": "^1.0.0",
"required": true,
"reason": "Enforces strict TypeScript type safety"
},
{
"packageId": "react-best-practices",
"version": "^2.0.0",
"required": true
}
]
}
]
}
For more details on creating collections, see the PRPM documentation at https://docs.prpm.dev or run prpm help collections.
Summary: prpm.json can contain both packages (skills, agents, rules, slash-commands, etc.) and collections.
Validation Checklist
Before publishing, verify:
Required Fields:
- All packages have
name,version,description - All packages have
formatandsubtype - All packages have
filesarray - Top-level has
authorandlicense
File Verification:
- All files in
filesarrays exist - File paths are relative to repo root
- No missing or broken file references
No Duplicates:
- No duplicate package names
- Package names are unique across entire manifest
Tags:
- Tags use kebab-case
- 3-8 relevant tags per package
- Tags include technology, domain, and purpose
Organization:
- Private packages listed first
- Packages grouped by format and subtype
- Consistent versioning across related packages
Lockfile Management
Understanding prpm.lock
The prpm.lock file is auto-generated and tracks installed packages. It serves as the source of truth for what's installed in your project.
IMPORTANT: Do NOT add packages to prpm.json if they already exist in prpm.lock:
prpm.locktracks installed dependencies (packages you use)prpm.jsondefines published packages (packages you create and share)
When to Use prpm.json vs prpm.lock
Use prpm.json when:
- You're creating a package to publish to the registry
- You want to define metadata for YOUR packages
- You're setting up a multi-package repository
Use prpm.lock (auto-generated) when:
- You install packages with
prpm install - You want to track which packages are installed
- You want reproducible installations across environments
Common Mistake: Duplicating Dependencies
❌ WRONG - Don't add installed packages to prpm.json:
// prpm.json
{
"name": "my-project",
"packages": [
{
"name": "typescript-safety", // ❌ This is an INSTALLED package
"version": "1.0.0",
"format": "cursor",
"subtype": "rule",
"files": [".cursor/rules/typescript-safety.mdc"]
}
]
}
// prpm.lock (auto-generated)
{
"packages": {
"@prpm/typescript-safety": { // ✅ Already tracked here
"version": "1.0.0",
"format": "cursor",
"subtype": "rule"
}
}
}
✅ CORRECT - prpm.json only for YOUR packages:
// prpm.json - Only YOUR packages you're publishing
{
"name": "my-project",
"packages": [
{
"name": "my-custom-rule", // ✅ This is YOUR package
"version": "1.0.0",
"format": "cursor",
"subtype": "rule",
"files": [".cursor/rules/my-custom-rule.mdc"]
}
]
}
// prpm.lock - Installed dependencies (auto-generated)
{
"packages": {
"@prpm/typescript-safety": { // ✅ Installed from registry
"version": "1.0.0",
"format": "cursor",
"subtype": "rule"
}
}
}
Key Principles
- Lockfile is Auto-Generated - Never manually edit
prpm.lock - Separation of Concerns:
prpm.json= What you PUBLISHprpm.lock= What you INSTALL
- Check Lockfile First - Before adding to
prpm.json, check if it's already inprpm.lock - Trust the Lockfile - It's the authoritative record of installed packages
Workflow Example
# Install a package (updates prpm.lock automatically)
prpm install @prpm/typescript-safety
# This creates/updates prpm.lock - DO NOT add to prpm.json!
# Only create prpm.json entries for packages YOU create:
# 1. Create your custom rule/skill/agent
# 2. Add entry to prpm.json
# 3. Publish with: prpm publish
Publishing Workflow
1. Validate Manifest
# Validate JSON syntax
cat prpm.json | jq . > /dev/null
# Check for duplicates
cat prpm.json | jq -r '.packages[].name' | sort | uniq -d
# Verify files exist
# (see File Verification section)
2. Bump Versions
Update version numbers for changed packages.
3. Test Locally
# Test package installation
prpm install . --dry-run
4. Publish
# Publish all packages
prpm publish
# Or publish specific package
prpm publish --package my-skill
Common Mistakes to Avoid
❌ Missing Required Fields
{
"name": "my-skill",
// Missing: version, description, format, subtype, files
}
❌ Wrong Tag Format
{
"tags": ["TypeScript", "Code_Quality", "bestPractices"]
// Should be: ["typescript", "code-quality", "best-practices"]
}
❌ Duplicate Names
{
"packages": [
{ "name": "my-skill", "format": "claude" },
{ "name": "my-skill", "format": "cursor" }
// Second should be: "my-skill-rule" or similar
]
}
❌ Missing Files
{
"files": [".claude/skills/my-skill/SKILL.md"]
// But .claude/skills/my-skill/SKILL.md doesn't exist in the repo
}
❌ Absolute Paths
{
"files": ["/Users/me/project/.claude/skills/my-skill/SKILL.md"]
// Should be: ".claude/skills/my-skill/SKILL.md" (relative to project root)
}
❌ Missing Directory Prefix
{
"files": ["agents/my-agent.md"]
// Should be: ".claude/agents/my-agent.md" (include .claude/ prefix)
}
Remember
prpm.jsonis only for publishing YOUR packages/collections, not for installed dependencies- Never add packages from
prpm.locktoprpm.json- they serve different purposes prpm.locktracks what you INSTALL,prpm.jsondefines what you PUBLISH- Use
collectionsarray to bundle existing packages (references by packageId) - Use
packagesarray to define packages with files - Can combine both
packagesandcollectionsin same repo - Always validate before committing
- Keep versions in sync for related packages
- Use consistent, searchable tags
- Verify all file paths exist
- Check for duplicate names
- Follow semver for version management
Goal: Create maintainable, well-organized package manifests and curated collections that are easy to publish and discover in the PRPM registry.