Skip to content

Manifest Schema

Normative reference: this page documents the v0.2 working-draft manifest schema as implemented by the current CLI. The normative, ratified contract for v0.1 is defined in OpenAPM v0.1, Section 4 (Manifest) and published as JSON Schema at manifest-v0.1.schema.json.

Version
0.2 (Working Draft)
Date
2026-05-10
Editors
Daniel Meppiel (Microsoft)
Repository
https://github.com/microsoft/apm
Format
YAML 1.2

This is a Working Draft. It may be updated, replaced, or made obsolete at any time. It is inappropriate to cite this document as other than work in progress.

This specification defines the manifest format (apm.yml) used by the Agent Package Manager (APM). Feedback is welcome via GitHub Issues.


The apm.yml manifest declares the full closure of agent primitive dependencies, MCP servers, scripts, compilation settings, consumer-side policy controls, and (optionally) marketplace authoring metadata for a project. It is the contract between package authors, runtimes, and integrators: any conforming resolver can consume this format to install, compile, run, and pack agentic workflows.


The key words “MUST”, “MUST NOT”, “REQUIRED”, “SHALL”, “SHALL NOT”, “SHOULD”, “SHOULD NOT”, “RECOMMENDED”, “MAY”, and “OPTIONAL” in this document are to be interpreted as described in RFC 2119.

A conforming manifest is a YAML 1.2 document that satisfies all MUST-level requirements in this specification. A conforming resolver is a program that correctly parses conforming manifests and performs dependency resolution as described herein.


A conforming manifest MUST be a YAML mapping at the top level with the following shape:

apm.yml
name: <string> # REQUIRED
version: <string> # REQUIRED
description: <string>
author: <string>
license: <string>
target: <enum | list<enum>>
type: <enum>
scripts: <map<string, string>>
includes: <enum | list<string>>
registries: <map<string, RegistryEntry> & {default?: <string>}>
dependencies:
apm: <list<ApmDependency>>
mcp: <list<McpDependency>>
devDependencies:
apm: <list<ApmDependency>>
mcp: <list<McpDependency>>
compilation: <CompilationConfig>
policy: <PolicyConfig>
marketplace: <MarketplaceConfig> # OPTIONAL; marketplace authoring

Two fields are REQUIRED at parse time: name and version. All other fields are OPTIONAL. Unknown top-level keys MUST be preserved by writers but MAY be ignored by resolvers.

The marketplace: block is the source for apm pack’s marketplace output. Repositories that do not publish a marketplace omit it entirely. See Section 7.

Newly initialised projects (apm init) are scaffolded by the CLI; see apm init for the templates.


Typestring
RequiredMUST be present
DescriptionPackage identifier. Free-form string (no pattern enforced at parse time). Convention: alphanumeric, dots, hyphens, underscores.
Typestring
RequiredMUST be present
Pattern^\d+\.\d+\.\d+ (semver; pre-release/build suffixes allowed)
DescriptionSemantic version. A value that does not match the pattern SHOULD produce a validation warning (non-blocking).
Typestring
RequiredOPTIONAL
DescriptionBrief human-readable description. Inherited by the marketplace: block when not overridden.
Typestring
RequiredOPTIONAL
DescriptionPackage author or organization.
Typestring
RequiredOPTIONAL
DescriptionSPDX license identifier (e.g. MIT, Apache-2.0).
Typestring or list<string>
RequiredOPTIONAL
DefaultAuto-detect from folder presence (see below).
Allowed valuesvscode, agents, copilot, claude, cursor, opencode, codex, gemini, windsurf, all

Controls which output targets are generated during compilation, installation, and packing. Accepts a single string or a YAML list. Unknown values MUST raise a parse error at load time, naming the offending token.

When target: is omitted, a conforming resolver SHOULD auto-detect: vscode if .github/ exists, claude if .claude/ exists, codex if .codex/ exists, windsurf if .windsurf/ exists, all if multiple are present, minimal if none. Auto-detection applies only when target: is unset; once set, the field is authoritative.

# Single target
target: copilot
# Multiple targets (flow-list form)
target: [claude, copilot]
# Multiple targets (block-list form, equivalent)
target:
- claude
- copilot

When a list is specified, only those targets are compiled, installed, and packed; no output is generated for unlisted targets. all cannot be combined with other values.

