Architecture
This page is for contributors changing pm-cli internals. Users should start with Quickstart. Agents should start with Agent Guide.
Agent Quick Context
- CLI wiring lives in
src/cli/. - Domain behavior lives in
src/core/. - Public SDK exports live in
src/sdk/. - Items are stored as TOON by default; history is append-only JSONL.
pm contractsis the machine-readable runtime contract source.
Tracked documentation work: pm-u9d0.
System Overview
pm-cli is a TypeScript ESM CLI for Node.js 20+. It is file-backed, git-native, deterministic, and designed for concurrent human plus agent workflows.
High-level flow:
- Commander parses CLI input in
src/cli/main.tswith commands registered via per-family modules (register-setup.ts,register-list-query.ts,register-mutation.ts,register-operations.ts). - Command modules normalize options and call domain services.
- Domain services load settings, acquire locks when needed, mutate canonical item documents, and append history.
- Renderers emit TOON by default, JSON when requested, and markdown for calendar views.
- Extensions can add commands, schema, renderers, import/export handlers, search providers, lifecycle hooks, and selected service overrides.
Source Tree
src/
cli.ts
cli/
main.ts
register-setup.ts
register-list-query.ts
register-mutation.ts
register-operations.ts
registration-helpers.ts
commands/
help-content.ts
error-guidance.ts
extension-command-options.ts
core/
extensions/
fs/
history/
item/
lock/
output/
search/
store/
front-matter-cache.ts
test/
shared/
sdk/
cli-contracts.ts
index.ts
types/
tests/
unit/
integration/
.agents/
pm/
extensions/
docs/
scripts/
Important public docs:
Storage Layout
Project tracker root defaults to .agents/pm/.
.agents/pm/
settings.json
epics/
features/
tasks/
chores/
issues/
decisions/
events/
reminders/
milestones/
meetings/
history/
locks/
index/
search/
extensions/
Required data:
- item documents under type folders
history/<id>.jsonlsettings.json
Optional rebuildable data:
- keyword and vector search cache files
- generated index metadata
Item Documents
Default format is TOON:
id: pm-a1b2
title: Implement restore replay
description: Restore should rebuild target item state from history.
type: Task
status: in_progress
priority: 1
tags[2]: history,restore
body: |
Implementation notes.
Legacy JSON-front-matter markdown files are read only for one-way migration into TOON. Runtime internals use metadata as the item metadata model key.
Built-in item types:
EpicFeatureTaskChoreIssueDecisionEventReminderMilestoneMeeting
Runtime type resolution merges built-ins, settings.item_types.definitions, and extension registerItemTypes(...) registrations.
Mutation Contract
Every item mutation follows the same safety path:
- Resolve project root and settings.
- Acquire item lock when mutating existing item state.
- Read and parse the current canonical item document.
- Enforce ownership and policy gates.
- Compute
before_hash. - Apply mutation in memory.
- Set
updated_at. - Compute RFC6902 patch and
after_hash. - Write item atomically through temp-file plus rename.
- Append one history JSONL line.
- Release lock.
If a write fails after state changes begin, mutation code attempts rollback before returning the error.
History and Restore
History entries are append-only JSONL records:
{
"ts": "2026-05-01T12:00:00.000Z",
"author": "codex-agent",
"op": "update",
"patch": [],
"before_hash": "sha256...",
"after_hash": "sha256...",
"message": "Start implementation"
}
pm restore <id> <timestamp-or-version> replays history from create through the target record and appends a restore event. Restore does not rewrite prior history.
Useful diagnostics:
pm history <id> --full --diff --verify
pm activity --id <id> --limit 50
pm validate --check-history-drift
Command Contracts
Command/action metadata is centralized in src/sdk/cli-contracts.ts and used by:
- CLI option normalization
- help output
- completion generation
- provider-safe tool schemas
pm contracts- extension command/action contract exposure
Use runtime contracts instead of duplicating flag lists:
pm contracts --json
pm contracts --command create --flags-only --json
pm help create --json
Telemetry Schema Negotiation
Telemetry preserves wire compatibility through an explicit client/server negotiation split:
- Event payloads keep
event.schema_versionas the event-document schema (currently v1). - Queue envelopes include
client_schema_versionso client/runtime evolution can be tracked independently from event payload versioning. pm health --check-telemetryprobes/healthzand records any advertised max-version header for observability/debugging.
This keeps v1 behavior stable while providing a forward path for future telemetry schema upgrades.
Output Pipeline
Core output formats:
- TOON for sparse, token-efficient default command output
- JSON for strict machine parsing
- markdown for calendar-oriented views
The renderer omits null, undefined, empty arrays, and empty objects from sparse TOON fallback output. JSON preserves the machine payload.
Search Architecture
Search supports:
- keyword mode, always available
- semantic mode, when an embedding provider and vector store are available
- hybrid mode, combining keyword and semantic results
Keyword scoring uses weighted fields such as title, description, tags, status, body, comments, notes, learnings, reminders, events, and dependencies. Semantic indexing uses the same core corpus so calendar-heavy work remains discoverable through normal search and reindex flows.
Runtime semantic components can come from built-ins or extensions:
- provider selection:
settings.search.provider - vector adapter selection:
settings.vector_store.adapter - extension registration:
registerSearchProvider(...)andregisterVectorStoreAdapter(...)
Useful commands:
pm search "restore history" --mode keyword --limit 10
pm reindex --mode hybrid --progress
pm health --check-only
Extension Host
Load order:
- core commands
- global extensions
- project extensions
Project extensions take precedence over global extensions for matching command or renderer keys. Extension dispatch is extension-first when a registered handler matches a core command path.
Extension override planes:
- commands
- parser overrides
- preflight overrides
- service overrides
- renderers
- import/export handlers
- item fields and item types
- migrations
- search providers and vector adapters
- lifecycle hooks
See Extensions and SDK.
Testing Architecture
Tests live under:
tests/unit/
tests/integration/
All tests must run with sandboxed PM_PATH and PM_GLOBAL_PATH. Use:
node scripts/run-tests.mjs test
node scripts/run-tests.mjs coverage
Linked-test execution also creates sandbox roots and can seed settings/extensions for schema parity. See Testing.
Coverage governance is literal all-source under src/:
vitest.config.tscoverage include patterns aresrc/*.tsandsrc/**/*.ts.- Global thresholds are explicit ratchet baselines for the measured all-source corpus and should only move upward until they reach 100%.
- Do not maintain a curated include/exclude allowlist for production
srcmodules. - When a module is hard to test end-to-end (for example CLI orchestration), extract pure logic helpers into small modules and cover those directly instead of weakening thresholds.
Terminal Compatibility
Runtime behavior should remain terminal-neutral:
- no required ANSI or custom terminal protocol
- deterministic TOON/JSON/markdown output
- graceful
process.exitCodehandling - broken-pipe-safe output writes
- explicit TTY rejection for stdin token paths that require piped input
- non-interactive linked-test subprocess handling
Public Documentation Boundary
Architecture docs should describe source structure and public runtime behavior only. Ignored local operations material and host-specific runbooks must stay out of tracked docs.