CLI Reference¶
The pulp CLI wraps common build, test, validation, and shipping operations.
Source: tools/cli/pulp_cli.cpp
Commands¶
create¶
Status: usable
Create a new plugin project from templates. Checks environment, scaffolds source files, configures the project, builds the generated test plus the default platform outputs, and runs tests.
Default behavior is product-first:
- SDK mode (default for external projects): uses find_package(Pulp), generates pulp.toml, and either reuses a checkout-built local SDK or downloads a cached SDK release when no local checkout hint is available
- Source-tree mode (--in-tree / --example): uses local source and adds the project to examples/
pulp create "My Gain" # effect plugin (default)
pulp create "My Synth" --type instrument # instrument plugin
pulp create "My App" --type app # standalone audio application
pulp create "My Project" --type bare # minimal skeleton
pulp create "My FX" --manufacturer "Acme Audio" # custom manufacturer
pulp create "My FX" --output ~/projects/my-fx # custom output directory
pulp create "Debug Knob" --in-tree # add an example under examples/
pulp create "Kit Gain" --template ./my-template-kit # scaffold from a local template kit
pulp create "My FX" --no-build # scaffold only, skip build
pulp create "My FX" --no-interactive # CI/scripting mode (no prompts)
Available types: effect (default), instrument, app, bare. Built-in templates are in tools/templates/<type>/.
--template <name-or-kit-dir> accepts either a built-in template name or an explicit local template kit directory containing pulp.package.json. Local template kits are validated before scaffolding, must declare kind template, and must export exactly one safe template directory. They do not execute package CMake, JavaScript, scripts, or dynamic libraries. If a template kit declares dependency packages, install those curated dependencies explicitly with pulp add <id> first; pulp create will not widen the trust boundary by adding them implicitly. A template kit only builds format targets for entry templates it exports, so a small CLAP/Standalone starter does not pretend to support VST3/AU.
Default output location resolution:
1. --output <dir>
2. PULP_PROJECTS_DIR
3. ~/.pulp/config.toml with:
- If run inside the Pulp repo: create the project next to the repo root
- Otherwise: create the project in
./<name>
Set PULP_HOME to move the default ~/.pulp/ home used for SDK/cache/config storage.
This flow is meant to behave the same from a normal terminal, CI, or agent-driven workflows. The CLI prints which mode it selected and why, while --no-interactive, PULP_PROJECTS_DIR, and PULP_HOME make project creation predictable for automation.
Mode truth:
- SDK mode means you are in an external project building against a pinned installed Pulp SDK artifact.
- Source-tree mode means you are inside the Pulp checkout building the repo or its examples against live source.
- pulp create defaults to SDK mode unless you explicitly ask for an in-tree example with --in-tree.
What it does:
1. Runs pulp doctor checks (fails fast if environment is broken)
2. In standalone product mode: if run from inside a Pulp checkout, prepares pinned dependencies from that checkout and caches a local SDK install; otherwise downloads and caches the SDK release
3. Scaffolds source files from built-in templates or an explicitly named local template kit (processor, format entries, test, CMakeLists.txt)
4. In in-tree mode: adds the project to examples/CMakeLists.txt
5. In standalone product mode: generates pulp.toml with pinned SDK version and local SDK hints when created from a checkout
6. Configures, builds the generated test target plus the default platform outputs, and runs tests
7. Leaves the project ready for pulp build, which you use for rebuilds, explicit targets, or optional deliverables after changing configuration
Notes:
- pulp create is meant to prove that a fresh machine and fresh project can scaffold, configure, build the default native outputs, and pass the generated tests.
- Use pulp build after pulp create when you want to rebuild, target a specific format/app target, or materialize optional deliverables after changing configuration.
- Browser targets such as WAM/WebCLAP are separate lanes and are not emitted by default by pulp create.
Default formats are platform-gated: - macOS: VST3, AU, CLAP, Standalone - Linux: VST3, CLAP, LV2, Standalone - Windows: VST3, CLAP, Standalone - app/bare: Standalone only
On macOS and Windows, AAX is optional. pulp create only scaffolds aax_entry.cpp
and includes the AAX target when an AAX SDK is already configured via
PULP_AAX_SDK_DIR. Linux and Ubuntu do not support AAX.
build¶
Status: usable
Configure and build the project. Auto-detects when CMake reconfiguration is needed. Works with both repo-based and standalone projects.
pulp build # Build all targets
pulp build --target PulpGain_VST3 # Build specific target
pulp build -j8 # Parallel jobs
pulp build --watch # Build and watch for changes
pulp build --watch --test # Build, watch, run tests on change
pulp build --allow-unsupported-sdk # Bypass the CLI-vs-project SDK guard (unsupported)
pulp build --check-identity # Verify .pulp/identity.lock before configure (Track 3.12)
pulp build --check-identity --allow-identity-change # Treat identity drift as a warning
Extra arguments are passed through to cmake --build.
--check-identity runs the same comparison as pulp identity check before the configure step, so a PR that changes an AU 4CC / VST3 FUID / CLAP id / AAX product code without re-recording the lock fails the build with a per-field diff. See docs/reference/identity-lock.md.
The --watch flag enters a file-watching loop after the initial build. It polls source files every 500ms and rebuilds on changes. Combine with --test to run tests after each successful rebuild and --validate to run quick dlopen checks.
For standalone projects (detected via pulp.toml), automatically sets CMAKE_PREFIX_PATH to the hinted local SDK when available, otherwise to the cached SDK release.
Before configure/build, pulp build also compares the active project's pinned sdk_version / cli_min_version against the running CLI. If the project is ahead, it fails fast and points at pulp upgrade; use --allow-unsupported-sdk only as an explicit unsupported escape hatch.
When pulp build decides a CMake reconfigure is required, it also runs the FetchContent cache preflight from pulp doctor --caches first. If the shared cache (~/Library/Caches/Pulp/fetchcontent-src/ on macOS, $XDG_CACHE_HOME/pulp/fetchcontent-src/ on Linux, %LOCALAPPDATA%/Pulp/fetchcontent-src/ on Windows) contains a dangling symlink or stale-commit entry, pulp build aborts with a one-screen remediation message instead of letting cmake blow up 200 lines into the configure log. Run pulp doctor --caches --fix to heal user-owned drift, or set PULP_SKIP_CACHE_PREFLIGHT=1 to bypass the gate.
On Windows, pulp build also selects a Visual Studio generator automatically when no active MSVC shell is detected on PATH.
test¶
Status: usable
Run the test suite via CTest. Builds first if no build directory exists. Works with both repo-based and standalone projects.
Extra arguments are passed through to ctest.
When pulp test triggers a cold-start build (no build/CMakeCache.txt), the FetchContent cache preflight from pulp doctor --caches runs first and aborts with a clear remediation message on any unhealthy entry — same gate pulp build applies, same PULP_SKIP_CACHE_PREFLIGHT=1 bypass.
status¶
Status: usable
Show project information for either SDK mode or source-tree mode.
pulp status reports which mode you are in so external projects never silently depend on a random checkout and repo/examples never silently pick up a cached SDK.
In SDK mode it also reports the pinned SDK version plus the resolved SDK path and checkout hints when present.
In source-tree mode it also reports the effective PR workflow (shipyard,
github, or manual) and whether the selected workflow's local tool is
available. Public Pulp installs do not install Shipyard or GitHub CLI; those
are contributor/source-checkout tools checked when the PR workflow needs them.
It also reports the effective pulp import-design defaults, including whether
they came from the built-in live/js default, ~/.pulp/config.toml, or
PULP_IMPORT_DESIGN_DEFAULT_* environment overrides.
On macOS and Windows it also reports whether an optional AAX SDK is detected. On
Linux and Ubuntu it reports AAX as unsupported.
validate¶
Status: usable
Run plugin format validators on all built plugins.
pulp validate # CLAP + VST3 (pluginval) + AU + optional AAX
pulp validate --all # Also run vstvalidator and full AAX validation if installed
pulp validate --json # Print JSON report to stdout
pulp validate --report out.json # Write JSON report to file
pulp validate --strict # CI gate: skipped-because-missing-tool ⇒ exit 1
Validator-discovery preflight. Before launching any validator,
pulp validate runs the same discovery that powers pulp doctor --validators.
If any validator on disk has a broken code signature (the classic case:
a copy of pluginval ripped out of its .app bundle, where amfid will
SIGKILL the process at launch with exit 137 and zero stderr), pulp
validate aborts with the exact path + remediation instead of letting
the run die mid-validation. Run pulp doctor --validators --fix to
clean up user-owned broken copies, or follow the printed sudo one-liner
for root-owned ones.
Missing-validator policy. If clap-validator, pluginval, auval,
or the AAX validator is not installed, affected plugins are reported as
SKIPPED. The default mode prints a loud warning listing which tools
are absent and how to install them — a green run without all four
validators is not the same as a run where all four passed. Use
--strict in CI (or any environment where partial coverage is a bug)
to treat those skips as hard failures.
Checks:
- CLAP: uses clap-validator if installed, otherwise falls back to CTest dlopen checks
- VST3: uses pluginval (strictness level 5, 30s timeout) if installed, otherwise skips
- AU: uses auval on macOS
- AAX: uses DigiShell + AAX Validator on macOS/Windows if installed via PULP_AAX_VALIDATOR_DIR or the recommended ~/SDKs/avid/aax-validator/ layout
- VST3 (--all): also runs vstvalidator if installed (Steinberg SDK tool, optional)
- AAX (--all): runs the broader aaxval test suite instead of the faster describe-validation probe
Flags:
- --all — run every available validator, including vstvalidator and full AAX validation
- --json — emit a machine-readable JSON report to stdout (conforms to validation-report-v1.schema.json)
- --report <path> — write the JSON report to a file
When a validator tool is not installed, the check is reported as SKIPPED with a clear message.
The JSON report conforms to docs/contracts/validation-report-v1.schema.json.
On Linux and Ubuntu, AAX validation is never attempted because AAX is unsupported there.
Prints a summary with pass/fail/skip counts.
run¶
Status: usable
Launch a standalone Pulp application from the build directory.
pulp run # find and launch first standalone binary
pulp run PulpGain # launch a specific target
pulp run MyApp -- --arg1 # pass arguments to the launched binary
pulp run --headless --screenshot ui.png # CI: render offscreen, save PNG
pulp run --headless --screenshot ui.png --frames 60 # render N frames before capture
pulp run --watch # re-launch on source-file changes
pulp run --audio-inspector # open the live Audio Inspector window
pulp run --audio-probe-json probe.json # dump live probe metrics as JSON, then exit
pulp run --audio-scope-json scope.json # dump live scope acquisition/measurements JSON
Searches the active project's build output:
- standalone projects: build/bin/
- in-repo examples: build/examples/
pulp run launches a standalone host, so it may activate the system audio
output even when the UI is headless. The CLI prints a pre-launch notice by
default; set PULP_RUN_AUDIO_NOTICE=0 only for deliberate quiet automation.
For no-speaker signal evidence, prefer the offline HeadlessHost / Audio
Doctor paths under pulp audio validate.
Headless / screenshot flags¶
--headless— run without a window. The CLI forwards--headlessto the launched binary and also setsPULP_HEADLESS=1, so binaries that read either source pick up the headless mode.--screenshot <file>— save a PNG to<file>. Implies--headless. Forwarded as--screenshot <file>and viaPULP_SCREENSHOT=<file>. If--headlessis given without an explicit screenshot path, the CLI defaults tobuild/<target>.png.--frames <n>— number of frames to render before capture. Default- Forwarded as
--frames <n>and viaPULP_FRAMES=<n>. --watch— re-launch the binary whenever a source file changes. Composes with--headless/--screenshotso the dev loop can re-render PNGs on every save.
These flags are intended for CI auto-validation: any plugin standalone
that respects PULP_HEADLESS / PULP_SCREENSHOT / PULP_FRAMES (or
the matching argv flags) can be exercised end-to-end on every PR
without a real window or virtual display.
Live Audio Inspector flags¶
The live Audio Inspector is a floating developer window that observes the
realtime output-boundary probe (peak/RMS/dBFS, clip/NaN counts, silence
runs). See the Audio Inspector guide for
the full picture, including the in-app Cmd/Ctrl+Shift+A chord and the
dev-on/ship-off PULP_ENABLE_AUDIO_PROBES gating.
--audio-inspector— open the live Audio Inspector window. Forwarded as--audio-inspectorand viaPULP_AUDIO_INSPECTOR=1. Does not imply--headless(a dev may want the visible window); composes with--screenshot, which then also captures the panel as<stem>.audio-inspector.png.--audio-probe-json <file>— the programmatic readout: after the frame delay, write the live probe's latest snapshot as a flat JSON object to<file>, then exit. Implies--headless. Forwarded as--audio-probe-json <file>and viaPULP_AUDIO_PROBE_JSON=<file>. This is visually headless, but it still launches the standalone host and may open the live audio device. It is distinct from the offlinepulp audio validateDoctor: the Inspector reads live metrics from a running host, the Doctor analyses a rendered WAV offline without speakers.--audio-scope-json <file>— the programmatic Scope readout: after the frame delay, write versionedpulp.audio.scope.v1JSON containing a copied sample-window acquisition plus measurements such as peak-to-peak, RMS, DC offset, crest factor, and conservative rising-zero frequency. Implies--headless, but still launches the standalone host and may open the live audio device. It cannot be combined with--audio-inspectorbecause both are single consumers of the live capture FIFO.--audio-scope-window <samples>/--audio-scope-trigger <none|raw|off|rising-zero>/--audio-scope-channel <index>— acquisition controls for--audio-scope-json.
Display-only waveform controls:
PULP_AUDIO_INSPECTOR_TRIGGER=rising-zero— opt into rising-zero-crossing trace alignment for a more stable visual display. Raw buffer display remains the default.PULP_AUDIO_INSPECTOR_GRID=0— hide the waveform grid.PULP_AUDIO_INSPECTOR_SCALE=<n>— zoom the horizontal window over the copied real samples (1.0shows the full captured buffer).
cache¶
Status: usable
Manage the Pulp SDK and asset cache at ~/.pulp/ by default.
pulp cache # Show help
pulp cache status # Show cached SDKs and assets with sizes
pulp cache fetch skia # Download Skia GPU rendering binaries
pulp cache clean # Remove all cached assets
Subcommands:
| Subcommand | What it does |
|---|---|
status |
List cached SDK versions and downloaded assets |
fetch skia |
Download platform-specific Skia GPU binaries to ~/.pulp/cache/ by default |
clean |
Remove all files from the asset cache |
GPU rendering requires Skia binaries. If a standalone project enables GPU features, run pulp cache fetch skia to download them.
Set PULP_HOME to relocate the cache, SDK, and config root.
doctor¶
Status: usable
Diagnose environment issues. Checks C++20 compiler, CMake version, git-lfs, LFS file state, external SDKs (VST3, AudioUnit), platform-specific dependencies, and the expected project mode.
pulp doctor # show all checks
pulp doctor --fix # auto-fix issues where possible
pulp doctor --ci # non-interactive, exit codes only
pulp doctor --dry-run # show what --fix would do
pulp doctor --versions # CLI/SDK/Plugin version diagnostics
pulp doctor --versions --scan-parents # ALSO walk CWD ancestors for pulp_add_* projects
pulp doctor --versions --json # emit the diagnostic as stable JSON
pulp doctor --validators # discover auval / pluginval / clap-validator + verify signatures
pulp doctor --validators --fix # ALSO remove user-owned broken copies; root-owned breakage prints sudo one-liner
pulp doctor --validators --dry-run # preview what --fix would do
pulp doctor --caches # FetchContent shared-source cache health
pulp doctor --caches --fix # heal user-owned dangling/stale-commit entries
pulp doctor --caches --fix --dry-run # preview heal without removing anything
pulp doctor --caches --json # emit the cache report as stable JSON
pulp doctor --host-quirks # show the runtime DAW host-quirks policy + enforced accommodations
pulp doctor quirks # synonym for --host-quirks
pulp doctor --host-quirks reports whether Pulp is enforcing DAW
host-quirk accommodations and under which policy. It prints the effective
tier policy (off / validated-only / all) and where it came from
(compile-time default, the PULP_HOST_QUIRKS env var, or a programmatic
set_host_quirk_policy() call), the detected host + version, and the list
of currently-enforced accommodations with their validation tier. The same
section is appended to the default pulp doctor output.
Override the policy at runtime without recompiling:
PULP_HOST_QUIRKS=off pulp doctor --host-quirks # disable all accommodations
PULP_HOST_QUIRKS=validated-only pulp doctor --host-quirks # only bench-validated fixes
PULP_HOST_QUIRKS=all pulp doctor --host-quirks # every detected quirk (default)
Per-quirk provenance — source_type, evidence, and last_verified
dates — lives in core/format/host-quirks.json. See
host-quirks policy for the full opt-in / opt-out
story and the precedence rules.
Checks are platform-gated — only relevant checks run on each OS: - macOS: git, compiler, CMake, git-lfs, LFS files, VST3 SDK, AudioUnitSDK, optional AAX SDK/validator, build state - Linux: git, compiler, CMake, git-lfs, LFS files, VST3 SDK, ALSA dev headers, build state - Windows: git, compiler, CMake, git-lfs, LFS files, VST3 SDK, optional AAX SDK/validator, build state
Mode-specific checks:
- SDK mode: verifies pulp.toml, the installed SDK path or cache, optional checkout hints, and build configuration for the external project
- Source-tree mode: verifies the active checkout, pinned external SDKs, LFS state, and build configuration for the repo/examples workflow
For AAX-specific setup details and download guidance, see AAX Setup.
Exit code is 0 if all checks pass, 1 if any fail.
pulp doctor --versions is a dedicated diagnostic (not a drift-check)
that prints the three surface versions side-by-side plus advisory
skew warnings:
- CLI — the version baked into the running
pulpbinary - Plugin — read from
.claude-plugin/plugin.jsonin the active repo (falls back to~/.claude/plugins/pulp/plugin.jsonfor installed plugin layouts) - SDK — the active project's SDK version, from
pulp.toml(standalone projects) orCMakeLists.txt(source-tree mode)
Skew warnings fire when the project's pulp.toml declares a
cli_min_version higher than the installed CLI, or when the
project SDK is newer than the running CLI. Warnings are advisory
— the command always exits 0 so it's safe to wire into scripts.
The first release that defines the following optional pulp.toml
field activates the cli_min_version check:
Untagged CLI builds (anything not matching M.N.P exactly) are
skipped silently, matching the design's forward-compatible
convention.
Multi-project skew. When ~/.pulp/projects.json
contains registered projects, pulp doctor --versions lists each
project with its own SDK / cli_min_version pair and surfaces any
skew inline. Entries whose path no longer exists are shown with a
(missing) tag and a pulp projects remove <path> hint — we never
auto-prune; only explicit removal mutates the registry.
--scan-parents is an opt-in escape hatch for projects that were
never registered (for instance, a cloned example). It walks the
current directory's ancestor chain looking for CMakeLists.txt
files that invoke any pulp_add_* macro and surfaces those matches
in-line with a (scanned) tag. Ancestor hits are NOT added to the
registry — the design decision is "registry is authoritative,
ancestor scan is a diagnostic escape hatch."
If both a parent directory and a nested child directory contain a
pulp_add_* invocation, both appear in the report (deepest-first —
closest ancestor to the current directory comes first). Resolving
"which is the canonical project" is the user's call; the diagnostic
surfaces both so the ambiguity is visible.
--json emits the same information as a single JSON object with a
stable shape — {"cli": {...}, "plugin": {...}, "plugin_min_cli":
{...}, "project_sdk": {...}, "projects": [{"path": ..., "sdk": {...},
"cli_min": {...}, "missing_on_disk": bool, "scanned": bool}, ...],
"findings": [...]}. Scripts should use the findings[] array for
user-visible warnings; per-field semver fields carry
comparable: true only when they parse as pure M.N.P.
plugin_min_cli is populated from the plugin's plugin.json
min_cli_version field; absent in
older plugin builds.
--validators¶
pulp doctor --validators is a dedicated diagnostic that discovers the
three plugin-format validators pulp validate shells out to —
auval, pluginval, clap-validator — and verifies each candidate's
code signature is intact. It catches the failure mode where a binary
copied OUT of its .app bundle (commonly /usr/local/bin/pluginval
copied from /Applications/pluginval.app/Contents/MacOS/) retains a
signature claim that references peer files inside the bundle. macOS
amfid kills such a binary at launch with exit 137 and zero stderr —
the user has no diagnostic without already knowing this exists.
Per-validator output is one of:
OK auval: /usr/bin/auval (valid signature on disk …)FAIL pluginval: /usr/local/bin/pluginval — invalid Info.plist (plist or signature have been modified). Root-owned — sudo required.WARN clap-validator: not installed. Runcargo install clap-validator.
--fix removes broken user-owned copies in place (counted in the
Auto-fixed summary). Broken root-owned copies are never
auto-elevated — the doctor prints a sudo rm <path> one-liner for the
user to run manually. --fix --dry-run previews the same actions
without mutating anything. --fix is a no-op on a fully healthy env.
Discovery walks each validator's well-known paths in priority order
(system path → cask app bundle → PATH lookup → ~/.cargo/bin) and
stops at the first existing candidate. The first-existing path wins
deliberately: that's the binary the user's shell will dispatch, so
that's the copy pulp validate will SIGKILL on if it's broken.
Masking it with a healthy copy further down the list would defeat the
diagnostic.
Exit code is 0 only when every validator is Healthy. Missing validators
also contribute to a non-zero exit because a host without the validators
can't run pulp validate at all — the doctor must surface that.
pulp validate runs the same discovery as a preflight before launching
any validator. If any validator is in the Broken state, pulp validate
aborts with the exact remediation instead of letting amfid SIGKILL the
run mid-validation.
FetchContent cache health. pulp doctor --caches audits
the shared-source FetchContent cache that Pulp uses to avoid
re-cloning external SDKs across builds. The cache root is the same
path pulp_register_fetchcontent_source populates:
- macOS:
~/Library/Caches/Pulp/fetchcontent-src/ - Linux:
$XDG_CACHE_HOME/pulp/fetchcontent-src/(default~/.cache/pulp/fetchcontent-src/) - Windows:
%LOCALAPPDATA%/Pulp/fetchcontent-src/
Each cache entry is classified as one of:
| Status | Meaning | --fix action |
|---|---|---|
[ok] |
Healthy — entry exists and (if a symlink) target exists, cached REF matches the declared pulp_register_fetchcontent_source(... REF ...) in the active project's CMakeLists.txt. |
(no-op) |
[!!] dangling-symlink |
Entry is a symlink whose target no longer exists. CMake's FETCHCONTENT_SOURCE_DIR_* override would fail at configure time. |
rm the symlink — next configure refetches. |
[!!] stale-commit |
The directory name's REF suffix differs from the declared REF. The pin in CMakeLists.txt advanced but the user's old cache is still authoritative. |
rm -rf the entry — next configure refetches. |
[!!] root-owned |
Entry is not user-writable (likely owned by root from a stray sudo). |
Reported only; agent never tries to sudo rm. Manual: sudo rm -rf <path>. |
Exit code is 0 when every entry is [ok], 1 otherwise. The same
discovery code runs as a preflight inside pulp build and pulp test
when a CMake reconfigure is needed — set PULP_SKIP_CACHE_PREFLIGHT=1
to bypass the gate (intended for sealed CI environments that can't
auto-heal).
--caches --fix removes only user-owned entries marked [!!]
dangling-symlink or [!!] stale-commit. Root-owned entries are
report-only by design — automatic sudo is out of scope so agents
don't silently elevate. --caches --fix --dry-run previews what
would be removed without touching the filesystem.
--caches --json emits a stable shape:
{"cache_root": "...", "healthy": bool, "entries": [{"name", "path",
"status", "is_symlink", "resolved_target", "declared_ref",
"cached_ref", "dep_name", "reason", "remediation", "fixable"}, ...]}.
The status field uses the lowercased label set above
(healthy, dangling-symlink, stale-commit, root-owned,
unknown).
projects¶
Status: usable
Manage the ~/.pulp/projects.json registry. pulp create and
pulp new register new projects automatically on successful
scaffold; these commands exist so users can add projects created
outside of pulp create (clones, manual checkouts) and prune stale
entries. Registry entries are read by pulp doctor --versions to
produce per-project skew reports.
pulp projects list # show registered projects
pulp projects list --json # machine-parseable JSON output
pulp projects add # register the current directory
pulp projects add ~/code/my-plugin # register a specific directory
pulp projects remove ~/code/old-plugin # forget a project by path
--json emits the same shape the Rust CLI port emits, so cross-binary consumers see byte-identical output. Schema: {registry, projects: [{path, name, registered_at, missing_on_disk}]}. missing_on_disk is true when the project directory has been deleted since registration.
The registry is a plain JSON file with one top-level projects
array; each entry has path, name, and registered_at. The
location is $PULP_HOME/projects.json (defaulting to
~/.pulp/projects.json). A missing registry file is treated as an
empty list — no first-run setup is required.
project¶
Status: usable
Per-project SDK pin management. Updates a consumer project's pinned
Pulp SDK version and records an undo batch at
~/.pulp/bump-undo-<timestamp>.json so mistakes are one command away
from recovery.
In standalone SDK-mode projects (pulp.toml present), the SDK pin is
pulp.toml sdk_version. pulp project bump updates that field and
the versioned find_package(Pulp X.Y.Z ...) line together. It does
not rewrite project(NAME VERSION ...); that remains the app/plugin
product version. If sdk_path points at a managed Pulp SDK cache for
the old version, it is moved to the matching new cache path. Custom
sdk_path values are left alone and verified later by pulp build.
In legacy source-embedded projects, the command recognizes
FetchContent_Declare(pulp ... GIT_TAG vX.Y.Z),
pulp_add_project(NAME VERSION X.Y.Z ...), and
project(NAME VERSION X.Y.Z ...).
pulp project bump # bump CWD project to CLI's own version
pulp project bump 0.32.0 # bump to explicit version (positional)
pulp project bump --to=0.32.0 # bump to explicit version (named)
pulp project bump --all # iterate ~/.pulp/projects.json
pulp project bump --all --dry-run # show plan without writing
pulp project bump --force-dirty # skip the git-clean check
pulp project bump --allow-downgrade # target older than current pin
pulp project bump --allow-cli-skew # target newer than installed CLI
pulp project bump --allow-redundant # ignore origin/main already-newer guard
pulp project bump --verify-builds # build after bump; roll back on failure
pulp project undo # revert the newest batch
pulp project undo <timestamp> # revert a specific batch
Cross-binary parity: pulp project bump and pulp project undo round-trip byte-exactly between the C++ and Rust CLI implementations. A bump written by one binary's bump is correctly understood by the other binary's undo, including the optional notes:[...] field the Rust port emits. The C++ undo-batch parser silently skips unknown ARRAY / OBJECT fields it doesn't recognize so future schema additions don't desync the parser.
Safety rails: branch pins (GIT_TAG main) and SHA pins are
skipped with a diagnostic; dirty pin-bearing files are gated behind
--force-dirty; target older than current is gated behind
--allow-downgrade; target newer than the installed CLI is gated
behind --allow-cli-skew; worktrees where origin/main already pins
the target-or-newer SDK are skipped unless --allow-redundant is set;
and --all isolates per-project failures so one broken project
doesn't abort the rest. Running inside the Pulp source checkout is
refused because that is a framework release/version operation, not a
consumer project SDK bump.
Migration notes print after a successful bump so users see any API changes the hop introduced.
When to use pulp upgrade vs pulp project bump: use
pulp upgrade to replace the installed Pulp CLI/SDK toolchain. Use
pulp project bump after that when the current project should move to
that SDK. The Claude /upgrade flow exposes this as "upgrade the
tool" vs "upgrade the tool and bump this project's SDK pin".
Post-upgrade hook: the update.bump_projects config key (prompt
| auto | off; default prompt) controls whether pulp upgrade prints
a pulp project bump --all hint after a successful CLI upgrade.
ci-host¶
Status: experimental
Optional, discoverable wrapper around tools/ci/setup-ci-host.sh for onboarding a
Mac as a Tart-VM CI host (install prereqs, create the local VM stores, register the
host-class runner label, optionally copy a golden in and run a one-shot validation
build). This is an advanced/contributor path — never required, and Shipyard
stays encouraged-not-mandated. The real work lives in the script; the command just
makes it discoverable and forwards flags.
pulp ci-host setup --class m5 # minimum: register the m5 host-class label
pulp ci-host setup --class m5 --copy-from 'macstudio:/Volumes/Workshop/VMs/vms/pulp-build-runner:latest'
pulp ci-host setup --class m5 --validate # also run a one-shot VM build to prove it
pulp ci-host setup --help # full flag list (delegated to the script)
Common flags (forwarded verbatim to setup-ci-host.sh):
--class <name>— required host class for the runner label (m5,studio,macbook, …)--copy-from <ssh:path | path>— rsync a golden in from another host/drive (sparse-safe)--validate— after setup, run one ephemeral VM build on the host-only label--no-agent— do everything except install/load the launchd agent
Runs from inside a Pulp checkout (it resolves tools/ci/setup-ci-host.sh). For the
from-scratch host recipe and gotchas, see
mac-ci-host-setup.md and the tart-ci skill.
macos¶
Status: experimental
Retarget just the macOS leg of a PR without disturbing the Linux/Windows matrix. This is an operator command for cases where a PR should move between local, Namespace, or GitHub-hosted macOS capacity.
pulp macos status --pr 1910
pulp macos retarget --pr 1910 --to local
pulp macos retarget --pr 1910 --to namespace
pulp macos retarget --pr 1910 --to github-hosted
retarget cancels in-flight macOS-bearing runs for that PR and dispatches
build-macos.yml on the selected runner pool. Branch protection is satisfied by
the latest workflow that publishes the required macos check. See
local-ci.md for the
runner variables and operator workflow.
overflow¶
Status: experimental
Configure the macOS overflow routing variables read by build.yml. This is a
repo-operator surface for deciding where new macOS jobs go when local capacity is
busy; it does not cancel in-flight jobs.
pulp overflow status
pulp overflow enable
pulp overflow enable --to '"namespace-profile-generouscorp-macos"'
pulp overflow disable
pulp overflow threshold
pulp overflow threshold 1
enable sets the overflow target, disable removes it for future dispatches,
and threshold gets or sets the busy-run count that trips overflow. See
local-ci.md for the
exact repository variables and rollback notes.
ci-local¶
Status: legacy (prefer Shipyard)
Note: For most CI workflows, use
shipyard runfor validation andshipyard prfor PR creation/shipping/tracking instead ofpulp ci-local. Shipyard is Pulp's primary CI tool and provides the same target matrix with evidence-gated merges.pulp ci-localremains available as an advanced fallback while legacy workflows are still supported.
Local-first CI control plane for Pulp. This is the shared operator surface for:
- machine-global local/SSH queueing
- exact-SHA validation on this Mac and configured hosts
- deliberate GitHub Actions dispatch/status when cloud orchestration is needed
pulp ci-local run
pulp ci-local run --smoke
pulp ci-local check 123
pulp ci-local status
pulp ci-local cleanup
pulp ci-local cleanup --dry-run
pulp ci-local cleanup --apply
pulp ci-local cloud workflows
pulp ci-local cloud defaults
pulp ci-local cloud history
pulp ci-local cloud compare build
pulp ci-local cloud recommend build
pulp ci-local cloud run build feature/my-branch
pulp ci-local cloud run build feature/my-branch --provider namespace
pulp ci-local cloud run build feature/my-branch --provider namespace --macos-runner-selector-json '"namespace-profile-big-apple"'
pulp ci-local cloud run build feature/my-branch --provider namespace --macos-runner-selector-json '"nscloud-macos-tahoe-arm64-6x14"'
pulp ci-local cloud run docs-check feature/my-branch --provider namespace --wait
pulp ci-local cloud run docs-check feature/my-branch --provider namespace --runner-selector-json '"namespace-profile-big-apple"'
pulp ci-local cloud namespace doctor
pulp ci-local cloud namespace setup
pulp ci-local cloud status latest --refresh
Local queue commands:
run— queue validation and wait for completioncheck— queue validation for an existing PRship— push, open PR, queue CI, merge on greenenqueue/drain/bump/cancel— queue managementlogs/evidence/status— saved results and operator visibilitycleanup— inspect or prune retained local-CI artifacts; dry-run by default,--applyis blocked while jobs are running, and--include-preparedalso removes cached build/install state that later reruns will rebuild
Cloud companion commands:
cloud workflows— list the GitHub workflows and supported runner providers known to this checkoutcloud defaults— show the effective workflow/provider defaults plus where current selector values came from (local config versus repo-variable fallback)cloud history— show recent tracked cloud runs plus any configured billing-period rollupcloud compare [workflow]— compare observed cloud providers for one workflow using recorded history, including latest success timingcloud recommend [workflow]— recommend a cloud provider from recorded historycloud run [workflow] [branch]— dispatch a GitHub Actions workflow by branch;docs-checkaccepts--runner-selector-json, whilebuildalso accepts one-off--linux-runner-selector-json,--windows-runner-selector-json, and--macos-runner-selector-jsonoverrides for per-leg routingcloud status [dispatch-id|latest]— show tracked GitHub run state plus queue-delay/elapsed timing when available; Namespace-backed runs also report provider runtime/machine-shape truth whennsccan match the instances;--refreshre-queries GitHub for the selected runcloud namespace doctor— verify thatnscis installed, login is valid, and the current workspace is visiblecloud namespace setup— thin wrapper that runsnsc loginif needed and then shows the same Namespace status
Current cloud scope:
- GitHub Actions remains the orchestrator
docs-checkis the first runner-provider pilot and supportsgithub-hostedandnamespacebuildnow also supportsgithub-hostedandnamespace; the default cloud build covers Linux and Windows only so macOS can stay local-firstdocs-checkcan use an explicit--runner-selector-jsonoverride or a docs-check-specific local config default before falling back to the repo Namespace selector variablebuildcan take Linux/Windows Namespace selectors from the local config keysgithub_actions.workflows.build.providers.namespace.linux_runner_selector_jsonand.windows_runner_selector_json, or from the repo variablesPULP_NAMESPACE_BUILD_LINUX_RUNS_ON_JSONandPULP_NAMESPACE_BUILD_WINDOWS_RUNS_ON_JSON- macOS Namespace is opt-in for
build: set--macos-runner-selector-jsonfor a one-off run, or setgithub_actions.workflows.build.providers.namespace.macos_runner_selector_jsonlocally orPULP_NAMESPACE_BUILD_MACOS_RUNS_ON_JSONif you want an explicit macOS Namespace validation run - selector overrides can use either a Namespace profile label like
"namespace-profile-generouscorp-macos"or a direct machine label like"nscloud-macos-tahoe-arm64-6x14" - that selector must point at a real macOS-capable Namespace profile: GitHub labels and matrix names alone do not prove the underlying OS, so a Linux Namespace profile can appear as a macOS leg while actually executing on Linux
- if macOS should remain local by default, keep the shared macOS selector unset
and use
--macos-runner-selector-jsononly for one-off cloud validation runs - if you plan to use the Namespace provider, install the
nscCLI and runnsc loginfirst; that is the recommended operator setup path for this phase - VM/SSH target configuration and Namespace provider configuration remain separate: local/SSH hosts stay in the normal local CI target config, while Namespace routing and login state live behind the
cloud namespacehelper surface validateandsanitizersremaingithub-hostedonly in this phase- cloud dispatch records are persisted beside local CI state, but they do not enter the local queue
statusincludes recent tracked cloud summaries without contacting GitHub; usecloud status --refreshwhen you want live GitHub state- tracked cloud runs now persist queue-delay and elapsed-duration timing so later comparison commands can report real provider speedups instead of ad hoc estimates
- estimated cost reporting is optional and local-config driven; every derived number is labeled
estimated; verify provider pricing - provider-reported billing totals are opt-in and off by default; when enabled, Pulp shows them separately from tracked-run estimates because they are repo-wide current-period figures
- if the provider CLI does not expose billing totals, Pulp still reports runtime and machine shape instead of inventing invoice truth
statusalso reports the current local-CI footprint for bundles, prepared state, logs, results, and tracked cloud runscleanupsupports the operator-facing retention workflow: inspect reclaimable space first, then re-run with--applyonly when no local CI job is active
Namespace profile setup note:
nscis enough for login verification and instance/history inspection, but GitHub Actions runner-profile creation is still a Namespace dashboard step in this phase- create new profiles under
GitHub Actions -> Profiles - the UI profile name omits the GitHub selector prefix; for example a profile
shown as
generouscorp-macosis referenced from Pulp as"namespace-profile-generouscorp-macos" - for ad hoc runs, Namespace also supports direct machine labels without a saved
profile, for example
"nscloud-macos-tahoe-arm64-6x14" - after creating a one-off macOS profile, validate it with:
pulp ci-local cloud run build <branch> --provider namespace --macos-runner-selector-json '"namespace-profile-generouscorp-macos"' - confirm the backing shape with
nsc instance history --all -o json; a valid macOS profile should reportshape.os = "macos"andshape.machine_arch = "arm64"
harness¶
Run the catalog coverage harness and deterministic visual snapshot harness. Coverage mode delegates to tools/harness/verifier.py and compares compat.json support claims against machine-derived oracles. Visual mode delegates to tools/harness/visual/runner.py and compares fixture-declared JSON or PNG captures against checked-in goldens.
pulp harness --surface=yoga
pulp harness coverage --surface=yoga
pulp harness --all
pulp harness --surface=yoga --json
pulp harness --surface=yoga --no-docs
pulp harness visual --verify --all
pulp harness visual --verify --all --actuals-dir build/visual-actuals
pulp harness visual --generate --surface=yoga --entry=yoga/box-sizing
By default, coverage mode writes build/harness-coverage-<sha>.json, build/harness-coverage.md, and docs/reports/harness-coverage.md. Use --json for stdout-only machine output. Coverage JSON includes validation_routes per surface for supported entries with a typed validation route, legacy test reference, or explicit exclusion. It also includes visual_coverage, counted from typed runtime fixture refs that resolve to checked-in goldens; visual_pass remains as a compatibility alias.
Visual mode requires the pulp-test-visual target to be built first. Use --build-dir or --binary when the binary is not under the default build/ or build-visual/ directories. Fixture metadata decides whether a golden is semantic JSON or raster PNG. JSON snapshots use tolerance-aware semantic diffs; PNG snapshots use exact-byte comparison on the canonical raster lane. When verification fails, pass --actuals-dir build/visual-actuals to write failed actual JSON/PNG captures under build/visual-actuals/<surface>/<fixture>.<json|png> for inspection or artifact upload.
ship¶
Status: experimental
Signing and packaging subcommands.
pulp ship sign --identity "Developer ID Application: ..."
pulp ship sign --identity "..." --entitlements path/to/entitlements.plist
pulp ship sign --identity "..." --path MyApp.app # sign one explicit artifact
pulp ship package --version 1.0.0
pulp ship check
pulp ship doctor # make signing non-interactive (no keychain/1Password prompt)
pulp ship notarize --api-key ~/key.p8 --api-key-id ABC --api-issuer <uuid>
pulp ship notarize --path MyApp-1.0.dmg # notarize + staple one artifact
pulp ship notarize --dry-run # print resolved argv, no submit
pulp ship share MyApp.app --identity "..." # one-shot: sign+notarize+verify
Subcommands:
| Subcommand | What it does |
|---|---|
sign |
Code-sign all built plugin bundles (VST3, CLAP, AU), or one --path artifact |
notarize |
Submit signed bundles (or --path artifacts) to Apple notarytool (macOS) |
package |
Create macOS .pkg/.dmg installers or Linux .deb/.tar.gz packages in artifacts/ |
release |
macOS one-command pipeline: sign → package → notarize the .pkg/.dmg it builds → staple |
share |
One-shot for sharing a single artifact: sign → wrap .app in DMG → notarize → staple → Gatekeeper-verify |
check |
Check signing status of all built plugins |
doctor |
Make signing+notarization non-interactive (no keychain/1Password prompt): self-heal the dedicated signing keychain and validate the file-based .p8 notary key. Run automatically as a best-effort preflight by sign. |
doctor materializes a dedicated signing keychain authorized for codesign (so the login keychain / 1Password is never consulted) and validates a file-based App Store Connect .p8 notary key. --check-online also proves the .p8 against Apple (read-only) and refreshes the optional pulp-notary keychain profile; --print-env emits resolved identity/keychain handles (no secret values). Secrets live in ~/.config/pulp/secrets/ (keychain.env + notary.env), never in the repo; same-named env vars override the files. No build directory is required.
sign requires --identity. The default entitlements file is ship/templates/entitlements.plist.
--path signs exactly one .app, .dmg, or plugin bundle instead of scanning the build dirs;
.pkg installers are signed at creation time with a Developer ID Installer identity, not here.
package creates per-format .pkg files using pkgbuild on macOS, or .dmg files with --dmg. On Linux, it packages VST3/CLAP/LV2 bundles as a .deb using dpkg-deb, with a .tar.gz fallback when dpkg-deb is unavailable. If no Linux plugin bundles are present, it reports no VST3/CLAP/LV2 plugins found instead of creating an empty macOS-style artifact summary.
pulp ship share — one-off "sign it for a friend"¶
share is the opinionated, single-command path for handing a build to someone
without running the full release pipeline. Point it at a .app, .dmg, or
.pkg:
pulp ship share MyApp.app --identity "Developer ID Application: Name (TEAMID)"
pulp ship share MyApp.app --dry-run # print the plan, do nothing
For a .app it code-signs (hardened runtime + secure timestamp), wraps it in a
DMG under artifacts/, signs the DMG, notarizes + staples it, then runs the
exact spctl -a -t open --context context:primary-signature check Gatekeeper
performs on download. A green result means the recipient will not see
"Unnotarized Developer ID". Notarization credentials resolve through the same
chain as pulp ship notarize (App Store Connect API key preferred). .dmg
inputs skip the wrap step; .pkg inputs are assumed already installer-signed
and are only notarized + verified.
release --dmg/--pkg notarizes and staples the distributable it produces, so
the artifact it leaves in artifacts/ is Gatekeeper-ready, not merely signed.
pulp ship notarize¶
Two credential lanes are supported. The App Store Connect API key flow is preferred (Apple's modern, scope-controlled path); the legacy Apple-ID + app-specific-password flow remains as a fallback for existing users.
Preferred — App Store Connect API key (xcrun notarytool submit
--key/--key-id/--issuer):
| Flag | Env var | notary.env key |
|---|---|---|
--api-key |
PULP_NOTARY_KEY_PATH |
PULP_NOTARY_KEY_PATH |
--api-key-id |
PULP_NOTARY_KEY_ID |
PULP_NOTARY_KEY_ID |
--api-issuer |
PULP_NOTARY_ISSUER_ID |
PULP_NOTARY_ISSUER_ID |
Legacy (xcrun notarytool submit --apple-id/--team-id/--password):
| Flag | Env var | config.toml |
|---|---|---|
--apple-id |
PULP_APPLE_ID |
signing.apple.apple_id |
--team-id |
PULP_TEAM_ID |
signing.apple.team_id |
--password |
— | signing.apple.password (default @keychain:AC_PASSWORD) |
Resolution precedence (highest wins): CLI flag → environment variable →
~/.config/pulp/secrets/notary.env (override path via PULP_NOTARY_ENV or
--env-file <path>) → ~/.pulp/config.toml (legacy fields only).
The ASC lane wins when all three pieces resolve. Otherwise the legacy lane
applies when --apple-id + --team-id resolve. If neither lane is complete,
the command prints both setup recipes and exits non-zero.
Other flags: --staple (skip submission, staple already-notarized bundles),
--dry-run (print the resolved xcrun notarytool argv and exit 0; never
contacts Apple — useful in CI and for verifying credential resolution).
pr¶
Status: usable
Create, validate, and merge a PR through the canonical ship flow.
shipyard pr is the primary "ship this" orchestrator referenced by the CI
skill. pulp pr remains a compatibility wrapper that delegates to
shipyard pr by default, with explicit github and manual workflows for
humans who opt out of Shipyard in their local checkout. Use --native only
for diagnostics when debugging the CLI-side fallback path. Natural-language
triggers in agent conversations
("push to main", "ship this", "ship it", "we're done", "merge this",
"push it", "run CI", "push a PR") all route here — see the
CI skill (.agents/skills/ci/SKILL.md)
for the authoritative trigger list.
shipyard pr
pulp pr
pulp pr --base origin/main
pulp pr --title "feat(cli): document pulp pr and sync CI policy"
pulp pr --workflow github
pulp pr --workflow manual
pulp pr --no-ship
pulp pr --no-push
pulp pr --dry-run
# Fallback when the local `pulp` binary is broken (for example wgpu
# dylib load failure). Equivalent for the default ship cycle:
shipyard pr
Flags:
| Flag | Description |
|---|---|
--base <ref> |
Diff base (default: origin/main) |
--title <s> |
PR title (default: tip commit subject) |
--workflow <m> |
One-shot workflow override: shipyard, github, or manual |
--no-ship |
Diagnostics-only native fallback flag; do not use as the normal PR path |
--no-push |
Stop after bump commit; do not push or create PR |
--dry-run |
Print the plan without executing steps |
-h, --help |
Show help |
Notes:
- Default behavior is shim delegation to
shipyard pr— the single source of truth for ship orchestration. Do not treatgh pr create+shipyard shipas a substitute; that sequence bypasses the skill-sync and version-bump gates and can leave a PR outside Shipyard's tracked state. - The PR workflow is resolved in this order:
--workflow, thenPULP_PR_WORKFLOW, then~/.pulp/config.toml[pr] workflow, then the defaultshipyard. githubmeans direct GitHub CLI mode throughgh; it requires an installed and authenticatedghand does not create Shipyard tracking state.manualprints the intended commands and exits before pushing or creating a PR. It is for people who want to use the GitHub UI, forks, or other tooling.shipyardis intentionally not silently downgraded togithubwhen the Shipyard binary is missing. Install the pinned source-checkout tool with./tools/install-shipyard.sh, or choose another workflow explicitly.--nativeruns an in-CLI fallback that performs the same gates + PR flow without delegating to Shipyard. Diagnostic use only.- Direct
gh pr createis an explicit emergency/manual bypass only. If used, document the Shipyard tracking gap and reconcile by resuming or re-shipping through Shipyard when possible. - For the canonical list of natural-language ship triggers and the full
policy, see the CI skill
(
.agents/skills/ci/SKILL.md).
docs¶
Status: usable
Browse local documentation and status manifests. All subcommands read from local files in docs/ only -- no web calls.
pulp docs # Show help
pulp docs index # List available docs
pulp docs search <query> # Search docs for a string
pulp docs open <slug> # Print a doc by slug
pulp docs show support <thing> # Look up support status
pulp docs show command <name> # Look up a CLI command
pulp docs show cmake <name> # Look up a CMake function
pulp docs show style # Show code style rules
pulp docs check # Validate docs consistency
Subcommands:
| Subcommand | What it does |
|---|---|
index |
Print a readable list of available docs from docs-index.yaml |
search <query> |
Case-insensitive search across all Markdown files in docs/ |
open <slug> |
Resolve slug via docs-index.yaml and display the file |
show support <thing> |
Look up platform/format/subsystem support from support-matrix.yaml |
show command <name> |
Look up a CLI command from cli-commands.yaml |
show cmake <name> |
Look up a CMake function from cmake-functions.yaml |
show style |
Display style rules from style-rules.yaml with links to policy docs |
check |
Validate docs consistency: manifest links, index completeness, status vocabulary, module dependencies vs CMake |
design¶
Status: experimental
Launch the local AI-powered design tool used for token, shader, and style iteration.
pulp design
pulp design path/to/design-tool-core.js
pulp design --script path/to/design-tool-core.js
pulp design --build-dir /tmp/pulp-design-parity-build
pulp design now configures/builds pulp-design-tool on demand before launch. When run inside
a Pulp checkout it loads examples/design-tool/design-tool-core.js from that checkout and builds
into that checkout's build/ directory. The design-tool UI is split across design-tool-*.js
concern modules; the entry module is the path you pass, and the host loads its sibling modules in
order from the same directory.
Use --script to point at a different JS entry, and --build-dir when you are working from a
nonstandard build tree such as a separate worktree build directory.
When run outside a Pulp checkout, pulp design can currently auto-bind only when the CLI
binary lives inside a Pulp build tree such as .../build/pulp (Rust) with its
.../build/tools/cli/pulp-cpp delegate. Generic PATH-installed or symlinked CLI setups
are not fully SDK-mode aware yet; use --build-dir and --script explicitly in split
layouts where the project repo and the Pulp SDK live in different directories.
The selected build environment is the authority for supported behavior. pulp design prints the
chosen root, build dir, and script path so the provenance is explicit.
The design tool chat now supports provider/model-aware local execution in the UI.
The current app exposes a provider selector (Claude, Codex), a model selector,
and a reasoning-effort selector for Codex/OpenAI models.
design-debug¶
Status: experimental
Run the before/after/diff harness for design-chat prompts. This is the
automation/debug companion to pulp design.
pulp design-debug --prompt "make the gain knob look like macOS 7" --target k1
pulp design-debug --prompt "design a cyberpunk interface for a modern synth plugin" --target all --provider claude --model claude-sonnet-4-6
pulp design-debug --prompt "make the gain knob look like a precision analyzer control" --target k1 --provider codex --model gpt-5.4 --reasoning-effort xhigh
pulp design-debug --prompt "warm analog EQ" --target all --response-file saved-response.json
pulp design-debug --prompt "make the gain knob look like premium brushed aluminum" --target k1 --capture-backend live-gpu
Artifacts are written by default under build/design-debug/:
- *-before.png
- *-after.png
- *-diff.png
- *-target-before.png, *-target-after.png, *-target-diff.png for targeted runs
- *-prompt.txt
- *-response.txt
- *-debug-state.json
- *-apply-summary.txt
- *-report.json
- latest-report.json
- latest-run.json
- runs.jsonl
The JSON report records:
- provider, model, reasoning_effort
- target and target_bounds
- target_region, target_region_source, and target-only diff stats (target_diff_pixels, target_diff_pct) when a widget target is selected
- debug_state from the design tool (changedColors, changedDimensions, widgetLookIds, summary, request text)
- the exact ai_command or live driver_command used for local execution
- screenshot-diff stats (similarity_pct, diff_pixels, mean_error)
Useful flags:
- --provider claude|codex
- --model <name>
- --reasoning-effort low|medium|high|xhigh
- --capture-backend skia|coregraphics|live-gpu
- --response-file <json-or-text> to replay a saved model response without calling AI
- --script <path> to load a custom design-tool JS file
- --design-tool-bin <path> to point live-gpu runs at a built pulp-design-tool
- --output-dir <dir> to redirect artifact output
- --width, --height, --scale to control the render size
- --delay-ms, --after-delay-ms to control baseline/post-apply capture timing in live-gpu mode
- --ai-cli <template> to override the local AI command template
Backend behavior:
- The default --capture-backend skia path renders through an offscreen Skia surface,
so widget SkSL is present in the before/after images and the report records
render_backend: "skia-headless" with widget_sksl_render_supported: true.
- --capture-backend live-gpu drives the real pulp-design-tool app in automation
mode, captures before/after images from the actual Skia/Graphite presentation path,
and records render_backend: "skia-live-gpu" with sksl_gpu_supported: true.
- --capture-backend coregraphics is still available for comparison, but it does not
faithfully render custom widget SkSL.
Remaining limitation:
- skia and coregraphics still validate headless render paths, not the live app.
Use live-gpu when you need proof from the actual design-tool renderer.
inspect¶
Status: experimental
Connect to a running Pulp inspector server. With no --port, the CLI
auto-discovers the newest pulp-inspector-*.port file in the system temp
directory.
pulp inspect
pulp inspect --port 49152
pulp inspect --command DOM.getDocument
pulp inspect --command Capture.screenshot --output shot.json
Options:
--host HOST- connect to a host other than127.0.0.1--port PORT- connect to an explicit inspector port--command METHOD- send one inspector command and print the response--params JSON- JSON params for--command--output FILE- write a one-shot command response to a file
tweaks¶
Status: experimental
Inspect the pulp-tweaks.json sidecar the inspector overlay writes for
direct-manipulation edits. Tweaks are keyed by stable_anchor_id; after a
design re-import an anchor may no longer exist in the live tree, so the
tweak silently stops applying. pulp tweaks diff surfaces this — it is the
CLI mirror of the inspector's drift drawer, sharing the same drift-detection
logic underneath.
pulp tweaks diff
pulp tweaks diff --tweaks pulp-tweaks.json --design design.json
pulp tweaks diff --design design.json --json
tweaks diff classifies every stored tweak against a design snapshot:
- clean - the anchor (and property) still resolve
- drifted - the anchor survives but the targeted property is gone
- orphaned - the anchor itself is gone (re-import removed the element)
The design snapshot comes from --design FILE, a small JSON "anchors
manifest" in any of three shapes:
The first two forms enable anchor-only matching (orphan detection); the
third (anchor → property-path map) additionally enables property-level
drift detection. With --design omitted, the snapshot is empty and every
tweak is reported as orphaned.
Options:
--tweaks FILE- path topulp-tweaks.json(default: auto-resolved project sidecar)--design FILE- anchors-manifest JSON to diff against (default: empty)--json- emit the drift report as JSON instead of human-readable text
Exit code: 0 when no drift, 1 when drift is found, 2 on a usage or
file error.
import-design¶
Status: experimental
Import designs from Figma, Stitch, v0, Pencil, Claude Design, React JSX, or Google DESIGN.md source files into generated Pulp UI code.
pulp import-design --from figma --file frame.json
pulp import-design --from figma --url 'https://figma.com/design/...' --frame 'Plugin UI'
pulp import-design --from stitch --file screen.html --screen 'Main'
pulp import-design --from v0 --url 'https://v0.dev/t/abc123' --output ui.js
pulp import-design --from pencil --file ui.json --output ui.js --tokens tokens.json
pulp import-design --from v0 --file card.tsx --dry-run
pulp import-design --from claude --file design.html --classnames classnames.json
pulp import-design --from designmd --file DESIGN.md --tokens out.json
pulp import-design --from jsx --file bundle.js --mode live --emit js --output live-ui.js
pulp import-design --from jsx --file bundle.js --mode baked --emit cpp --output imported_ui.cpp
Accepted --from values: figma, stitch, v0, pencil, claude, designmd, jsx.
Supports --url (fetched through an argv-safe curl invocation into a unique temporary file), --frame (Figma frame selection), and --screen (Stitch screen selection). See Design Import API Reference for the full flag list.
For --from claude, the CLI emits a classnames.json artifact alongside the generated JS view and tokens.json. The artifact maps classname → { cssProp(camelCase): cssValue, ... } for every <style> rule with a plain classname selector — @pulp/css-adapt (and downstream) consumes it to merge class-based styles into inline before forwarding to bridge calls. Mirrors the output shape of Spectr's tools/extract-html-bundle/extract.mjs.
For --from designmd, the CLI emits only a tokens.json (W3C
DTCG) — no ui.js, because DESIGN.md describes a design system, not
a screen. See Import: DESIGN.md for the full
contract (supported subset, reference resolution, detection rules,
exit codes, diagnostics, and the staged rollout split).
| Flag | Description |
|---|---|
--output <path> |
Destination for the primary generated artifact. The built-in default primary artifact is live JS at ui.js; sidecars remain anchored beside this path when not explicitly overridden. |
--emit {js\|ir-json\|cpp\|swiftui} |
Select the primary artifact kind. js, ir-json, cpp, and swiftui are implemented; cpp and swiftui require --mode baked. swiftui emits a baked native SwiftUI view (ImportedPulpView.swift + a per-view <RootView>Theme.swift + binding manifest). Built-in default: js; persistent default: import_design.default_emit. |
--mode {live\|baked} |
Select the import runtime model. Built-in default: live; persistent default: import_design.default_mode. baked emits canonical IR or baked C++ via --emit ir-json\|cpp. |
--snapshot-semantics {fail\|warn\|accept} |
JSX baked snapshot policy. fail rejects dynamic APIs by default, warn proceeds with diagnostics, and accept proceeds silently. |
--allow-network-fetch |
Allow DesignIR asset-manifest HTTP(S) fetches at import time. |
--asset-cache <path> |
Asset cache directory for HTTP(S) imports. Defaults to PULP_IMPORT_ASSET_CACHE or the user cache. |
--asset-timeout-ms <ms> |
Per-request network asset timeout. |
--asset-hash <uri=sha256> |
Expected content hash for an asset URI; may be repeated. |
--classnames <path> |
Where to write the classname artifact (default: classnames.json). Only emitted for --from claude. |
--emit classnames |
Force-emit classnames.json (default on for --from claude). |
--no-emit-classnames |
Skip the classname artifact for the run. |
--tokens <path> |
Output token file (default: tokens.json; theme.css for --format css-variables). |
--screenshot-backend {skia\|coregraphics} |
--validate render backend. skia (default) composites file-backed images; coregraphics draws an image's filename placeholder, so it is not faithful for asset-rich designs. |
--format {w3c\|css-variables\|tailwind\|json-tailwind\|css-tailwind} |
Token export format. w3c (DTCG JSON) is the default; css-variables emits CSS custom properties (.dark modes → @media (prefers-color-scheme: dark)); the tailwind variants require --from designmd. Unknown values exit 2. |
With --emit ir-json, relative asset references from a --url import resolve
against the source URL. The manifest keeps the authored relative URI and also
records the resolved source_url used for HTTP(S) fetching.
The shipped default remains live runtime import: --mode live --emit js.
Persist different defaults with:
Set import_design.default_emit cpp for baked C++ by default. If only
import_design.default_mode baked is set, ir-json is implied. For temporary
session overrides, use PULP_IMPORT_DESIGN_DEFAULT_MODE and
PULP_IMPORT_DESIGN_DEFAULT_EMIT; direct CLI flags override the matching
config and environment value. pulp status shows the effective defaults.
With --from jsx --mode live --emit js, the CLI writes the precompiled JSX
runtime bundle verbatim for runtime import. That pass-through path rejects
--validate, --reference, --diff, and --debug because it does not parse or
render the bundle. With --from jsx --mode baked --emit ir-json|cpp, the CLI
captures a runtime snapshot into DesignIR and records snapshot provenance. DOM
bundles are captured through the DOM walker; live/native bundles that render
through @pulp/react are frozen from the native WidgetBridge tree and record
runtime_native_snapshot plus snapshotSource: native-view.
Dynamic APIs such as setInterval, setTimeout, requestAnimationFrame,
Date.now, new Date, performance.now, Math.random, and fetch fail by
default under --snapshot-semantics fail; comments and string literals are
ignored. Use warn to continue with a structured diagnostic, or accept to
continue without that diagnostic.
export-tokens¶
Status: experimental
Export a theme as W3C Design Tokens JSON.
audio¶
Status: experimental
Repo-level audio analysis tooling. Manages offline audio models (text-to-audio / text-to-excerpt retrieval) and reads reproducible excerpt bundles. This is developer tooling for building datasets and evaluation corpora — not a runtime API.
pulp audio # Show help
pulp audio model list [--json] # List registered models
pulp audio model status [--json] # Show configured + resolved model
pulp audio model activate <model-id> [--json] # Pick the active model
pulp audio excerpt-find --text "warm analog pad" --input /path/to/corpus [options]
pulp audio read-bundle <path-to-bundle> [--json]
pulp audio scope [target] --window 2048 --trigger rising-zero --channel 0 [--json scope.json]
pulp audio scope --input-wav tone.wav --window 2048 [--json scope.json] [--png scope.png]
pulp audio validate summarize <file.wav> [--json]
pulp audio validate doctor <file.wav> [--thd] [--response f1,f2,...] [--fundamental <hz>]
pulp audio validate compare <a.wav> <b.wav> [--mode null|spectral] [--tolerance <dbfs>]
pulp audio validate assert <audio-run-dir-or-assertions.json>
Subcommands:
| Subcommand | What it does |
|---|---|
model list |
List all registered audio models with backend and tags |
model status |
Show the configured model, resolved checkpoint, and whether it is loadable |
model activate <id> |
Select the active model and persist the state file |
excerpt-find |
Score audio files (or a directory) against a text query and emit an excerpt bundle |
read-bundle |
Pretty-print a previously emitted excerpt bundle |
scope |
Capture pulp.audio.scope.v1 JSON from a live standalone target or a speakerless offline WAV; offline mode can also write a PNG trace artifact |
validate summarize |
Decode a WAV and print an agent-readable signal summary (peak/RMS/DC/dominant pitch); --json for machine output |
validate doctor |
Offline Audio Doctor over a WAV: THD/THD+N (--thd) and/or spectrum magnitude at checkpoints (--response); writes a JSON curve artifact |
validate compare |
Sample-residual (null) verdict between two WAVs; exits nonzero past tolerance. --mode spectral currently applies a looser default tolerance to the same residual (a true spectral-distance metric is a later slice) |
validate assert |
Re-check a stored assertions.json (or an audio-run/ dir holding one); exits nonzero on any failing assertion |
Useful excerpt-find flags: --text, --input, --model, --recursive, --top, --window-ms, --hop-ms, --min-score, --max-candidates-per-file, --bundle-out, --dry-run. The model/excerpt-find/read-bundle subcommands accept --json for machine-readable output.
The validate subcommands are the offline harness CLI over captured audio (Phase 7). They analyze WAVs and stored artifact bundles with the reusable pulp::audio-analysis library — they do not instantiate a plugin (the generic CLI is not tied to a Processor; controlled-stimulus render is the test-side RenderScenario). The assertions.json schema is a {"schema_version", "assertions": [...]} document where each entry names a check (not_silent, silent, no_nan_inf, peak_below, frequency_near), a file (relative to the JSON), and the check's named tolerance.
pulp audio scope is the lower-level sample-window view. Live mode wraps
pulp run --audio-scope-json and may open the audio device; use
--input-wav <path> for speakerless offline analysis. Both paths emit the same
versioned JSON schema so CLI, MCP, and plugin agents can compare live and
offline captures without duplicating trigger or measurement logic. --png is
offline-only and writes a deterministic trace image of the acquired real
samples.
sdk¶
Status: usable
Manage the pinned Pulp SDK installation at ~/.pulp/sdk*/ used by standalone projects. pulp create and pulp build already pull the SDK in transparently when needed; pulp sdk is the explicit control surface.
pulp sdk # Show help
pulp sdk install # Download and cache the pinned SDK from GitHub releases
pulp sdk install --version 0.2.0 # Install a specific version
pulp sdk install --local # Build and install the SDK from the current Pulp checkout
pulp sdk status # Show cached and locally-built SDK versions
pulp sdk clean # Remove all cached SDK versions
Set PULP_HOME to relocate the SDK cache, asset cache, and config root.
dev¶
Status: usable
Unified development loop. Combines build --watch with optional test, validate, and launch-an-app steps in a single command so you can keep one terminal open while iterating.
pulp dev # Watch and rebuild
pulp dev --test # Watch, rebuild, run tests
pulp dev --test --test-filter=Knob # Watch, rebuild, run tests matching Knob
pulp dev --test --validate # Watch, rebuild, test, and validate built plugins
pulp dev --run pulp-gain-standalone # Watch, rebuild, relaunch app on each rebuild
pulp dev --design ui.js # Watch, rebuild pulp-design-tool, relaunch with ui.js
pulp dev --target pulp-format # Pass --target to cmake --build
pulp dev --run my-app -- --arg1 --arg2 # Arguments after `--` go to the launched binary
pulp dev --allow-unsupported-sdk # Bypass the CLI-vs-project SDK guard (unsupported)
Flags:
| Flag | Description |
|---|---|
--test, -t |
Run tests after each successful build |
--test-filter=PATTERN |
Run only tests matching PATTERN (implies --test) |
--validate |
Run quick plugin dlopen validation after build |
--run TARGET |
Launch TARGET from the build dir; relaunch on rebuild |
--design SCRIPT |
Build pulp-design-tool and launch it with SCRIPT |
--target T |
Pass --target T to cmake --build |
--allow-unsupported-sdk |
Bypass the CLI-vs-project SDK compatibility guard and continue anyway (unsupported) |
-- args... |
Arguments passed to the launched app |
pulp dev runs the same active-project compatibility preflight as pulp build. If the project pins an SDK or cli_min_version newer than the installed CLI, the command stops before SDK resolution/build and points at pulp upgrade.
loop¶
Status: experimental
Leveraged-prototype focus mode. pulp loop is the explicit "I'm in single-platform iteration mode" switch. It records the focus platform in ~/.pulp/config.toml under [loop] so the user can leave the mode and return to cross-platform iteration deliberately, then drives the same watch + rebuild + screencap loop as pulp dev using the current project's normal build configuration. Surrounding tooling can read the focus marker when it needs platform-specific behavior.
pulp loop # Enter focus mode on the auto-detected host
pulp loop --platform=macos # Mark macOS focus explicitly
pulp loop --platform=linux --test # Mark Linux focus + run tests on every save
pulp loop --status # Print the current focus state
pulp loop --off # Restore cross-platform mode
Flags:
| Flag | Description |
|---|---|
--platform=<macos\|linux\|windows> |
Override the auto-detected focus marker |
--off |
Restore cross-platform mode by clearing the focus marker |
--status |
Print the current focus state and exit |
--no-watch |
Persist focus state and exit without entering the watch loop |
--watch-issues N1,N2,... |
Recognized compatibility flag; prints a diagnostic and continues the normal loop unless --no-watch is also passed |
--ar-swap-from <ref> |
Recognized compatibility flag; prints a diagnostic and continues the normal loop unless --no-watch is also passed |
--test, -t |
Run tests after each successful build |
--test-filter=PATTERN |
Run only tests matching PATTERN (implies --test) |
--validate |
Run quick plugin dlopen validation after build |
--run TARGET |
Launch TARGET from build dir, relaunch on rebuild |
--target T |
Pass --target T to cmake --build |
--allow-unsupported-sdk |
Bypass the CLI-vs-project SDK compatibility guard |
-- args... |
Arguments passed to the launched app |
The CLI persists [loop] focus_platform = "..." in ~/.pulp/config.toml. Run pulp loop --off (or pair with shipyard pr / pulp pr) before landing the consumer-side PR — the ship path validates cross-platform regardless, but exiting focus mode explicitly keeps subsequent local iteration honest.
See docs/guides/focus-mode.md for the full playbook (when to use, when not to, and how to file framework issues from a focus-mode session).
scan¶
Status: usable
Walk the OS plug-in paths and print every VST3 / AU / AUv3 / CLAP / LV2 plug-in that was found. Mirrors what pulp::host::PluginScanner does at runtime. Useful for sanity-checking your local plug-in installation or for narrowing down which plug-in to feed to pulp host.
pulp scan # Scan every supported format the build includes
pulp scan --format clap # Scan only CLAP
pulp scan --format vst3 # Only VST3
pulp scan --format au # Only AU v2
pulp scan --format auv3 # Only AUv3
pulp scan --format lv2 # Only LV2
pulp scan -f clap # Short alias for --format
pulp scan --no-load # Filesystem-only walk escape hatch
pulp scan --help # Print usage; never opens any plug-in
Output is one line per plug-in: [<format>] header per section, then <name> <bundle-path>.
--no-load skips the dlopen step entirely. Names are filename-derived; vendor / version / unique-id metadata is not surfaced. The trade-off: --no-load cannot crash on a malformed plug-in whose static-init code throws across the dlopen boundary. Use it when the rich path errors out with libc++abi: terminating or when you want a quick path-only listing.
pulp scan --help is short-circuited — it does NOT dlopen any plug-in, so it remains safe even when one of the installed plug-ins would crash the rich path.
host¶
Status: experimental
Load a plug-in out-of-DAW and run a short synthetic audio block through it. Smoke-tests the hosting pipeline without launching Logic, Reaper, or Ableton.
pulp host /path/to/MyPlugin.clap # Load a CLAP bundle
pulp host /path/to/MyPlugin.clap --format clap # Explicit format
pulp host /path/to/MyPlugin.vst3 --format vst3
pulp host /path/to/MyPlugin.component --format au
pulp host /path/to/MyPlugin.lv2 --format lv2 --id https://example.com/plugins/my-plugin
pulp host -h # Show help
Flags:
| Flag | Description |
|---|---|
--format <fmt>, -f <fmt> |
Format: clap (default), vst3, au, auv3, lv2 |
--id <unique-id> |
Select a specific plug-in descriptor by URI / unique-id (used for LV2 and multi-plugin CLAP bundles) |
Prints plug-in metadata (name, vendor, version, format, parameter count) and the peak output level from a 256-sample synthetic block at 48 kHz. Exit code 0 on success, 1 if the bundle could not be loaded, 2 if prepare() failed.
import¶
Status: experimental
Read an existing audio-plugin project read-only and emit a Pulp migration scaffold. Framework importers are vendor-specific add-on tools that live in their own private repos; the Pulp SDK owns only the generalized substrate — a discovery index of known frameworks, a JSON-over-stdio service-provider interface (SPI) to drive an installed importer, and the emission step (the SDK writes files; the importer only proposes a plan).
The command is vendor-agnostic: framework identity is runtime DATA loaded from tools/import/known-frameworks.json, the one place real source-framework markers appear. The SDK code names no framework and no vendor.
pulp import detect ./MyProject # Rank candidates; print install hint
pulp import ./MyProject # Alias for detect
pulp import inspect --from <framework> ./MyProject -o ir.json # Resolve importer → SPI analyze → ProjectIR
pulp import inspect --from <framework> ./MyProject --importer-cmd "python3 spi.py"
pulp import emit --from <framework> ./MyProject --output ./scaffold
Subcommands:
| Subcommand | Description |
|---|---|
detect <dir> |
Scan the directory against the known-frameworks markers and print ranked candidates (framework id + confidence + evidence) plus the install hint for the top match. Works with no importer installed. |
inspect --from <fw> <dir> |
Resolve the importer (tool registry or --importer-cmd) and run its SPI analyze verb to produce a ProjectIR. When no importer is resolvable, prints the install hint and exits non-zero. |
emit --from <fw> <dir> --output <out> |
Resolves the importer, runs its SPI analyze then emit verbs to get an EmissionManifest, then the SDK writes a buildable Pulp migration scaffold under <out>. The importer only proposes files (generated/stub carry inline content; verbatim portable-core copies carry an absolute copy_from); the SDK materialises them, runs a clean-room output scan over every generated file, and writes migration_status.json + a .pulp-import-provenance.json marker. Skewed/symmetric source parameter curves emit as shaped ParamRanges (skew + symmetric fields), no longer downgraded to linear. |
inspect / emit flags:
| Flag | Description |
|---|---|
--from <framework> |
Framework id (see pulp import detect) |
--framework-path <path> |
The user's own framework checkout (read-only; never vendored) |
--extra-include <dir> |
Extra include directory passed to the importer (repeatable) |
-o, --output-ir <file> |
inspect: write the ProjectIR JSON to a file |
--report <file.md> |
inspect: write a human-readable report |
--output <dir> |
emit: scaffold output directory |
--importer-cmd <cmd> |
Override importer resolution with an explicit command string |
--accept-importer-terms |
Accept the importer's terms of use non-interactively (CI). The acceptance is still recorded under ~/.pulp. |
Accept-to-run terms gate. Before inspect / emit drives an importer, the user must give explicit affirmative acceptance of that importer's terms of use (mirrors pulp add --accept-license). The terms body is runtime DATA carried by the add-on importer (the SDK names no vendor and ships no terms body of its own); the SDK surfaces it, then records acceptance under ~/.pulp/importer-terms-accepted.json keyed by importer id + a hash of the terms text. A changed terms version (new hash) re-prompts. Interactively the gate is a type-to-accept prompt; in CI pass --accept-importer-terms. Without a terminal and without the flag, the gate blocks with a non-zero exit rather than hanging. An importer that declares no terms passes the gate transparently.
The importer is resolved against tools/packages/tool-registry.json: an importer tool declares the frameworks it handles plus spi_min / spi_max (the SPI version window) and sdk_min / sdk_max. The SDK negotiates the SPI version on every call and fails loudly on a mismatch ("upgrade Pulp" / "upgrade the importer") rather than misbehaving silently. The data contracts are tools/import/schemas/project-import-ir-v0.schema.json and tools/import/schemas/import-spi-v0.schema.json.
Who writes what (clean-room boundary). The importer is a separate add-on and never writes into the user's tree — it returns an EmissionManifest over the SPI emit verb. The SDK writes every file, and before writing each generated file it runs a clean-room output denylist scan (sourced from the known-frameworks content markers) that rejects framework source or vendor banners; a copied-user-file is the user's own DSP, copied verbatim and recorded in provenance, so it is exempt. A misbehaving importer therefore cannot smuggle framework code into the scaffold. The SDK also writes migration_status.json (the migration verdict + TODO list) and .pulp-import-provenance.json (importer id, framework, SPI version, emit timestamp, source-tree hash, per-file provenance).
identity¶
Status: experimental
Manage .pulp/identity.lock — the committed pin of each plugin's AU 4CC, manufacturer code, AAX product code, optional VST3 FUID, and optional CLAP plugin id. The lock is the audit trail that "this plugin's host-visible identity has not silently changed". See docs/reference/identity-lock.md for the schema and Track 3.12 of the macOS plugin-authoring plan for the rationale.
pulp identity record # Write/refresh .pulp/identity.lock
pulp identity record --allow-identity-change # Accept drift and overwrite the lock
pulp identity record --dry-run # Print what would be written
pulp identity check # Compare lock vs project, exit 1 on drift
pulp identity check --allow-identity-change # Treat drift as success
The same check is wired into pulp build --check-identity so CI can fail any PR that changes a host-visible identity field without an explicit pulp identity record --allow-identity-change step.
tool¶
Status: usable
Manage the third-party developer tools Pulp can optionally use (formatters, validators, importers). Tools are described in tools/packages/tool-registry.json and installed under ~/.pulp/tools/ — they are kept out of the system PATH so Pulp-managed installs can never clobber the system copy.
pulp tool # Show help
pulp tool list # Show every registered tool and its install state
pulp tool install clap-validator # Download and install one tool
pulp tool install --all # Install every tool available on the current platform
pulp tool install <id> --force # Reinstall even if already present
pulp tool install <importer> # Install a framework importer add-on (checksummed, version-window-checked)
pulp tool install <importer> --from <path|file://...> # Install from a local package (offline / pinned artifact)
pulp tool uninstall <id> # Remove a pulp-managed tool, or an importer (also removes its skill)
pulp tool path <id> # Print the absolute path to the installed tool's binary
pulp tool run <id> [args...] # Run the installed tool with pass-through arguments
pulp tool doctor # Health check: which tools are installed, which are missing, which are unavailable on this platform
pulp add <importer> # Alias for `pulp tool install <importer>`
Install methods come from the registry — today binary_download (pinned release artifact), python_pip (pipx-style isolated install), and importer_package (a checksummed, per-platform framework-importer archive). pulp tool doctor is the per-platform companion to pulp doctor.
Framework importers. An importer is a vendor-specific add-on (described in the tool-registry with category: "importer") that drives Pulp's JSON-over-stdio import SPI. Installing one is gated three ways: the importer's [sdk_min, sdk_max] must include the running SDK and its [spi_min, spi_max] window must overlap the SDK's supported import-SPI window (a mismatch fails loudly with an "upgrade Pulp" / "upgrade the importer" message); the fetched or local package's sha256 must match the digest pinned in the registry (a mismatch refuses to install); and the importer's bundled SKILL.md is installed into ~/.agents/skills/<importer>/ on install and removed on uninstall. Each install is recorded under ~/.pulp/importers/<id>.json (id, version, sha256, SDK version, SPI window, paths, terms metadata) so uninstall and version checks work, and so the importer-terms accept-gate composes with the same record. pulp add <importer> routes to the same install path. Use --from <path|file://...> to install from a local package rather than the registry URL (offline installs, pinned artifacts, CI). The producer side — how prebuilt per-platform artifacts are built, hosted, pinned per SDK release, and signed/notarized, and the bundled-libclang choice — is documented in framework-importer-packaging.md; this CLI consumes that contract, it does not decide it.
upgrade¶
Status: usable
Update the Pulp CLI binary to the latest (or a specific) version.
pulp upgrade # upgrade to latest release
pulp upgrade 0.2.0 # install specific version
pulp upgrade --check-only # report cached latest release; no download
pulp upgrade --notes # print migration notes for installed -> cached latest
pulp upgrade --notes --json # same, stable-shape JSON (agent-consumable)
pulp upgrade --notes --from 0.25.0 --to 0.29.0 # explicit hop override
Downloads the release from GitHub, installs the archive's top-level companion payloads next to the current binary, replaces the current binary, and verifies. Current archives install Rust pulp plus the pulp-cpp fallthrough delegate together. Requires curl.
--check-only reads the on-disk cache written by the on-every-invocation background refresh and prints installed/latest/notes. If the cache is empty (first run), it falls through to a single live GitHub query.
--notes filters the embedded migration index (built from docs/migrations/*.md at compile time) through each entry's applies_if expression and prints only the notes relevant to the upgrade hop. No network, no binary swap. The JSON variant emits a stable-shape document (from, to, entries[].{version, breaking, summary, applies_if, body}) for the /upgrade Claude Code skill.
config¶
Status: usable
Read or write ~/.pulp/config.toml settings.
pulp config get pr.workflow
pulp config set pr.workflow github
pulp config get update.mode
pulp config set update.mode manual
pulp config set update.check_interval_hours 12
pulp config set import_design.default_mode baked
pulp config set import_design.default_emit ir-json
pulp config set claude.send_user_file off
pulp config list
Supported PR workflow key:
pr.workflow— one ofshipyard | github | manual(defaultshipyard).shipyarddelegates to the pinned Shipyard contributor tool,githubuses the GitHub CLI (gh) directly, andmanualprints instructions without mutating PR state.PULP_PR_WORKFLOWoverrides this value for one command.
Supported update keys:
update.mode— one ofauto | prompt | manual | off(defaultprompt). All four modes are wired into the invocation path:auto— silently stages the new release via~/.pulp/pending-upgrade; the swap completes on the next invocation.prompt— prints a one-line banner per new version; 24h snooze via~/.pulp/update-snoozerespected if present.manual— prints a one-line "Runpulp upgradewhen you're ready" notice per new version; never prompts.off— zero network calls, zero notices. Suitable for CI and air-gapped environments.
Changing update.mode clears ~/.pulp/update-snooze so the new mode takes effect on the next invocation.
- update.check_interval_hours — integer hours between background checks (default 24). The 24h default stays under the 60/hour anonymous GitHub API rate limit by a wide margin.
- update.channel — stable | beta (default stable). Reserved for future release-channel support; ignored today.
- update.bump_projects — prompt | auto | off (default prompt). Controls whether a successful pulp upgrade nudges the user toward pulp project bump --all.
Supported import-design keys:
import_design.default_mode—live | baked(defaultlive). Controls the default import runtime model whenpulp import-designis run without--mode.PULP_IMPORT_DESIGN_DEFAULT_MODEoverrides this value for one environment/session.import_design.default_emit—js | ir-json | cpp(defaultjs). Controls the default primary artifact whenpulp import-designis run without--emit.PULP_IMPORT_DESIGN_DEFAULT_EMIToverrides this value for one environment/session. If the default mode isbakedand this key is unset, Pulp impliesir-json.
Supported Claude Code plugin keys:
claude.send_user_file—on | off(defaulton). Whenon, the Pulp Claude Code plugin'sSessionStarthook injects a preference telling the agent to surface generated image/file artifacts (screenshots, rendered designs, diagrams, build outputs) with theSendUserFiletool so they embed in the Claude app, instead of only printing a path. Setoffto suppress the injection. Read at session start byhooks/scripts/inject-claude-prefs.sh.
clean¶
Status: usable
Remove the build directory.
fmt¶
Status: usable
Run clang-format against Pulp source files using the project root's
.clang-format. Walks core/, examples/, inspect/, test/,
tools/, and ship/ by default, or the path arguments you pass.
pulp fmt # rewrite all .cpp/.hpp/.h/.mm in place
pulp fmt path/to/file.cpp ... # restrict to specific paths
pulp fmt --dry-run # report diffs without rewriting
pulp fmt --check # CI-friendly alias for --dry-run
pulp fmt requires clang-format on PATH. Skips build/, _deps/,
generated/, and external/ so vendored / build-output code is
untouched.
help¶
Print usage information.
version¶
Status: usable
Show, bump, or check version consistency across all surfaces (CMakeLists.txt, SDK constant, CHANGELOG, AU Info.plist).
pulp version # Show current SDK and project versions
pulp version bump patch # Increment patch version
pulp version bump minor # Increment minor version
pulp version bump major --plugin # Bump plugin version (pulp_add_plugin VERSION)
pulp version check # Verify version consistency
The bump subcommand updates CMakeLists.txt project(VERSION) and adds a CHANGELOG.md entry. The SDK version constant is derived from CMake via configure_file, so a rebuild picks up the change automatically. Use --plugin to bump the pulp_add_plugin(... VERSION ...) line instead.
The check subcommand verifies:
- SDK version constant matches CMakeLists.txt
- AU Info.plist template uses a computed version integer (not hardcoded)
- CHANGELOG latest heading matches CMakeLists.txt
kit¶
Status: experimental
Share reusable Pulp code, UI, and templates with a review step before they touch a project.
Use pulp kit for Pulp-native building blocks: DSP source, UI widgets, design tokens, templates, validation fixtures, and graph/native components. The value is practical:
- developers package real Pulp pieces once instead of copying example folders between projects;
- users and reviewers see the files, licenses, capabilities, and project changes before approval;
- agents can inspect structured metadata without running untrusted package code.
This is intentionally separate from pulp add. pulp add rubberband means "add a curated dependency from Pulp registry metadata." pulp kit validate ./thing or pulp kit validate ./thing.pulpkit means "inspect this local artifact before trusting or applying it."
The workflow is inspect, plan, verify, approve, apply:
validate/inspectreadpulp.package.jsonand declared files only.planpreviews the project changes without writing files.verifyruns declared validation-profile checks after the plan has been reviewed; optional screenshot execution requires an explicit flag.apply --yeswrites only reviewed, owned project files and records the reviewed manifest digest.remove --yesdeletes only constrained lock-recorded kit paths underpulp-kits/<kit-id>/...plus the known generated lock/CMake files.
Trust rules:
- metadata commands never run package CMake, JavaScript, shell scripts, dynamic libraries, remote search, or content installers;
.pulpkitand.pulpcontentarchives must includefiles.sha256.json, and every payload file must be listed and hash-matched before the manifest is trusted;- validation checks manifest shape, licenses,
requires.pulp,requires.cpp, known Pulp module dependencies, and declared evidence hashes before plan/apply; content-packmanifests can be searched, validated, and inspected, butpulp kit plan/apply/publishrejects them; usepulp content ...for data-only packs;- dependency packages declared by a kit resolve only through the existing curated
pulp add <id>machinery.
pulp kit search basic --root ./fixtures/packages --lane kit --json
pulp kit search content --root ./fixtures/packages --lane content --json
pulp kit validate ./fixtures/packages/gain-dsp-kit
pulp kit validate ./fixtures/packages/basic-ui-kit --json
pulp kit inspect ./fixtures/packages/simple-plugin-template --json
pulp kit plan ./fixtures/packages/gain-dsp-kit --project .
pulp kit verify ./fixtures/packages/basic-ui-kit --project . --json
pulp kit verify ./fixtures/packages/basic-ui-kit --project . --execute-screenshots --json
pulp kit apply ./fixtures/packages/basic-ui-kit --project . --yes
pulp kit remove dev.pulp.fixtures.basic-ui-kit --project . --yes
pulp kit pack ./fixtures/packages/basic-ui-kit --output ./dist/basic-ui-kit.pulpkit
pulp kit publish ./fixtures/packages/basic-ui-kit --dry-run --json
pulp kit publish ./fixtures/packages/basic-ui-kit --dry-run --registry-manifest ./registry/pulp-registry-manifest.json --json
pulp kit init --kind source --id com.example.my-kit --dir ./my-kit
pulp create "Kit Gain" --template ./fixtures/packages/simple-plugin-template --no-build --ci
pulp create "Kit Gain" --template ./fixtures/packages/simple-plugin-template --ci
Package-backed templates can include generated-project tests, but those tests are optional in standalone SDK builds unless the generated project can find Catch2. The required review artifact is the declared generated-project diff; the required build proof is the exported plugin/app format targets.
Subcommands:
| Subcommand | What it does |
|---|---|
search [query] |
Search local pulp.package.json, .pulpkit, and .pulpcontent artifacts without executing package code; archives must pass files.sha256.json checks before their manifest is indexed, and results are classified as kit or content lanes |
validate <path> |
Validate a kit directory, content-pack directory, pulp.package.json, .pulpkit, or .pulpcontent archive without executing package code |
inspect <path> |
Print the manifest summary, capabilities, dependency package ids, and validation issues |
plan <path> |
Produce a reviewable project-mutation plan from a kit directory, manifest, or .pulpkit archive without writing files; rejects content-pack manifests, and dependency packages are resolved only by curated pulp add <id> ids |
verify <path> |
Run declared validation-profile checks after plan review; default mode is metadata-only, and --execute-screenshots explicitly renders Pulp screenshot profiles through the project screenshot tool and compares expectedImage baselines when declared, honoring optional visualToleranceBytes |
apply <path> --yes |
Apply a reviewed local kit plan from a kit directory, manifest, or .pulpkit archive: rejects content-pack manifests, writes .pulp/kits.lock.json with manifest_sha256, generated cmake/pulp-kits.cmake, declared copied files, and UI-kit interface metadata |
remove <kit-id> --yes |
Remove an installed kit using only .pulp/kits.lock.json ownership records |
pack <path> |
Create a .pulpkit or .pulpcontent archive with files.sha256.json |
publish <path> --dry-run |
Run the metadata-only kit publish gate: rejects content-pack manifests, strict manifest validation, license inventory, NOTICE-compatible license files via exports.licenses, human review, validation profiles, kind-specific evidence, local quality badges, compatibility summary, and optional signed canonical registry-manifest verification. Remote publishing is still disabled. |
init |
Scaffold a developer-oriented fixture manifest for source, ui-kit, or template |
Developer notes:
searchis local discovery only. It never fetches from a registry and never makes a result safe to apply.- Template kits can seed a project with
pulp create "<name>" --template <kit-dir>. The template is validated, exported files are copied, and dependency packages are never installed implicitly. - UI kits copy scripts, tokens, and assets under
pulp-kits/<kit-id>/. After review, attach one reviewed UI script and optional tokens/assets withpulp_use_kit_ui(...); apply alone does not attach UI code. - Graph/native kits use the same inspect/plan/apply flow. Validation requires explicit realtime claims, and signed
node-packkits cannot claim iOS/AUv3 support. - Agent-authored kits need
authoring.humanReview.reviewed = truebefore publish dry-run can pass. pulp kit publish --dry-runis local readiness only. It checks NOTICE-compatible license exports, validation evidence, compatibility, local quality badges, and optional signedpulp-registry-manifest-v1; remote publishing is disabled.
content¶
Status: experimental
Validate and install data-only content packs for installed plugins.
Use pulp content for end-user data such as presets, themes, samples, and wavetables. Plugin authors get a standard expansion-pack format instead of custom installers. Users get validation, an explicit install target, and removal that leaves their own presets alone. Agents can reject mismatched packs before install because plugins declare the content kinds they actually support.
Keep the three lanes distinct:
pulp add <name>adds curated developer dependencies from the Pulp registry.pulp kit ...reviews artifacts that may transform a project.pulp content ...installs read-only data into a plugin-specific content directory.
The workflow is validate, preview, approve, install/update. Install, update, and remove require --yes. .pulpcontent archives must include files.sha256.json; every payload file must be listed and hash-matched before preview, install, or update.
preview reads the trusted pulp.plugin-runtime.json emitted by the plugin and reports compatibility, target plugin, accepted content kinds, and hot-reload/rescan/restart policy. update takes an explicit local path, not a registry name or URL, and rolls back a replaced version on failure. Content commands copy data only; they never run package CMake, JavaScript, scripts, dynamic libraries, or remote fetches. Removal deletes only the installed content-pack root, not user-created presets or edits.
Plugins opt in with ContentRegistry or PresetManager. Prefer declaring content support in CMake with pulp_add_plugin(... CONTENT_CAPABILITIES ... CONTENT_KINDS ...); that generates the pulp.plugin-runtime.json used by agents, previews, and ValidationHarness::validate_plugin_runtime_manifest(...).
pulp_add_plugin(MySynth
...
CONTENT_CAPABILITIES content.presets.v1 content.samples.v1
CONTENT_KINDS presets samples
CONTENT_HOT_RELOAD_KINDS presets)
{
"schema": "pulp.plugin-runtime.v1",
"pluginId": "dev.example.synth",
"content": {
"capabilities": ["content.presets.v1", "content.samples.v1"],
"kinds": ["presets", "samples"],
"reload": {
"hotReloadKinds": ["presets"],
"manualRescanKinds": []
}
}
}
pulp content validate ./fixtures/packages/basic-content-pack --json
pulp content preview ./fixtures/packages/basic-content-pack --plugin-runtime ./build/PulpSynth.pulp.plugin-runtime.json --plugin dev.example.synth --json
pulp kit pack ./fixtures/packages/basic-content-pack --output ./dist/basic-content-pack.pulpcontent
pulp content install ./dist/basic-content-pack.pulpcontent --plugin dev.example.synth --yes
pulp content update ./dist/basic-content-pack.pulpcontent --plugin dev.example.synth --yes
pulp content list --plugin dev.example.synth --json
pulp content rescan --json
pulp content reveal dev.pulp.fixtures.basic-content-pack --plugin dev.example.synth --version 0.1.0
pulp content remove dev.pulp.fixtures.basic-content-pack --plugin dev.example.synth --version 0.1.0 --yes
Subcommands:
| Subcommand | What it does |
|---|---|
validate <path> |
Validate a .pulpcontent archive or content-pack directory without executing package code |
preview <path> --plugin-runtime <manifest> |
Preview compatibility and reload/restart policy without installing anything |
install <path> --plugin <id> --yes |
Copy a validated content pack into the plugin-specific user content root and update the content index with plugin_id and manifest_sha256 |
update <path> --plugin <id> --yes |
Replace or add a validated local content pack, write the target plugin id and new manifest digest, and roll back a replaced version on failure |
list [--plugin <id>] |
List installed content packs |
rescan |
Rebuild Content/index.json from installed local manifests without copying, deleting, fetching, or executing package code; index entries include plugin_id and manifest_sha256 |
reveal <package-id> --plugin <id> |
Print the installed content path |
remove <package-id> --plugin <id> --yes |
Remove an installed content-pack root |
add¶
Status: usable
Add a curated third-party dependency from the Pulp package registry.
pulp add is intentionally narrow. Package names resolve through Pulp-controlled registry metadata, not arbitrary GitHub/GitLab URLs or manifest-bearing local paths. Use pulp kit validate/plan/apply for local or external Pulp-native artifacts that can transform a project.
pulp add signalsmith-stretch # add a package
pulp add rtneural --license-override commercial # accept a non-standard license
pulp add some-lib --platform-guard # add with platform guard
pulp add dr-libs --no-cmake # metadata only, skip CMake wiring
Performs license checking, platform compatibility analysis, overlap detection, CMake generation (cmake/pulp-packages.cmake), and updates packages.lock.json, DEPENDENCIES.md, and NOTICE.md.
remove¶
Status: usable
Remove a previously added package.
Cleans up the lock file, CMake declarations, and metadata entries.
list¶
Status: usable
Show installed packages.
search¶
Status: usable
Search the package registry.
update¶
Status: usable
Check for and apply package updates.
pulp update # dry-run: show available updates
pulp update --apply # apply updates and regenerate CMake
suggest¶
Status: usable
Context-aware package recommendations.
pulp suggest --description "pitch shifting"
pulp suggest --analyze src/my_processor.cpp
pulp suggest --alternative pffft
target¶
Status: usable
Manage project platform targets stored in pulp.toml.
pulp target list # show current targets
pulp target add Windows-arm64 # add a target
pulp target remove Linux-x64 # remove a target
Default targets (if none configured): macOS-arm64, Windows-x64, Linux-x64.
audit (package extensions)¶
The existing pulp audit command now supports package-specific flags:
pulp audit --packages # verify lock file integrity
pulp audit --platforms # check package/platform coverage
pulp audit --licenses # verify license compatibility
These flags are handled natively; without them, pulp audit delegates to the Python audit script as before.
Global Flags¶
| Flag | Description |
|---|---|
--no-color |
Disable color output (also respects NO_COLOR env var) |
Color output is auto-detected based on TTY. Non-TTY environments (pipes, CI) get plain text automatically.
Caveats¶
- Standalone projects are detected by walking up from the current directory looking for
pulp.tomlwithoutcore/. - If both a standalone project and a parent Pulp repo are present, the standalone project wins.
- Pulp repo mode is detected by walking up from the current directory looking for a directory with both
CMakeLists.txtandcore/. - The
shipsubcommands are macOS-specific (they usecodesignandpkgbuild). pulp upgraderequires internet access andcurl(macOS/Linux) or PowerShell (Windows).