A plural alias targets: (YAML list only) is also accepted and takes precedence over the legacy CSV form when both are declared. Prefer targets: in new manifests; target: remains supported for backward compatibility.

ValueEffect
vscodeEmits AGENTS.md at the project root (and per-directory files in distributed mode).
agentsAlias for vscode.
copilotAlias for vscode.
claudeEmits CLAUDE.md at the project root.
cursorEmits to .cursor/rules/, .cursor/agents/, .cursor/skills/.
opencodeEmits to .opencode/agents/, .opencode/commands/, .opencode/skills/.
codexEmits AGENTS.md and deploys skills to .agents/skills/, agents to .codex/agents/.
geminiEmits GEMINI.md and deploys to .gemini/commands/, .gemini/skills/, .gemini/settings.json.
windsurfEmits AGENTS.md and deploys to .windsurf/rules/, .windsurf/skills/, .windsurf/workflows/, .windsurf/hooks.json.
allAll targets. Cannot be combined with other values in a list.
minimalAGENTS.md only at project root. Auto-detected only: this value MUST NOT be set explicitly in manifests; it is an internal fallback when no target folder is detected.
Typeenum<string>
RequiredOPTIONAL
DefaultNone (behaviour driven by package content; synthesized plugin manifests use hybrid).
Allowed valuesinstructions, skill, hybrid, prompts

Declares how the package’s content is processed during install and compile. Today behaviour is driven by package content (presence of SKILL.md, component directories, etc.); this field is reserved for future explicit overrides.

ValueBehaviour
instructionsCompiled into AGENTS.md only. No skill directory created.
skillInstalled as a native skill only. No AGENTS.md output.
hybridBoth AGENTS.md compilation and skill installation.
promptsCommands/prompts only. No instructions or skills.
Typemap<string, string>
RequiredOPTIONAL
KeyScript name (free-form string).
ValueShell command string.

Named commands executed via apm run <name>. The script body MUST support --param key=value substitution ({key} placeholders in the command string are replaced before execution).

The script name start is the default invoked by a bare apm run (no name given) and SHOULD be present in publishable packages so consumers have a one-command entry point.

scripts:
start: "copilot -p 'README.prompt.md'"
review: "copilot -p 'code-review.prompt.md'"
impl: "copilot -p 'implement-feature.prompt.md'"
Typestring (literal auto) or list<string>
RequiredOPTIONAL
DefaultUndeclared (legacy implicit auto-publish; flagged by apm audit).
Allowed valuesauto or a list of paths relative to the project root.

Declares which local .apm/ content the project consents to publish when packing or deploying. Three forms are supported:

  1. Undeclared (field omitted). Legacy behaviour: all local .apm/ content is published as if auto were set. apm audit emits an includes-consent advisory whenever local content is deployed under this form.
  2. includes: auto. Explicit consent to publish all local .apm/ content via the file scanner. No path enumeration required. Default for newly initialised projects.
  3. includes: [<path>, ...]. Explicit allow-list of paths the project consents to publish. Strongest governance form; changes are reviewable in PR diffs.
# Form 1: undeclared (legacy; audit advisory)
# includes: <omitted>
# Form 2: explicit auto-publish (default for new projects)
includes: auto
# Form 3: explicit path list (strongest governance)
# includes:
# - .apm/instructions/
# - .apm/skills/my-skill/

includes: is allow-list only. There is no exclude: form. To keep maintainer-only primitives out of shipped artifacts, author them OUTSIDE .apm/ and reference them via a local-path devDependency. See Dev-only Primitives.

When policy.manifest.require_explicit_includes is true (see Policy reference), only form 3 passes; auto and undeclared are rejected at install/audit time by the explicit-includes check (not at YAML parse time).

Typemap<string, string>
RequiredOPTIONAL
DescriptionConsumer-side controls for org policy discovery and verification. All sub-keys are optional; defaults preserve current fail-open install behaviour.
policy:
fetch_failure_default: warn # warn | block (default warn)
hash: "sha256:<hex>" # optional consumer-side pin on the org policy bytes
hash_algorithm: sha256 # sha256 (default) | sha384 | sha512
Sub-keyTypeDefaultAllowed valuesSemantic
fetch_failure_defaultstringwarnwarn, blockPosture when no enforceable policy is available (fetch failures or no-policy outcomes). warn keeps installs unblocked; block opts into fail-closed semantics for both apm install and apm audit --ci.
hashstringunset<algo>:<hex-digest>Pin on the raw bytes of the fetched leaf org policy. Verified before YAML parsing; mismatch is always fail-closed regardless of fetch_failure_default.
hash_algorithmstringsha256sha256, sha384, sha512Digest algorithm for policy.hash. Inferred from the <algo>: prefix when present. MD5 and SHA-1 are rejected at parse time.

