YAML-driven command runner with nested subcommands

Readme

wand

wand is a tiny, cross-platform command runner driven by a simple YAML config file, written in Go. Define your commands and subcommands in a wand.yml, and run them from anywhere in your project tree.

Release Downloads License


πŸš€ Features

  • Simple YAML config: define commands, descriptions, and nested subcommands in a single file.
  • Auto-discovery: finds wand.yml by searching the current directory, parent directories, ~/, and ~/.config/.
  • Nested subcommands: commands can have arbitrarily deep children.
  • Positional arguments: pass arguments to commands and reference them with $1, $2, $@.
  • Custom flags: define typed flags (string or bool) with aliases, defaults, and descriptions, accessible as $WAND_FLAG_<NAME> environment variables.
  • Environment variables: define env vars globally in .config or per command, with command-level overrides.
  • Working directory: override the working directory for any command.
  • Aliases: define alternate names for commands.
  • Confirmation prompts: require y/N confirmation before running destructive commands.
  • Pre/post hooks: chain other wand commands to run before or after a command, with full flag/argument forwarding.
  • Built-in help: auto-generated --help for every command and subcommand.
  • Shell execution: runs commands via your $SHELL with proper stdin/stdout/stderr passthrough.

🎯 Installation

Download Precompiled Binaries

Grab the latest release for Linux, macOS, or Windows:

Homebrew (macOS/Linux)

Install directly from the tap:

brew install chenasraf/tap/wand

Or tap and then install the package:

brew tap chenasraf/tap
brew install wand

From Source

git clone https://github.com/chenasraf/wand
cd wand
make build

✨ Getting Started

Create a wand.yml in your project root:

main:
  description: run the main command
  cmd: echo hello from wand

build:
  description: build the project
  cmd: go build -o myapp

test:
  description: run tests
  cmd: go test -v ./...
  children:
    coverage:
      description: run tests with coverage
      cmd: go test -coverprofile=coverage.out ./...

Run a command

# run the main (default) command
wand

# run a named command
wand build

# run a nested subcommand
wand test coverage

# show help
wand --help
wand test --help

πŸ“ Config Resolution

wand searches for wand.yml (or wand.yaml) in the following order:

  1. Current working directory (./wand.yml)
  2. Parent directories (searching upward to the filesystem root)
  3. Home directory (~/.wand.yml)
  4. Config directory (~/.config/wand.yml)

The first config file found is used.

You can override config discovery with an explicit path:

# via flag
wand --wand-file ./other-config.yml build

# via environment variable
WAND_FILE=./other-config.yml wand build

The --wand-file flag takes precedence over WAND_FILE.


πŸ“– Config Reference

Each top-level key defines a command. The special key main becomes the root (no-argument) command.

FieldTypeDescription
descriptionstringShort description shown in --help
cmdstringShell command to execute
childrenmap[string]CommandNested subcommands (same structure)
flagsmap[string]FlagCustom flags (see below)
envmap[string]stringEnvironment variables for this command
working_dirstringWorking directory for the command
aliases[]stringAlternate names for the command
confirmbool or stringPrompt for confirmation before running
confirm_defaultstringDefault answer: "yes" or "no" (default)
pre[]stringWand commands to run before cmd
post[]stringWand commands to run after cmd

Flag fields

FieldTypeDescription
aliasstringSingle-letter shorthand (e.g. o for -o)
descriptionstringDescription shown in --help
defaultanyDefault value (string or bool)
typestring"bool" for boolean flags, omit for string flags

πŸ“Œ Positional Arguments

Commands receive any extra arguments passed on the command line. Use $1, $2, etc. for specific positions, or $@ for all arguments:

greet:
  description: greet someone
  cmd: echo "Hello, $1! You said: $@"
wand greet world foo bar
# β†’ Hello, world! You said: world foo bar

🚩 Flags

Define custom flags per command. Flag values are exposed as $WAND_FLAG_<NAME> environment variables (uppercased):

