Skip to content

entropyvortex/buckle

buckle

One verb for devcontainers — generate, build, up, and bash with user-wide templates.

License: MIT Node Coverage Status

buckle is a single TypeScript CLI that collapses the daily devcontainer ritual — hand-author .devcontainer/devcontainer.json → maybe a Dockerfile → maybe a compose file → devcontainer updocker exec -it … bash — into one verb. It also gives you user-wide templates so the same config travels with you, not with each repo.

$ buckle up claude-corp1     # build & up & bash in
$ buckle                     # land in any folder; TUI does the right thing
$ buckle node --feature dod  # render Node template + docker-outside-of-docker
$ buckle down                # tear it down

Why

Developers create, rebuild, and shell into devcontainers many times per day, in many folders. Doing this by hand means: copy-paste a devcontainer.json from the last project, tweak it, run devcontainer up, then a long docker exec -it … bash incantation. There is no good "my standard environment" story across folders.

buckle fixes that with three ideas:

  1. Templates are user-wide. ~/.config/buckle/templates/<name>/template.yaml — author once, use everywhere. Inheritance with extends:. Override with !replace.
  2. One verb does the obvious thing. Empty folder → wizard. Folder with .devcontainer/ → status panel. buckle up <tpl> builds and bashes. Nothing surprising.
  3. The same template emits all three artifacts (devcontainer.json, optional Dockerfile, optional docker-compose.yml) deterministically. Diffs are stable; round-trip is safe.

Install

Requires Node ≥ 20 and a working Docker daemon. The @devcontainers/cli is required for buckle up / rebuild because it correctly wires features and lifecycle hooks; buckle doctor will tell you if it's missing.

npm install -g buckle-cli @devcontainers/cli

Note: The package on npm is published as buckle-cli, but the executable command is buckle.

Verify:

buckle --version
buckle doctor

Quickstart

$ cd ~/scratch/myproj
$ buckle                  # opens TUI: detects project, suggests templates, writes .devcontainer
$ buckle up               # build, up, and drop into a shell

Or non-interactively:

$ buckle node --yes --trust    # render the built-in `node` template
$ buckle up                    # build & shell in

For the headline use case:

$ buckle up claude-corp        # Node + Claude Code + GitHub CLI + selected MCPs

Commands

Command What it does
buckle If no .devcontainer exists → wizard. Else → live status panel.
buckle <template> Render <template> into .devcontainer/. No build.
buckle up [<template>] Generate (if missing), build, up, exec a shell.
buckle down [--prune] Stop & remove the workspace's container. --prune cleans dangling images/volumes.
buckle bash Exec a shell into the running container.
buckle restart Restart in place (no rebuild).
buckle rebuild Down + force-rebuild + up + bash.
buckle logs [-f] [--tail N] Stream container logs.
buckle status Show container state and metadata.
buckle list [--installed-only] List built-in / user / installed templates.
buckle edit <template> Open user template in $EDITOR.
buckle new <name> [--extend <p>] Scaffold a new user template.
buckle install <origin> Install a template from gh:user/repo, gl:user/repo, https git URL, or file://.
buckle uninstall <name> Remove a previously installed template.
buckle doctor Diagnose the host environment.

Global flags

Flag Effect
--json Machine-readable output for every command.
--verbose Extra logs to stderr.
--yes Skip the diff confirmation when writing files.
--trust Skip the lifecycle-trust prompt.
--feature <spec> Add a buckle convenience feature; repeatable. (--feature dod --feature node:20)
--user <user> Run the shell as <user> when bashing in.
--rebuild Force rebuild on up.
--detach Don't attach a shell after up/rebuild.
--git-init Initialize a git repo in the workspace if there is none.
--preview, --dry-run Show what would be written without touching disk.

Templates

A buckle template is a YAML file at ~/.config/buckle/templates/<name>/template.yaml. Built-ins ship with the binary; user templates override built-ins with the same name; installed templates (via buckle install) live in their own subtree.

Minimal example

# ~/.config/buckle/templates/claude-corp1/template.yaml
name: Claude Corp 1
description: Node + Claude Code with my team's MCP set.
version: 0.1.0

extends: claude-corp           # inherit from a built-in