Full semantics (network failure matrix, hash pin verification, policy precedence) live in the Policy reference.

Typemap<string, RegistryEntry> with optional default: <string> key
RequiredOPTIONAL
DescriptionDeclares REST-based APM registries for the project. Strictly additive — absent or empty block leaves Git resolution unchanged unless a default registry is configured in ~/.apm/config.json. URLs from all layers are merged at install time; see the Registries guide.
registries:
jf-skills:
url: https://artifactory.example.com/artifactory/api/skills/jf-skills-local
default: jf-skills # OPTIONAL — name of one of the configured entries
Sub-keyTypeRequiredConstraintSemantic
<name>RegistryEntryat least one when block is non-emptyName uses lowercase letters, digits, -, .Registered registry.
<name>.urlstringREQUIRED per entryMUST start with https:// or http://; no trailing slash requiredBase URL the client appends /v1/... paths to.
defaultstringOPTIONALMUST name one of the configured entriesWhen set, plain string-shorthand APM deps and object-form deps without an explicit registry: key route through this registry. Project value wins over registry.<name>.default in ~/.apm/config.json.

Unknown keys under a registry entry MUST be rejected at parse time (typo guard).

Effective default registry: project registries.default if present; otherwise the registry marked "default": true in ~/.apm/config.json (via apm config set registry.<name>.default true). Only one default is active at a time.

For full client semantics — auth, lockfile fields, and routing rules — see the Registries guide. For the wire contract servers implement, see the Registry HTTP API.


Typeobject
RequiredOPTIONAL
Known keysapm, mcp

Contains two OPTIONAL lists: apm for agent primitive packages and mcp for MCP servers. Each list entry is either a string shorthand or a typed object. Additional keys MAY be present for future dependency types; conforming resolvers MUST ignore unknown keys for resolution but MUST preserve them when reading and rewriting manifests.


4.1. dependencies.apmlist<ApmDependency>

Section titled “4.1. dependencies.apm — list<ApmDependency>”

Each element MUST be one of two forms: string or object.

Grammar (ABNF-style):

dependency = url_form / shorthand_form / local_path_form
url_form = ("https://" / "http://" / "ssh://git@" / "git@") clone-url
shorthand_form = [host "/"] owner "/" repo ["/" virtual_path] ["#" ref]
local_path_form = ("./" / "../" / "/" / "~/" / ".\\" / "..\\" / "~\\") path

When a default registry is configured — via registries.default in apm.yml or registry.<name>.default true in ~/.apm/config.json — plain shorthand_form entries with a #<selector> route through that registry instead of Git.

clone-url MAY include a :port segment on https://, http://, and ssh://git@ forms (e.g. ssh://git@host:7999/owner/repo.git). The SCP shorthand git@host:path cannot carry a port — : is the path separator in that form. When a port is present, APM preserves it across all clone attempts: the SSH attempt uses ssh://host:PORT/... and the HTTPS fallback uses https://host:PORT/... (same port on both protocols).

SegmentRequiredPatternDescription
hostOPTIONALFQDN (e.g. gitlab.com)Git host. Defaults to github.com.
portOPTIONAL1-65535Non-default port on ssh://, https://, http:// clone URLs. Not expressible in SCP shorthand.
owner/repoREQUIRED2+ path segments of [a-zA-Z0-9._~-]+ on non-Azure-DevOps hosts; [a-zA-Z0-9._\- ]+ (allows spaces, not tilde) on Azure DevOpsRepository path. GitHub uses exactly 2 segments. Non-GitHub hosts MAY use nested groups (e.g. gitlab.com/group/sub/repo). Tilde supports Bitbucket Data Center personal-repo segments (/scm/~user/repo.git) and Sourcehut ~user paths.
virtual_pathOPTIONALPath segments after repoSubdirectory or file within the repo. See Section 4.1.3.
refOPTIONALBranch, tag, or commit SHAGit reference. Commit SHAs matched by ^[a-f0-9]{7,40}$. Semver tags matched by ^v?\d+\.\d+\.\d+.

