A plugin marketplace is a catalog that lets you distribute plugins to others. Marketplaces provide centralized discovery, version tracking, automatic updates, and support for multiple source types (git repositories, local paths, and more). This guide shows you how to create your own marketplace to share plugins with your team or community. Looking to install plugins from an existing marketplace? See Discover and install prebuilt plugins.
Overview
Creating and distributing a marketplace involves:
- Creating plugins: build one or more plugins with commands, agents, hooks, MCP servers, or LSP servers. This guide assumes you already have plugins to distribute; see Create plugins for details on how to create them.
- Creating a marketplace file: define a
marketplace.jsonthat lists your plugins and where to find them (see Create the marketplace file). - Host the marketplace: push to GitHub, GitLab, or another git host (see Host and distribute marketplaces).
- Share with users: users add your marketplace with
/plugin marketplace addand install individual plugins (see Discover and install plugins).
Once your marketplace is live, you can update it by pushing changes to your repository. Users refresh their local copy with /plugin marketplace update.
Walkthrough: create a local marketplace
This example creates a marketplace with one plugin: a /review skill for code reviews. You’ll create the directory structure, add a skill, create the plugin manifest and marketplace catalog, then install and test it.
1
2
3
4
5
6
To learn more about what plugins can do, including hooks, agents, MCP servers, and LSP servers, see Plugins.
Create the marketplace file
Create .claude-plugin/marketplace.json in your repository root. This file defines your marketplace’s name, owner information, and a list of plugins with their sources.
Each plugin entry needs at minimum a name and source (where to fetch it from). See the full schema below for all available fields.
{
"name": "company-tools",
"owner": {
"name": "DevTools Team",
"email": "[email protected]"
},
"plugins": [
{
"name": "code-formatter",
"source": "./plugins/formatter",
"description": "Automatic code formatting on save",
"version": "2.1.0",
"author": {
"name": "DevTools Team"
}
},
{
"name": "deployment-tools",
"source": {
"source": "github",
"repo": "company/deploy-plugin"
},
"description": "Deployment automation tools"
}
]
}
Marketplace schema
Required fields
| Field | Type | Description | Example |
|---|---|---|---|
name | string | Marketplace identifier (kebab-case, no spaces). This is public-facing: users see it when installing plugins (for example, /plugin install my-tool@your-marketplace). | "acme-tools" |
owner | object | Marketplace maintainer information (see fields below) | |
plugins | array | List of available plugins | See below |
Owner fields
| Field | Type | Required | Description |
|---|---|---|---|
name | string | Yes | Name of the maintainer or team |
email | string | No | Contact email for the maintainer |
Optional metadata
| Field | Type | Description |
|---|---|---|
metadata.description | string | Brief marketplace description |
metadata.version | string | Marketplace version |
metadata.pluginRoot | string | Base directory prepended to relative plugin source paths (for example, "./plugins" lets you write "source": "formatter" instead of "source": "./plugins/formatter") |
Plugin entries
Each plugin entry in the plugins array describes a plugin and where to find it. You can include any field from the plugin manifest schema (like description, version, author, commands, hooks, etc.), plus these marketplace-specific fields: source, category, tags, and strict.
Required fields
| Field | Type | Description |
|---|---|---|
name | string | Plugin identifier (kebab-case, no spaces). This is public-facing: users see it when installing (for example, /plugin install my-plugin@marketplace). |
source | string|object | Where to fetch the plugin from (see Plugin sources below) |
Optional plugin fields
Standard metadata fields:
| Field | Type | Description |
|---|---|---|
description | string | Brief plugin description |
version | string | Plugin version |
author | object | Plugin author information (name required, email optional) |
homepage | string | Plugin homepage or documentation URL |
repository | string | Source code repository URL |
license | string | SPDX license identifier (for example, MIT, Apache-2.0) |
keywords | array | Tags for plugin discovery and categorization |
category | string | Plugin category for organization |
tags | array | Tags for searchability |
strict | boolean | Controls whether plugin.json is the authority for component definitions (default: true). See Strict mode below. |
Component configuration fields:
| Field | Type | Description |
|---|---|---|
commands | string|array | Custom paths to command files or directories |
agents | string|array | Custom paths to agent files |
hooks | string|object | Custom hooks configuration or path to hooks file |
mcpServers | string|object | MCP server configurations or path to MCP config |
lspServers | string|object | LSP server configurations or path to LSP config |
Plugin sources
Plugin sources tell Claude Code where to fetch each individual plugin listed in your marketplace. These are set in the source field of each plugin entry in marketplace.json.
Once a plugin is cloned or copied into the local machine, it is copied into the local versioned plugin cache at ~/.claude/plugins/cache.
| Source | Type | Fields | Notes |
|---|---|---|---|
| Relative path | string (e.g. "./my-plugin") | — | Local directory within the marketplace repo. Must start with ./ |
github | object | repo, ref?, sha? | |
url | object | url (must end .git), ref?, sha? | Git URL source |
npm | object | package, version?, registry? | Installed via npm install |
pip | object | package, version?, registry? | Installed via pip |
Relative paths
For plugins in the same repository:
{
"name": "my-plugin",
"source": "./plugins/my-plugin"
}
GitHub repositories
{
"name": "github-plugin",
"source": {
"source": "github",
"repo": "owner/plugin-repo"
}
}
You can pin to a specific branch, tag, or commit:
{
"name": "github-plugin",
"source": {
"source": "github",
"repo": "owner/plugin-repo",
"ref": "v2.0.0",
"sha": "a1b2c3d4e5f6a7b8c9d0e1f2a3b4c5d6e7f8a9b0"
}
}
| Field | Type | Description |
|---|---|---|
repo | string | Required. GitHub repository in owner/repo format |
ref | string | Optional. Git branch or tag (defaults to repository default branch) |
sha | string | Optional. Full 40-character git commit SHA to pin to an exact version |
Git repositories
{
"name": "git-plugin",
"source": {
"source": "url",
"url": "https://gitlab.com/team/plugin.git"
}
}
You can pin to a specific branch, tag, or commit:
{
"name": "git-plugin",
"source": {
"source": "url",
"url": "https://gitlab.com/team/plugin.git",
"ref": "main",
"sha": "a1b2c3d4e5f6a7b8c9d0e1f2a3b4c5d6e7f8a9b0"
}
}
| Field | Type | Description |
|---|---|---|
url | string | Required. Full git repository URL (must end with .git) |
ref | string | Optional. Git branch or tag (defaults to repository default branch) |
sha | string | Optional. Full 40-character git commit SHA to pin to an exact version |
npm packages
Plugins distributed as npm packages are installed using npm install. This works with any package on the public npm registry or a private registry your team hosts.
{
"name": "my-npm-plugin",
"source": {
"source": "npm",
"package": "@acme/claude-plugin"
}
}
To pin to a specific version, add the version field:
{
"name": "my-npm-plugin",
"source": {
"source": "npm",
"package": "@acme/claude-plugin",
"version": "2.1.0"
}
}
To install from a private or internal registry, add the registry field:
{
"name": "my-npm-plugin",
"source": {
"source": "npm",
"package": "@acme/claude-plugin",
"version": "^2.0.0",
"registry": "https://npm.example.com"
}
}
| Field | Type | Description |
|---|---|---|
package | string | Required. Package name or scoped package (for example, @org/plugin) |
version | string | Optional. Version or version range (for example, 2.1.0, ^2.0.0, ~1.5.0) |
registry | string | Optional. Custom npm registry URL. Defaults to the system npm registry (typically npmjs.org) |
Advanced plugin entries
This example shows a plugin entry using many of the optional fields, including custom paths for commands, agents, hooks, and MCP servers:
{
"name": "enterprise-tools",
"source": {
"source": "github",
"repo": "company/enterprise-plugin"
},
"description": "Enterprise workflow automation tools",
"version": "2.1.0",
"author": {
"name": "Enterprise Team",
"email": "[email protected]"
},
"homepage": "https://docs.example.com/plugins/enterprise-tools",
"repository": "https://github.com/company/enterprise-plugin",
"license": "MIT",
"keywords": ["enterprise", "workflow", "automation"],
"category": "productivity",
"commands": [
"./commands/core/",
"./commands/enterprise/",
"./commands/experimental/preview.md"
],
"agents": ["./agents/security-reviewer.md", "./agents/compliance-checker.md"],
"hooks": {
"PostToolUse": [
{
"matcher": "Write|Edit",
"hooks": [
{
"type": "command",
"command": "${CLAUDE_PLUGIN_ROOT}/scripts/validate.sh"
}
]
}
]
},
"mcpServers": {
"enterprise-db": {
"command": "${CLAUDE_PLUGIN_ROOT}/servers/db-server",
"args": ["--config", "${CLAUDE_PLUGIN_ROOT}/config.json"]
}
},
"strict": false
}
Key things to notice:
commandsandagents: You can specify multiple directories or individual files. Paths are relative to the plugin root.${CLAUDE_PLUGIN_ROOT}: Use this variable in hooks and MCP server configs to reference files within the plugin’s installation directory. This is necessary because plugins are copied to a cache location when installed.strict: false: Since this is set to false, the plugin doesn’t need its ownplugin.json. The marketplace entry defines everything. See Strict mode below.
Strict mode
The strict field controls whether plugin.json is the authority for component definitions (commands, agents, hooks, skills, MCP servers, output styles).
| Value | Behavior |
|---|---|
true (default) | plugin.json is the authority. The marketplace entry can supplement it with additional components, and both sources are merged. |
false | The marketplace entry is the entire definition. If the plugin also has a plugin.json that declares components, that’s a conflict and the plugin fails to load. |
When to use each mode:
strict: true: the plugin has its ownplugin.jsonand manages its own components. The marketplace entry can add extra commands or hooks on top. This is the default and works for most plugins.strict: false: the marketplace operator wants full control. The plugin repo provides raw files, and the marketplace entry defines which of those files are exposed as commands, agents, hooks, etc. Useful when the marketplace restructures or curates a plugin’s components differently than the plugin author intended.
Host and distribute marketplaces
Host on GitHub (recommended)
GitHub provides the easiest distribution method:
- Create a repository: Set up a new repository for your marketplace
- Add marketplace file: Create
.claude-plugin/marketplace.jsonwith your plugin definitions - Share with teams: Users add your marketplace with
/plugin marketplace add owner/repo
Benefits: Built-in version control, issue tracking, and team collaboration features.
Host on other git services
Any git hosting service works, such as GitLab, Bitbucket, and self-hosted servers. Users add with the full repository URL:
/plugin marketplace add https://gitlab.com/company/plugins.git
Private repositories
Claude Code supports installing plugins from private repositories. For manual installation and updates, Claude Code uses your existing git credential helpers. If git clone works for a private repository in your terminal, it works in Claude Code too. Common credential helpers include gh auth login for GitHub, macOS Keychain, and git-credential-store.
Background auto-updates run at startup without credential helpers, since interactive prompts would block Claude Code from starting. To enable auto-updates for private marketplaces, set the appropriate authentication token in your environment:
| Provider | Environment variables | Notes |
|---|---|---|
| GitHub | GITHUB_TOKEN or GH_TOKEN | Personal access token or GitHub App token |
| GitLab | GITLAB_TOKEN or GL_TOKEN | Personal access token or project token |
| Bitbucket | BITBUCKET_TOKEN | App password or repository access token |
Set the token in your shell configuration (for example, .bashrc, .zshrc) or pass it when running Claude Code:
export GITHUB_TOKEN=ghp_xxxxxxxxxxxxxxxxxxxx
Test locally before distribution
Test your marketplace locally before sharing:
/plugin marketplace add ./my-local-marketplace
/plugin install test-plugin@my-local-marketplace
For the full range of add commands (GitHub, Git URLs, local paths, remote URLs), see Add marketplaces.
Require marketplaces for your team
You can configure your repository so team members are automatically prompted to install your marketplace when they trust the project folder. Add your marketplace to .claude/settings.json:
{
"extraKnownMarketplaces": {
"company-tools": {
"source": {
"source": "github",
"repo": "your-org/claude-plugins"
}
}
}
}
You can also specify which plugins should be enabled by default:
{
"enabledPlugins": {
"code-formatter@company-tools": true,
"deployment-tools@company-tools": true
}
}
For full configuration options, see Plugin settings.
Managed marketplace restrictions
For organizations requiring strict control over plugin sources, administrators can restrict which plugin marketplaces users are allowed to add using the strictKnownMarketplaces setting in managed settings.
When strictKnownMarketplaces is configured in managed settings, the restriction behavior depends on the value:
| Value | Behavior |
|---|---|
| Undefined (default) | No restrictions. Users can add any marketplace |
Empty array [] | Complete lockdown. Users cannot add any new marketplaces |
| List of sources | Users can only add marketplaces that match the allowlist exactly |
Common configurations
Disable all marketplace additions:
{
"strictKnownMarketplaces": []
}
Allow specific marketplaces only:
{
"strictKnownMarketplaces": [
{
"source": "github",
"repo": "acme-corp/approved-plugins"
},
{
"source": "github",
"repo": "acme-corp/security-tools",
"ref": "v2.0"
},
{
"source": "url",
"url": "https://plugins.example.com/marketplace.json"
}
]
}
Allow all marketplaces from an internal git server using regex pattern matching:
{
"strictKnownMarketplaces": [
{
"source": "hostPattern",
"hostPattern": "^github\\.example\\.com$"
}
]
}
How restrictions work
Restrictions are validated early in the plugin installation process, before any network requests or filesystem operations occur. This prevents unauthorized marketplace access attempts. The allowlist uses exact matching for most source types. For a marketplace to be allowed, all specified fields must match exactly:
- For GitHub sources:
repois required, andreforpathmust also match if specified in the allowlist - For URL sources: the full URL must match exactly
- For
hostPatternsources: the marketplace host is matched against the regex pattern
Because strictKnownMarketplaces is set in managed settings, individual users and project configurations cannot override these restrictions.
For complete configuration details including all supported source types and comparison with extraKnownMarketplaces, see the strictKnownMarketplaces reference.
Version resolution and release channels
Plugin versions determine cache paths and update detection. You can specify the version in the plugin manifest (plugin.json) or in the marketplace entry (marketplace.json).
Set up release channels
To support “stable” and “latest” release channels for your plugins, you can set up two marketplaces that point to different refs or SHAs of the same repo. You can then assign the two marketplaces to different user groups through managed settings.
Example
{
"name": "stable-tools",
"plugins": [
{
"name": "code-formatter",
"source": {
"source": "github",
"repo": "acme-corp/code-formatter",
"ref": "stable"
}
}
]
}
{
"name": "latest-tools",
"plugins": [
{
"name": "code-formatter",
"source": {
"source": "github",
"repo": "acme-corp/code-formatter",
"ref": "latest"
}
}
]
}
Assign channels to user groups
Assign each marketplace to the appropriate user group through managed settings. For example, the stable group receives:
{
"extraKnownMarketplaces": {
"stable-tools": {
"source": {
"source": "github",
"repo": "acme-corp/stable-tools"
}
}
}
}
The early-access group receives latest-tools instead:
{
"extraKnownMarketplaces": {
"latest-tools": {
"source": {
"source": "github",
"repo": "acme-corp/latest-tools"
}
}
}
}
Validation and testing
Test your marketplace before sharing. Validate your marketplace JSON syntax:
claude plugin validate .
Or from within Claude Code:
/plugin validate .
Add the marketplace for testing:
/plugin marketplace add ./path/to/marketplace
Install a test plugin to verify everything works:
/plugin install test-plugin@marketplace-name
For complete plugin testing workflows, see Test your plugins locally. For technical troubleshooting, see Plugins reference.
Troubleshooting
Marketplace not loading
Symptoms: Can’t add marketplace or see plugins from it Solutions:
- Verify the marketplace URL is accessible
- Check that
.claude-plugin/marketplace.jsonexists at the specified path - Ensure JSON syntax is valid using
claude plugin validateor/plugin validate - For private repositories, confirm you have access permissions
Marketplace validation errors
Run claude plugin validate . or /plugin validate . from your marketplace directory to check for issues. Common errors:
| Error | Cause | Solution |
|---|---|---|
File not found: .claude-plugin/marketplace.json | Missing manifest | Create .claude-plugin/marketplace.json with required fields |
Invalid JSON syntax: Unexpected token... | JSON syntax error | Check for missing commas, extra commas, or unquoted strings |
Duplicate plugin name "x" found in marketplace | Two plugins share the same name | Give each plugin a unique name value |
plugins[0].source: Path traversal not allowed | Source path contains .. | Use paths relative to marketplace root without .. |
Warnings (non-blocking):
Marketplace has no plugins defined: add at least one plugin to thepluginsarrayNo marketplace description provided: addmetadata.descriptionto help users understand your marketplace
Plugin installation failures
Symptoms: Marketplace appears but plugin installation fails Solutions:
- Verify plugin source URLs are accessible
- Check that plugin directories contain required files
- For GitHub sources, ensure repositories are public or you have access
- Test plugin sources manually by cloning/downloading
Private repository authentication fails
Symptoms: Authentication errors when installing plugins from private repositories Solutions: For manual installation and updates:
- Verify you’re authenticated with your git provider (for example, run
gh auth statusfor GitHub) - Check that your credential helper is configured correctly:
git config --global credential.helper - Try cloning the repository manually to verify your credentials work
For background auto-updates:
- Set the appropriate token in your environment:
echo $GITHUB_TOKEN - Check that the token has the required permissions (read access to the repository)
- For GitHub, ensure the token has the
reposcope for private repositories - For GitLab, ensure the token has at least
read_repositoryscope - Verify the token hasn’t expired
Git operations time out
Symptoms: Plugin installation or marketplace updates fail with a timeout error like “Git clone timed out after 120s” or “Git pull timed out after 120s”.
Cause: Claude Code uses a 120-second timeout for all git operations, including cloning plugin repositories and pulling marketplace updates. Large repositories or slow network connections may exceed this limit.
Solution: Increase the timeout using the CLAUDE_CODE_PLUGIN_GIT_TIMEOUT_MS environment variable. The value is in milliseconds:
export CLAUDE_CODE_PLUGIN_GIT_TIMEOUT_MS=300000 # 5 minutes
Plugins with relative paths fail in URL-based marketplaces
Symptoms: Added a marketplace via URL (such as https://example.com/marketplace.json), but plugins with relative path sources like "./plugins/my-plugin" fail to install with “path not found” errors.
Cause: URL-based marketplaces only download the marketplace.json file itself. They do not download plugin files from the server. Relative paths in the marketplace entry reference files on the remote server that were not downloaded.
Solutions:
- Use external sources: Change plugin entries to use GitHub, npm, or git URL sources instead of relative paths:
{ "name": "my-plugin", "source": { "source": "github", "repo": "owner/repo" } } - Use a Git-based marketplace: Host your marketplace in a Git repository and add it with the git URL. Git-based marketplaces clone the entire repository, making relative paths work correctly.
Files not found after installation
Symptoms: Plugin installs but references to files fail, especially files outside the plugin directory
Cause: Plugins are copied to a cache directory rather than used in-place. Paths that reference files outside the plugin’s directory (such as ../shared-utils) won’t work because those files aren’t copied.
Solutions: See Plugin caching and file resolution for workarounds including symlinks and directory restructuring.
For additional debugging tools and common issues, see Debugging and development tools.
See also
- Discover and install prebuilt plugins - Installing plugins from existing marketplaces
- Plugins - Creating your own plugins
- Plugins reference - Complete technical specifications and schemas
- Plugin settings - Plugin configuration options
- strictKnownMarketplaces reference - Managed marketplace restrictions