features:
  - claude-code
  - grok
  - mcp:filesystem
  - mcp:github

env:
  ANTHROPIC_LOG: warn

lifecycle:
  postCreate:
    - npm install -g pnpm
  postAttach:
    - "echo 'corp1 ready'"

Full schema (cheat sheet)

name: Display name
description: Long-form description.
version: 0.1.0

# Inheritance: single parent or ordered MRO array (rightmost wins on conflicts).
extends: ubuntu-base
# extends: [base, mixin-a, mixin-b]

# EXACTLY ONE of image / build / compose may be set per template level.
image: mcr.microsoft.com/devcontainers/base:ubuntu
build:
  dockerfile: Dockerfile
  context: .
  args: { NODE_VERSION: "20" }
  target: dev
compose:
  file: docker-compose.yml
  service: app
  runServices: [app, db]
  shutdownAction: stopCompose

# Buckle convenience features (sugar). Compile to native devcontainer features + hooks.
features:
  - dod                  # docker-outside-of-docker
  - dind                 # docker-in-docker (privileged)
  - gh
  - git-config
  - claude-code
  - grok                  # Grok Build (xAI agentic CLI/TUI)
  - mcp:filesystem
  - aws
  - gcloud
  - kube
  - terraform
  - node:20
  - python:3.12
  - go:1.22
  - rust:1.75
  - java:21
  # native devcontainer features pass through:
  - ghcr.io/devcontainers/features/git:1

# Pass-through native devcontainer features (advanced).
nativeFeatures:
  ghcr.io/devcontainers/features/python:1: { version: "3.12" }

forwardPorts:
  - 3000
  - { port: 8080, label: api, onAutoForward: notify }
appPort: 3000

mounts:
  - { source: "${localEnv:HOME}/.aws", target: /home/vscode/.aws, type: bind, readOnly: true }

env:
  NODE_ENV: development

runArgs:
  - "--cap-add=SYS_PTRACE"
  - "--init"

customizations:
  vscode:
    extensions: [dbaeumer.vscode-eslint, esbenp.prettier-vscode]
    settings: { "editor.formatOnSave": true }

remoteUser: vscode
containerUser: vscode
workspaceFolder: /workspaces

# Lifecycle hooks. Append-merge by default; first element "!replace" replaces parent.
lifecycle:
  initialize:    [ ./bin/preflight ]
  onCreate:      [ ]
  updateContent: [ ]
  postCreate:    [ corepack enable, npm ci ]
  postStart:     [ ]
  postAttach:    [ "node --version" ]

Inheritance & merging

  • extends: accepts a string or an ordered list (MRO; rightmost wins).

  • Cycles are detected; depth is capped at 8.

  • Plain objects merge key-by-key.

  • Arrays append by default. To replace the parent's array, prefix with the literal !replace:

    lifecycle:
      postCreate:
        - "!replace"
        - echo "this fully overrides the parent's postCreate"
  • image / build / compose are mutually exclusive at any single level. A child's choice replaces the parent's.

Common patterns & gotchas

Lifecycle hook ordering

Hooks run in this order (the devcontainer spec defines the timing):