Examples:

dependencies:
apm:
# GitHub shorthand (default host); each line shows a syntax variant
- microsoft/apm-sample-package # latest (lockfile pins commit SHA)
- microsoft/apm-sample-package#v1.0.0 # pinned to tag (immutable)
- microsoft/apm-sample-package#main # branch ref (may change over time)
# Non-GitHub hosts (FQDN preserved)
- gitlab.com/acme/coding-standards
- bitbucket.org/team/repo#main
# Full URLs
- https://github.com/microsoft/apm-sample-package.git
- http://github.com/microsoft/apm-sample-package.git
- git@github.com:microsoft/apm-sample-package.git
- ssh://git@github.com/microsoft/apm-sample-package.git
# Custom ports (e.g. Bitbucket Datacenter, self-hosted GitLab)
- ssh://git@bitbucket.example.com:7999/project/repo.git
- https://git.internal:8443/team/repo.git
# Virtual packages
- ComposioHQ/awesome-claude-skills/brand-guidelines # subdirectory
- contoso/prompts/review.prompt.md # single file
# Azure DevOps
- dev.azure.com/org/project/_git/repo
# Local path (development only)
- ./packages/my-shared-skills # relative to project root
- ../sibling-repo/my-package # parent directory

REQUIRED when the shorthand is ambiguous (e.g. direct nested-group repos with virtual paths). NOT required for nested-group deps that route through a registry proxy (explicit host/artifactory/<key>/... FQDN, or bare shorthand under PROXY_REGISTRY_URL + PROXY_REGISTRY_ONLY=1): the install-time boundary probe HEAD-walks candidate splits against the proxy and locks in the first one whose archive responds. See Registry proxy guide.

FieldTypeRequiredPattern / ConstraintDescription
gitstringREQUIRED (remote)HTTPS URL, SSH URL, or FQDN shorthandClone URL of the repository. Required for remote dependencies.
pathstringOPTIONAL / REQUIRED (local)Relative path within the repo, or local filesystem pathWhen git is present: subdirectory or file (virtual package). When git is absent: local filesystem path (must start with ./, ../, /, or ~/).
refstringOPTIONALBranch, tag, or commit SHAGit reference to checkout.
aliasstringOPTIONAL^[a-zA-Z0-9._-]+$Local alias.

Remote dependency (git URL plus sub-path):

- git: https://gitlab.com/acme/repo.git
path: instructions/security
ref: v2.0
alias: acme-sec

ref: accepts either a literal git ref (main, v2.0, a 40-char commit SHA) or a semver range (^1.2.0, ~1.4, >=2.0 <3, 1.5.x). When ref: is a semver range, APM resolves it against the remote’s tags at install time, matching against v{version} and {name}--v{version} patterns (with {version} as a bare-tag fallback) and selecting the highest tag that satisfies the range.

The lockfile records the original constraint, the resolved_tag, the resolved version, the resolved_commit, and a resolved_at timestamp so subsequent installs replay the same tag deterministically — only apm install --update or a manifest change re-resolves. When no remote tag satisfies the range, APM surfaces NoMatchingTagError with the inspected patterns.

Local path dependency (development only):

- path: ./packages/my-shared-skills

Monorepo sibling reference (git: parent):

# In agents/pkg-a/apm.yml inside org/monorepo
- git: parent
path: skills/shared

The literal sentinel git: parent is valid only inside a transitively resolved package whose clone coordinates are known to the resolver. APM expands parent to the consumer’s host, repo_url, and resolved ref, with virtual_path set from path. The lockfile records the expanded coordinates: parent MUST NOT appear as durable identity (repo_url / source). path is REQUIRED for git: parent and is normalised to a single relative path; absolute paths and .. traversal are refused. ref and alias overrides are accepted; when ref is omitted the parent’s resolved ref is inherited.

Registry dependency (whole package or virtual sub-path):

# Whole package via the default registry
- id: acme/toolkit
version: ^2.0.0
# Whole package routed to a named registry
- registry: jf-skills # OPTIONAL — defaults to the effective default registry
id: acme/toolkit # REQUIRED — owner/repo identity at the registry
version: ^2.0.0 # REQUIRED — opaque version selector (semver when supported)
# Virtual package (sub-path inside a published package)
- registry: jf-skills
id: acme/prompt-pack
path: prompts/review.prompt.md # OPTIONAL — omit to install the whole package
version: 1.4.0
alias: review # OPTIONAL