build:
  description: build the project
  cmd: |
    echo "output=$WAND_FLAG_OUTPUT verbose=$WAND_FLAG_VERBOSE"
  flags:
    output:
      alias: o
      description: output path
      default: ./bin
    verbose:
      alias: v
      description: enable verbose output
      type: bool
wand build --output ./dist --verbose
# β†’ output=./dist verbose=true

wand build -o ./dist -v
# β†’ output=./dist verbose=true

wand build
# β†’ output=./bin verbose=false

🌍 Environment Variables

Define environment variables globally in .config or per command. Command-level env vars override global ones:

.config:
  env:
    NODE_ENV: production

build:
  description: build the project
  cmd: echo "env=$NODE_ENV out=$OUTPUT_DIR"
  env:
    OUTPUT_DIR: ./dist
wand build
# β†’ env=production out=./dist

⚠️ Confirmation Prompts

Add confirm: true for a default prompt, or provide a custom message:

deploy:
  description: deploy to production
  cmd: ./deploy.sh
  confirm: 'Deploy to production?'

clean:
  description: remove all build artifacts
  cmd: rm -rf dist/
  confirm: true

restart:
  description: restart service
  cmd: systemctl restart myapp
  confirm: 'Restart the service?'
  confirm_default: 'yes'
wand deploy
# β†’ Deploy to production? [y/N]

πŸ”— Pre & Post Hooks

Use pre and post to run other wand commands before or after a command. Each entry is a shell-style string: the first token is the wand command name (subcommands are nested with spaces), followed by any args and flags.

lint:
  description: lint the project
  cmd: golangci-lint run

test:
  description: run tests
  cmd: go test ./...

build:
  description: build the project
  pre:
    - lint
    - test
  post:
    - 'echo "build done: $WAND_FLAG_OUTPUT"'
  flags:
    output:
      alias: o
      default: ./bin
  cmd: go build -o $WAND_FLAG_OUTPUT
wand build -o ./dist
# runs: lint β†’ test β†’ go build -o ./dist β†’ echo "build done: ./dist"

Forwarding flags

Entries are passed through environment variable expansion ($VAR, ${VAR}) before being parsed, so $WAND_FLAG_<NAME> references resolve to the current command’s flag values:

deploy:
  flags:
    target:
      description: deploy target
      default: staging
  pre:
    - 'notify --channel deploys --message "deploying to $WAND_FLAG_TARGET"'
  cmd: ./deploy.sh $WAND_FLAG_TARGET

Arbitrary flags and arguments can be passed directly:

release:
  pre:
    - 'test --verbose'
    - 'build --output ./dist'
  cmd: ./release.sh

Failure semantics

  • If a pre entry fails, the main cmd and remaining pre/post entries are skipped.
  • If the main cmd fails, no post entries run.
  • If a post entry fails, subsequent post entries are skipped.

A command may omit cmd and define only pre/post to act as a pure aggregator.

Private commands

Prefix a command name with _ to mark it as private: it is hidden from --help output but remains fully runnable, both directly (useful for testing) and from pre/post entries. The same rule applies to nested children.

build:
  pre:
    - _ensure-deps
  cmd: go build

_ensure-deps:
  description: install required tools
  cmd: ./scripts/install-deps.sh
wand --help        # _ensure-deps is not listed
wand _ensure-deps  # still runs directly
wand build         # runs _ensure-deps then the build

πŸ› οΈ Contributing

I am developing this package on my free time, so any support, whether code, issues, or just stars is very helpful to sustaining its life. If you are feeling incredibly generous and would like to donate just a small amount to help sustain this project, I would be very very thankful!

Buy Me a Coffee at ko-fi.com

I welcome any issues or pull requests on GitHub. If you find a bug, or would like a new feature, don’t hesitate to open an appropriate issue and I will do my best to reply promptly.


πŸ“œ License

wand is licensed under the CC0-1.0 License.