Hook When it runs Typical use
initialize Very early, on the host (before container exists) Preflight scripts, secret fetching
onCreate Once, when the container is first created One-time heavy setup (tool installs that don't change often)
updateContent When content is refreshed (git pull, etc.) npm ci, go mod download
postCreate After the container is created and code is present Most common place for npm install -g, database migrations, corepack enable
postStart Every time the container starts Lightweight background services
postAttach Every time you buckle bash or attach in VS Code Welcome messages, node --version, git status

Tip: Use !replace at the start of a list when you want to completely override a parent's hooks instead of appending.

Variable expansion in mounts and env

Buckle (and the underlying devcontainer tooling) supports ${localEnv:VAR} syntax. This is not shell expansion — it is performed by the devcontainer CLI / VS Code when the container is created.

mounts:
  - source: "${localEnv:HOME}/.aws"
    target: /home/vscode/.aws
    type: bind
    readOnly: true

  - source: "${localEnv:HOME}/.claude"
    target: /home/vscode/.claude
    type: bind
  • Use ${localEnv:HOME} rather than hard-coding /Users/you or /home/you.
  • The variable must exist in your shell environment when you run buckle up / the wizard.
  • For secrets or tokens, prefer mounting a directory or using a short-lived initialize script rather than embedding them in the template.

Using both Claude Code and Grok Build

Buckle treats having both major agentic coding agents in the same devcontainer as a first-class experience.

# Quick start with both agents
buckle up --feature claude-code --feature grok

# Or use the dedicated template
buckle new myproject --extend ai-native

Why run both?

  • They have different strengths and personalities.
  • You can route different tasks to the agent that performs best on that workload.
  • Their context windows, tool use, and reasoning styles are complementary.

How buckle supports the combination:

  • Separate, persistent mounts: ~/.claude and ~/.grok
  • Both agents' official installers run cleanly in postCreate
  • No conflicts in configuration or PATH handling
  • The ai-native built-in template is specifically designed around this dual-agent setup

You can also mix and match freely in your own templates:

features:
  - claude-code
  - grok
  - mcp:filesystem
  - mcp:github

See the ai-native and claude-corp built-in templates for realistic examples of this pattern in action.

Important: The safety model and philosophy of the ai-native template

This template is deliberately built to give Claude Code and Grok Build maximum, unrestricted power inside the container:

alias claude='claude --dangerously-skip-permissions'
alias grok='grok --yolo'

This is not a bug or oversight. It is the entire point of this template.

Why this exists

  • The container is a throwaway, high-trust sandbox.
  • Your real source code lives on the host and is only bind-mounted in.
  • If something goes wrong, you can rm -rf .devcontainer && buckle up ai-native --rebuild and start fresh in seconds.
  • The agents run as the vscode user (with full write access to the mounted workspace).
  • The dangerous flags (--dangerously-skip-permissions / --yolo) are enabled by default because the whole purpose of this environment is to let the agents move fast without asking for confirmation on every file edit, terminal command, or package install.

If you do not want agents to have this level of autonomy, do not use the ai-native template (or extend it and remove the aliases).

This design is intentional and documented. The container is not meant to be a "safe" daily driver for your host machine — it is a disposable, high-agency environment for AI coding agents.

Working with compose (multi-service)

When you set compose: instead of image or build, buckle generates a minimal docker-compose.yml on first use.

Important realities:

  • Lifecycle hooks (postCreate, etc.) run only against the primary service you declared.
  • If you need to run commands in other services, use docker compose exec <service> ... inside your hooks.
  • Features like dind and dod only affect the primary service unless you duplicate the configuration in your compose file.
  • The generated compose file is a starting point. Edit it freely after the first render.

See the compose-demo built-in template (run buckle new myapp --extend compose-demo) and docs/PATTERNS.md for deeper multi-service guidance, including how hooks interact with secondary services and common customizations people make after first render.

Realistic corporate inheritance example

# ~/.config/buckle/templates/my-corp-base/template.yaml
name: My Corp Base
extends: ubuntu-base
features:
  - gh
  - git-config
  - aws
lifecycle:
  postCreate:
    - corepack enable
    - npm install -g pnpm@latest

# ~/.config/buckle/templates/claude-corp1/template.yaml
name: Claude Corp 1
extends: my-corp-base          # your internal base
features:
  - claude-code
  - grok                       # Both Claude Code and Grok Build as first-class citizens
  - mcp:filesystem
  - mcp:github
env:
  ANTHROPIC_LOG: warn
mounts:
  - source: "${localEnv:HOME}/.claude"
    target: /home/vscode/.claude
  - source: "${localEnv:HOME}/.grok"
    target: /home/vscode/.grok

Having both Claude Code and Grok Build in the same devcontainer is explicitly a first-class, well-supported experience in buckle. The two tools compose cleanly (separate config directories, separate postCreate install steps).

This pattern lets you keep company policy in one place while individual teams or roles add their own layers.

Built-in templates

Name Description
ubuntu-base Plain Ubuntu LTS with the vscode user. Good base to extend.
node Node.js 20 (Bullseye), corepack pre-enabled.
python Python 3.12 with pip / venv / uv.
go Go 1.22 with delve, gopls, persistent module cache.
rust Rust stable, rust-analyzer, persistent target cache.
bun Bun runtime on Debian.
deno Deno runtime, secure by default.
polyglot Node + Python + Go on a Debian universal base.
claude-corp Node + Claude Code + GitHub CLI + selected MCPs (good real-world reference).
compose-demo Minimal compose-based multi-service starter (app + sidecar pattern).
ai-native First-class dual-agent setup: Claude Code + Grok Build together, plus common MCPs and tooling. The recommended starting point when you want both major AI coding agents available.

Run buckle list to see the live catalog (built-in + user + installed).

Installing third-party templates

buckle install gh:acme/devcontainer-templates/node-strict
buckle install gh:acme/devcontainer-templates#v2
buckle install gl:acme/templates
buckle install https://example.com/x.git#v1
buckle install file:///abs/path/to/template-dir

Installed templates land under ~/.config/buckle/templates/_installed/<origin-hash>/<template-name>/. They appear in buckle list with an installed (origin-hash) marker.


The 10× developer flow

# Land in a brand-new project, no devcontainer.
$ cd ~/code/some-fresh-clone
$ buckle             # TUI: detects 'package.json' + 'pnpm-lock.yaml' → suggests `node`
                     # Pick template, toggle features, hit "u" → writes .devcontainer & ups.

# Iterate.
$ buckle             # already has .devcontainer → status panel: r rebuild · u up · s/d down · b bash

# Try a quick docker-in-docker test.
$ buckle up --rebuild --feature dind

# Save your favorite stack as a personal template.
$ buckle new claude-corp1 --extend claude-corp
# … edit template.yaml in $EDITOR …
$ buckle up claude-corp1

TUI

Two flows, picked by whether the cwd already has .devcontainer/devcontainer.json:

Wizard (no devcontainer) — auto-detects language signals, suggests top templates, lets you toggle features, previews changes, then either just writes (y) or writes-and-ups (u).

Status panel (devcontainer present) — a one-screen dashboard:

╭───╮  buckle
╰───╯  one verb for devcontainers

workspace: /home/me/myproj
container: buckle.myproj.node
status:    running
image:     mcr.microsoft.com/devcontainers/javascript-node:1-20-bullseye
ports:     3000 → 3000/tcp

r rebuild · u up · s/d down · b bash · q quit

Refresh interval defaults to 5 s; override with BUCKLE_STATUS_REFRESH=2000.


JSON mode

Every command supports --json. The envelope:

{
  "ok": true,
  "timestamp": "2026-05-08T12:34:56.789Z",
  "workspace": "/home/me/proj",
  "data": { "...": "..." }
}

On error:

{
  "ok": false,
  "timestamp": "...",
  "error": { "code": "E_DOCKER_DOWN", "message": "...", "hint": "..." }
}

Error codes are stable: E_DOCKER_DOWN, E_TEMPLATE_NOT_FOUND, E_TEMPLATE_INVALID, E_TEMPLATE_CONFLICT, E_HOOK_FAILED, E_BUILD_FAILED, E_PORT_CONFLICT, E_HASH_MISMATCH, E_CYCLE, E_PERMISSION, E_NO_GIT, E_USER_ABORT, E_INSTALL_FAILED, E_UNSUPPORTED, E_INTERNAL.


Trust model

Templates can run shell commands (postCreate, postStart, …) and mount paths from your host file system. buckle therefore prompts you on first use of any template whose executable surface (lifecycle, mounts, runArgs, features, native features, customizations) is unseen. The trust store lives at ~/.config/buckle/trust.json and maps the merged-template SHA-256 to the date you trusted it. If the surface changes, you're prompted again.

You can:

  • pass --trust to skip the prompt for that one run
  • inspect a template before trusting with buckle edit <name>
  • review what would be written without committing with --preview / --dry-run

buckle does no network access during template resolution. buckle install does network clones via git. buckle doctor is the only command that probes outside the workspace.

See SECURITY.md for the full threat model.


Compatibility

  • Linux (amd64, arm64) — first-class.
  • macOS (Apple Silicon) — first-class with Docker Desktop, OrbStack, or Colima. Buckle inherits DOCKER_HOST / DOCKER_CONTEXT from your shell, so any of these work.
  • WSL2 — works; run buckle from inside WSL.
  • Podman — supported when a Docker-compatible API socket is available (via DOCKER_HOST or alias docker=podman). Many devcontainer features and the official @devcontainers/cli have reduced functionality. Run buckle doctor to see the detected runtime (container.runtime). First-class support is a v1.x goal.

buckle requires the @devcontainers/cli for up / rebuild; buckle doctor will tell you what's missing.

Lifecycle hooks (postCreateCommand etc.) are always emitted as a single flat string joined with &&. If a template declares per-step user: on a hook, buckle folds it away — the whole hook runs as the template's remoteUser (which is vscode in every built-in). The { command, user } per-step form crashes @devcontainers/cli (still broken in 0.87.0) and isn't part of the devcontainer JSON spec for named-object hooks.


Configuration

Per-user file at ~/.config/buckle/config.yaml (optional):

editor: code              # falls back to $VISUAL → $EDITOR → vi
defaultTemplate: node     # used by the wizard if autodetect can't decide

Environment variables:

Variable Effect
BUCKLE_SHELL Preferred shell when buckle bash. Default: zsh > bash > sh.
BUCKLE_STATUS_REFRESH Status-panel refresh in ms (default 5000).
BUCKLE_BUILTIN_DIR Override built-in template directory (testing).
BUCKLE_NO_COLOR Strip ANSI from output (also honors NO_COLOR).
BUCKLE_DEBUG Print full stack traces on uncaught errors.

Testing

Run the full suite (179 tests at the time of writing):

npm test
npm run test:coverage   # coverage thresholds: 90% lines/statements/functions, 80% branches

Coverage excludes the docker-subprocess plumbing layer and the TUI render layer — those are exercised by integration runs against a real docker daemon (make integration in CI).


Architecture

src/
  cli/                # argument parsing, render pipeline, JSON envelope, command shells
    commands/         # one file per subcommand
    parse.ts          # commander program & `buckle <template>` rewrite
    render.ts         # resolve → trust → plan → apply
    install.ts        # gh: / gl: / https / file:// origins
  templates/
    schema.ts         # zod schema (single source of truth)
    loader.ts         # discovery (built-in / user / installed)
    resolver.ts       # extends, deep merge, !replace, cycle detection, hashing
    autodetect.ts     # project signals → suggested templates
    trust.ts          # SHA-256 trust store
    builtin/          # bundled templates
  features/
    catalog.ts        # buckle convenience-feature catalog
    compile.ts        # convenience features → native features + hooks/env/mounts
  generators/
    devcontainer.ts   # Template → devcontainer.json (deterministic, sorted)
    dockerfile.ts     # minimal Dockerfile when build:.dockerfile is missing
    compose.ts        # single-service docker-compose.yml when compose: is set & missing
    writer.ts         # plan / apply / preview / diff
  docker/
    naming.ts         # buckle.<cwd>.<template> with collision suffix
    inspect.ts        # 5-state status (absent | built | running | dead | broken)
    devcontainer-cli.ts  # @devcontainers/cli wrapper
    driver.ts         # high-level: status, up, down, bash, restart, logs
  tui/
    Wizard.tsx        # interactive setup
    StatusPanel.tsx   # interactive dashboard
  util/               # logging, paths (XDG), errors, slug, fs

Stability

buckle is alpha. The CLI surface, JSON shape, error codes, and template schema are stable for 0.x but may shift before 1.0. Expect to pin a specific version in CI.

We follow semver:

  • Pre-1.0: minor versions can be breaking; PATCH versions are bug-fix only.
  • Post-1.0: breaking changes only on MAJOR.

Contributing

PRs welcome. The 4-round design consensus is documented; if you'd change the spec, please reference it.

To work on buckle locally:

git clone https://github.com/buckle-dev/buckle.git
cd buckle
npm install
npm run build
npm test
node bin/buckle.mjs --help

Style: TypeScript strict, ESM, Node ≥ 20, vitest, ink, commander, zod. Lint with npm run lint, auto-format with npm run format. Coverage thresholds in vitest.config.ts are enforced in CI.


License

MIT.

About

One verb for devcontainers — user-wide templates with first-class AI agent support.

Resources

License

Code of conduct

Contributing

Security policy

Stars

Watchers

Forks

Packages

 
 
 

Contributors