id: (or registry:) and git: are mutually exclusive on the same entry. version: MUST be a non-empty string — opaque selectors such as stable, main, or commit pins are valid; semver ranges (^1.2.3) are interpreted as ranges when the registry publishes semver-tagged versions. When registry: is omitted, a default registry MUST be configured — in project apm.yml or via registry.<name>.default true in ~/.apm/config.json; APM hard-fails otherwise.

A dependency MAY target a subdirectory or a file within a repository rather than the whole repo. Conforming resolvers MUST classify virtual packages using the following rules, evaluated in order:

KindDetection ruleExample
Filevirtual_path ends in .prompt.md, .instructions.md, .agent.md, or .chatmode.mdowner/repo/prompts/review.prompt.md
Subdirectoryvirtual_path does not match any file extension aboveowner/repo/skills/security

Classification is by extension only, never by path segment. A path like owner/repo/collections/security (no extension) is a Subdirectory: the on-disk shape (APM package with apm.yml, skill bundle, or plugin) is resolved at fetch time by probing for apm.yml first.

Removed (#1094): the legacy .collection.yml / .collection.yaml virtual-package form is no longer supported. Convert any such reference to an apm.yml with a dependencies: section, then reference the resulting subdirectory as a regular subdirectory virtual package.

Conforming writers MUST normalise entries to canonical form on write. github.com is the default host and MUST be stripped; all other hosts MUST be preserved as FQDN.

InputCanonical form
https://github.com/microsoft/apm-sample-package.gitmicrosoft/apm-sample-package
git@github.com:microsoft/apm-sample-package.gitmicrosoft/apm-sample-package
gitlab.com/acme/repogitlab.com/acme/repo

4.2. dependencies.mcplist<McpDependency>

Section titled “4.2. dependencies.mcp — list<McpDependency>”

Each element MUST be one of two forms: string or object.

A plain registry reference: io.github.github/github-mcp-server.

FieldTypeRequiredConstraintDescription
namestringREQUIREDNon-emptyServer identifier (registry name or custom name).
transportenum<string>Conditionalstdio, sse, http, streamable-httpTransport protocol. REQUIRED when registry: false. Values are MCP transport names, not URL schemes; remote variants connect over HTTPS.
envmap<string, string>OPTIONALEnvironment variable overrides. Values may contain ${VAR}, ${env:VAR}, or ${input:<id>} references; see Section 4.2.4.
argsdict or listOPTIONALDict for overlay variable overrides (registry); list for positional args (self-defined).
versionstringOPTIONALPin to a specific server version.
registrybool or stringOPTIONALDefault: true (public registry)false = self-defined (private) server. String = custom registry URL.
packageenum<string>OPTIONALnpm, pypi, ociPackage manager type hint.
headersmap<string, string>OPTIONALCustom HTTP headers for remote endpoints. Same variable syntax as env.
toolslist<string>OPTIONALDefault: ["*"]Restrict which tools are exposed.
urlstringConditionalEndpoint URL. REQUIRED when registry: false and transport is http, sse, or streamable-http.
commandstringConditionalSingle binary path; no embedded whitespace unless args is also presentBinary path. REQUIRED when registry: false and transport is stdio.

4.2.3. Validation Rules for Self-Defined Servers

Section titled “4.2.3. Validation Rules for Self-Defined Servers”

When registry is false, the following constraints apply:

  1. transport MUST be present.
  2. If transport is stdio, command MUST be present.
  3. If transport is http, sse, or streamable-http, url MUST be present.
  4. If transport is stdio, command MUST be a single binary path with no embedded whitespace. APM does not split command on whitespace; use args for additional arguments. A path that legitimately contains spaces (e.g. /opt/My App/server) is allowed when args is also provided (including an explicit empty list args: []), signaling the author has taken responsibility for the shape.
dependencies:
mcp:
# Registry reference (string)
- io.github.github/github-mcp-server
# Registry with overlays (object)
- name: io.github.github/github-mcp-server
tools: ["repos", "issues"]
env:
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
# Self-defined server (object, registry: false)
- name: my-private-server
registry: false
transport: stdio
command: ./bin/my-server
args: ["--port", "3000"]
env:
API_KEY: ${{ secrets.KEY }}

4.2.4. Variable References in headers and env

Section titled “4.2.4. Variable References in headers and env”

Values in headers and env may contain three placeholder syntaxes. APM resolves them per-target so secrets stay out of generated config files where possible.

SyntaxSourceVS CodeCopilot CLI / Codex / Gemini / Cursor
${VAR}host environmentTranslated to ${env:VAR} (resolved at server-start by VS Code)Resolved at install time from env (or interactive prompt)
${env:VAR}host environmentNative; passed through verbatimResolved at install time from env (or interactive prompt)
${input:<id>}user promptNative; VS Code prompts at runtimeNot supported; use ${VAR} or ${env:VAR} instead
<VAR> (legacy)host environmentNot recognizedResolved at install time (kept for back-compat)
  • VS Code has native ${env:VAR} and ${input:VAR} interpolation, so APM emits placeholders rather than baking secrets into mcp.json. Bare ${VAR} is normalized to ${env:VAR} for you.
  • Copilot CLI, Codex, Gemini, and Cursor have no runtime interpolation, so APM resolves ${VAR}, ${env:VAR}, and the legacy <VAR> at install time using os.environ (or an interactive prompt when missing). Resolved values are not re-scanned, so a value containing literal ${...} text is preserved.
  • Recommended: Use ${VAR} or ${env:VAR} in all new manifests — they work on every target that supports remote MCP servers. <VAR> is legacy; in VS Code it would silently render as literal text in the generated config.
  • Registry-backed servers — APM auto-generates input prompts from registry metadata for ${input:...}.
  • Self-defined servers — APM detects ${input:...} patterns in apm.yml and generates matching input definitions automatically.

GitHub Actions templates (${{ ... }}) are intentionally left untouched.

dependencies:
mcp:
- name: my-server
registry: false
transport: http
url: https://my-server.example.com/mcp/
headers:
Authorization: "Bearer ${MY_SECRET_TOKEN}" # bare env-var
X-Tenant: "${env:TENANT_ID}" # env-prefixed
X-Project: "${input:my-server-project}" # VS Code input prompt

Typeobject
RequiredOPTIONAL
Known keysapm, mcp

Development-only dependencies installed locally but excluded from plugin bundles produced by apm pack (plugin format is the default). Uses the same structure as dependencies.

devDependencies:
apm:
- owner/test-helpers
- owner/lint-rules#v2.0.0

Created automatically by apm plugin init. Use apm install --dev to add packages:

Terminal window
apm install --dev owner/test-helpers

Plain apm install (no flag) deploys both dependencies and devDependencies. There is no --omit=dev flag today; the dev/prod separation kicks in at apm pack (plugin format, the default). The local-content scanner that builds plugin bundles operates on .apm/ only and does not consult the devDep marker. To keep maintainer-only primitives out of shipped artifacts, author them outside .apm/ and reference them via a local-path devDependency. See Dev-only Primitives.

Local-path devDependency example:

devDependencies:
apm:
- path: ./dev/skills/release-checklist

The compilation key is OPTIONAL. It controls apm compile behaviour. All fields have sensible defaults; omitting the entire section is valid.

FieldTypeDefaultConstraintDescription
targetenum<string>allSame values as Section 3.6Output target. Defaults to all when set explicitly in compilation config.
strategyenum<string>distributeddistributed, single-filedistributed generates per-directory target files (e.g. AGENTS.md, CLAUDE.md). single-file generates one monolithic file at output.
single_fileboolfalseLegacy alias. When true, overrides strategy to single-file.
outputstringAGENTS.mdFile pathCustom output path for the compiled file.
chatmodestringunsetChatmode filter for compilation.
resolve_linksbooltrueResolve relative Markdown links in primitives.
source_attributionbooltrueInclude source-file origin comments in compiled output.
excludelist<string> or string[]Glob patternsDirectories to skip during compilation (e.g. apm_modules/**).
placementobjectunsetPlacement tuning. See Section 6.1.
FieldTypeDefaultDescription
min_instructions_per_fileint1Minimum instruction count to warrant a separate AGENTS.md file.
compilation:
target: all
strategy: distributed
source_attribution: true
exclude:
- "apm_modules/**"
- "tmp/**"
placement:
min_instructions_per_file: 1

The OPTIONAL marketplace: block declares the metadata apm pack needs to emit a Claude-Code-compatible plugin marketplace (marketplace.json). It is read by apm marketplace subcommands and ignored by everything else. Repositories that do not publish a marketplace omit it entirely.

The block was previously a standalone marketplace.yml file (still loadable for back-compat); the in-apm.yml form is canonical and is what apm marketplace init scaffolds.

Three keys are inherited from the top-level manifest unless explicitly overridden inside marketplace::

KeyInherited from
nametop-level name
descriptiontop-level description
versiontop-level version

Overrides exist for the rare case where the published marketplace identity differs from the package identity. Inherited values are omitted from the generated marketplace.json per the Claude-Code convention.

FieldTypeRequiredDefaultDescription
namestringOPTIONAL (override)inheritedOverride of top-level name.
descriptionstringOPTIONAL (override)inheritedOverride of top-level description.
versionstringOPTIONAL (override)inheritedOverride of top-level version. Validated as semver.
ownerOwnerREQUIREDMarketplace publisher identity. See Section 7.3.
outputstringOPTIONAL.claude-plugin/marketplace.jsonOutput path for the generated marketplace JSON.
metadataobjectOPTIONAL{}Free-form metadata forwarded verbatim to marketplace.json (e.g. homepage, support).
buildBuildOPTIONALtagPattern: "v{version}"Build configuration for resolving package refs. See Section 7.4.
packageslist<Package>OPTIONAL[]Packages exposed in the marketplace. See Section 7.5.

Unknown keys inside marketplace: are rejected at parse time.

FieldTypeRequiredDescription
namestringREQUIREDOwner display name (org or person).
emailstringOPTIONALContact email.
urlstringOPTIONALOwner homepage.
marketplace:
owner:
name: contoso
url: https://github.com/contoso
email: maintainers@contoso.example
FieldTypeDefaultDescription
tagPatternstringv{version}Pattern used to construct git tags for packages. MUST contain at least one of {version} or {name}. Per-package overrides live on packages[].tag_pattern.

Each entry MUST be a mapping. Unknown keys are rejected.

FieldTypeRequiredDescription
namestringREQUIREDPackage identifier as it appears in the marketplace.
sourcestringREQUIREDOne of: <owner>/<repo> (remote on the default host), <host.tld>/<owner>/<repo> (remote on a non-default host such as GitHub Enterprise or self-hosted GitLab — shorthand), https://<host.tld>/<owner>/<repo>[.git] (same, full URL form — a trailing .git is stripped), or ./<path> (local). Must match the source pattern; path traversal (..) is refused, and URL forms with userinfo (user@host), ports, query strings, or non-https schemes are rejected.
subdirstringOPTIONALSubdirectory inside the source repo. Path-traversal-validated. Ignored for local sources.
versionstringConditionalSemver range (e.g. ^1.0.0, ~2.1.0, >=3.0). Stored as a string; resolution happens at pack time. REQUIRED for remote packages unless ref is given.
refstringConditionalExplicit git ref (SHA, tag, or branch). Overrides version range when both are present. REQUIRED for remote packages unless version is given.
tag_patternstringOPTIONALPer-package override of build.tagPattern. Same placeholder rule.
include_prereleaseboolfalseWhether semver pre-release tags are eligible for resolution.
descriptionstringOPTIONALPass-through to marketplace.json.
homepagestringOPTIONALPass-through to marketplace.json.
tagslist<string>OPTIONALPass-through to marketplace.json. Limited to 50 tags, 100 chars each.
keywordslist<string>OPTIONALAlias merged into tags (deduplicated).
authorstring or objectOPTIONALEither a non-empty string (treated as name) or an object with name (REQUIRED), email, url.
licensestringOPTIONALPass-through (SPDX identifier).
repositorystringOPTIONALPass-through.

Remote packages MUST declare at least one of version or ref. Local packages (sources beginning with ./) skip git resolution and have no version requirement.

The first three source forms target a remote git host; the second and third name a non-default host (e.g. GitHub Enterprise, self-hosted GitLab) as either a shorthand or a full HTTPS URL with an optional .git suffix that is normalized away. Path traversal (..) in local paths, userinfo (user@host), ports, query strings, and non-https URL schemes are rejected at parse time.

Non-default hosts authenticate via the standard APM token chain — see the authentication guide for the per-host-class lookup order. A token resolved for the default host is never forwarded to a non-default host.

marketplace:
# name, description, version inherit from top-level apm.yml
owner:
name: contoso
url: https://github.com/contoso
output: .claude-plugin/marketplace.json
metadata:
homepage: https://contoso.example/marketplace
build:
tagPattern: "v{version}"
packages:
- name: code-review
source: contoso/code-review
version: "^1.0.0"
description: AI code-review skills
tags: [review, quality]
- name: pinned-helper
source: contoso/pinned-helper
ref: main # explicit ref overrides version
tag_pattern: "pinned-helper-v{version}"
- name: local-tool # local-path package
source: ./packages/local-tool
description: Vendored tool
- name: enterprise-agents # GHE shorthand
source: ghe.corp.example.com/platform/agents
version: "^0.3.0"
- name: gitlab-helper # full URL form
source: https://gitlab.corp.example.com/team/helper.git
ref: v1.2.0

The legacy standalone marketplace.yml (top-level keys, no marketplace: wrapper) is still loadable but deprecated; new repositories SHOULD use the in-apm.yml form scaffolded by apm marketplace init.


After successful dependency resolution, a conforming resolver MUST write a lockfile capturing the exact resolved state. The lockfile MUST be a YAML file named apm.lock.yaml at the project root and SHOULD be committed to version control.

The full lockfile schema is specified in the Lockfile specification. At a minimum, every resolver MUST record lockfile_version, dependencies[].repo_url, dependencies[].resolved_commit, and dependencies[].deployed_files so subsequent installs are reproducible and apm uninstall can remove every placed file.

Resolver behaviour:

  1. First install — Resolve all dependencies, write apm.lock.yaml.
  2. Subsequent installs — Read apm.lock.yaml, use locked commit SHAs. A resolver SHOULD skip download when the local checkout already matches.
  3. --update flag — Re-resolve from apm.yml, overwrite the lockfile.

Any runtime adopting this format (e.g. GitHub Agentic Workflows, CI systems, IDEs) MUST implement these steps:

  1. Parse — Read apm.yml as YAML. Validate the two REQUIRED fields (name, version) and the dependencies object shape.
  2. Resolve dependencies.apm — For each entry, clone or fetch the git repo (respecting ref), locate the .apm/ directory (or virtual path), and extract primitives.
  3. Resolve dependencies.mcp — For each entry, resolve from the MCP registry or validate self-defined transport config per Section 4.2.3.
  4. Transitive resolution — Resolved packages MAY contain their own apm.yml with further dependencies, forming a dependency tree. Resolvers MUST resolve transitively. Conflicts are merged at instruction level (by applyTo pattern), not file level.
  5. Write lockfile — Record exact commit SHAs and deployed file paths in apm.lock.yaml per Section 8 and the Lockfile specification.

name: my-project
version: 1.0.0
description: AI-native web application
author: Contoso
license: MIT
target: [claude, copilot]
type: hybrid
includes: auto
scripts:
start: "copilot -p 'README.prompt.md'"
review: "copilot -p 'code-review.prompt.md'"
impl: "copilot -p 'implement-feature.prompt.md'"
dependencies:
apm:
- microsoft/apm-sample-package#v1.0.0
- gitlab.com/acme/coding-standards#main
- git: https://gitlab.com/acme/repo.git
path: instructions/security
ref: v2.0
mcp:
- io.github.github/github-mcp-server
- name: my-private-server
registry: false
transport: stdio
command: ./bin/my-server
env:
API_KEY: ${{ secrets.KEY }}
devDependencies:
apm:
- owner/test-helpers
compilation:
target: all
strategy: distributed
exclude:
- "apm_modules/**"
placement:
min_instructions_per_file: 1
policy:
fetch_failure_default: warn
marketplace:
owner:
name: contoso
url: https://github.com/contoso
packages:
- name: code-review
source: contoso/code-review
version: "^1.0.0"
tags: [review, quality]

VersionDateChanges
0.12026-03-06Initial Working Draft.
0.22026-05-10Added Section 7 (Marketplace authoring block). Documented scripts.start as the default apm run entry point. Cross-links updated to reference CLI paths. ASCII-only enforcement.