Git worktrees
Git worktrees are one kind of workspace.
A workspace is the place where a task happens. When that workspace is backed by a git worktree, Pier creates a separate directory on a separate branch so parallel agents never step on each other.
This page covers the git-specific details: where worktrees live, how branches are chosen, and how to configure setup hooks, scripts, terminals, and long-running services through pier.json.
Layout and workflow
Worktrees live under $PIER_HOME/worktrees/ by default, grouped by a hash of the source checkout path. You can change the base directory with worktrees.root in config.json. Each worktree gets a random slug; the branch name is chosen when you first launch an agent.
~/.pier/worktrees/
└── 1vnnm9k3/ # hash of source checkout path
├── tidy-fox/ # worktree slug (branch set on first agent)
└── bold-owl/
With a custom root, Pier keeps the same hashed layout under that directory:
{
"worktrees": {
"root": "/mnt/fast/pier-worktrees"
}
}
- Create a worktree, Pier runs your setup hooks
- Launch an agent, a branch is created or assigned
- Review the diff against the base branch
- Merge or archive, archive runs teardown and removes the directory
pier.json
Drop a pier.json in your repo root. Pier reads it from the committed version of the base branch you picked, so uncommitted changes in other branches don't apply.
{
"worktree": {
"setup": "npm ci",
"teardown": "rm -rf .cache"
},
"scripts": {
"test": { "command": "npm test" },
"web": { "command": "npm run dev", "type": "service", "port": 3000 }
}
}
Setup and teardown
setup runs once after the worktree is created. A fresh worktree has no installed dependencies and no ignored files (like .env), so use setup to install and copy what you need. teardown runs during archive, before the directory is removed.
{
"worktree": {
"setup": "npm ci\ncp \"$PIER_SOURCE_CHECKOUT_PATH/.env\" .env\nnpm run db:migrate",
"teardown": "npm run db:drop || true"
}
}
Both fields accept a multiline shell script or an array of commands; commands run sequentially either way.
Commands run with the worktree as cwd. Use $PIER_SOURCE_CHECKOUT_PATH to reach files in the original checkout (untracked config, local caches, etc).
Scripts and services
scripts are named commands you can run inside a worktree on demand. Mark one as a service and Pier supervises it as a long-running process, assigns it a port, and routes HTTP traffic to it through the daemon's reverse proxy.
Plain scripts
{
"scripts": {
"test": { "command": "npm test" },
"lint": { "command": "npm run lint" },
"generate": { "command": "npm run codegen" }
}
}
Services
{
"scripts": {
"web": {
"type": "service",
"command": "npm run dev -- --port $PIER_PORT",
"port": 3000
},
"api": {
"type": "service",
"command": "npm run api -- --port $PIER_PORT"
}
}
}
Omit port to let Pier auto-assign one. Bind your process to $PIER_PORT rather than hard-coding, each worktree gets a distinct port so multiple copies of the same service coexist.
Reverse proxy
Every service is reachable through the daemon at a deterministic hostname:
http://<script>--<branch>--<project>.localhost:<daemon-port>
# on the default branch, the branch label is dropped:
http://<script>--<project>.localhost:<daemon-port>
*.localhost resolves to 127.0.0.1 on modern systems, so these URLs work out of the box. The proxy supports WebSocket upgrades.
Service-to-service
Services launched from the same workspace see each other's ports and proxy URLs. Given web and api above, each process gets:
PIER_PORT=3000 # this service's port
PIER_URL=http://web--my-app.localhost:6767 # this service's proxy URL
PIER_SERVICE_API_PORT=51732
PIER_SERVICE_API_URL=http://api--my-app.localhost:6767
PIER_SERVICE_WEB_PORT=3000
PIER_SERVICE_WEB_URL=http://web--my-app.localhost:6767
Script names are upper-cased and non-alphanumerics become _. Point your frontend at $PIER_SERVICE_API_URL instead of hard-coding a port.
Terminals
Open terminals automatically when a worktree is created. Useful for tailing logs or leaving a REPL ready to go.
{
"worktree": {
"terminals": [
{ "name": "logs", "command": "tail -f dev.log" },
{ "name": "shell", "command": "bash" }
]
}
}
Environment variables
Setup, teardown, scripts, and services all see:
$PIER_SOURCE_CHECKOUT_PATH, the original repo root$PIER_WORKTREE_PATH, the worktree directory$PIER_BRANCH_NAME, the worktree's branch$PIER_WORKTREE_PORT, legacy per-worktree port (prefer$PIER_PORTinside services)
Services additionally get:
$PIER_PORT, this service's assigned port$PIER_URL, this service's proxy URL$PIER_SERVICE_<NAME>_PORT/_URL, peer service ports and URLs$HOST,127.0.0.1for local-only daemons,0.0.0.0when the daemon binds all interfaces
CLI
pier run --worktree feature-auth --base main "implement auth"
pier worktree ls
pier worktree archive feature-auth