jb-worktree
wtp Git Worktrees
Based on satococoa's wtp workflow: https://dev.to/satococoa/wtp-a-better-git-worktree-cli-tool-4i8l
Use wtp instead of raw git worktree when you want predictable paths, easier branch handling, setup hooks, and fast navigation.
Why use it
wtp initcreates a starter.wtp.ymlin the repo rootwtp add feature/authcreates../worktrees/feature/authautomaticallywtp addcan create or reuse local/remote branches.wtp.ymlcan copy.env, create symlinks, and run bootstrap commandswtp cdandwtp execmake navigation and command execution easierwtp remove --with-branchcleans up both the worktree and its branch
Install
brew install satococoa/tap/wtp
Alternative:
go install github.com/satococoa/wtp/v2/cmd/wtp@latest
Quick start
# Create a starter .wtp.yml in the repository root
wtp init
# Create a worktree from an existing branch
wtp add feature/auth
# Create a new branch + worktree
wtp add -b feature/new-ui
# Create from a specific base ref
wtp add -b hotfix/login origin/main
# List all worktrees
wtp list
# Print the absolute path for a worktree
wtp cd feature/auth
wtp cd @
# Run a command inside a worktree
wtp exec feature/auth -- npm test
# Remove a worktree
wtp remove feature/auth
# Remove a worktree and delete its branch too
wtp remove --with-branch feature/auth
Agent workflow
When the user asks to work in a separate branch or isolated checkout:
- Check whether the repo already has
.wtp.yml; if not, preferwtp init - Inspect repo lockfiles before editing hooks:
- if
bun.lockorbun.lockbexists, usebun install - if
pnpm-lock.yamlexists, usepnpm install --frozen-lockfile - otherwise, if the repo has a
package-lock.jsonand no Bun lockfile, usepnpm installby default, and preferpnpm install --frozen-lockfilewhen reproducibility matters
- if
- Check the current worktrees with
wtp list - Create or open the target worktree with
wtp add ... - Run commands inside it with either:
cd "$(wtp cd <name>)" && ...wtp exec <name> -- <command>
- When the work is done and merged, clean up with
wtp remove --with-branch <name>
Prefer:
wtp add -b <branch>for new workwtp add <branch>when the branch already exists locally or remotelywtp exec <name> -- <command>for one-off commands
Avoid force removal of dirty worktrees unless the user explicitly asks.
wtp init
Use this first in repos that do not have a config yet:
wtp init
It creates .wtp.yml in the repository root with a starter configuration and example hooks. If .wtp.yml already exists, wtp init errors instead of overwriting it.
Recommended .wtp.yml
wtp init gives you a starter file. Customize the post_create install command based on the repo lockfile:
bun.lockorbun.lockbpresent →bun installpnpm-lock.yamlpresent →pnpm install --frozen-lockfile- otherwise,
package-lock.jsonpresent and no Bun lockfile →pnpm installby default, orpnpm install --frozen-lockfilefor stricter reproducible installs
Example for a Bun repo:
version: "1.0"
defaults:
base_dir: "../worktrees"
hooks:
post_create:
- type: copy
from: ".env"
to: ".env"
- type: symlink
from: ".bin"
to: ".bin"
- type: command
command: "bun install"
- type: command
command: "npm run db:setup"
Example for a repo with pnpm-lock.yaml:
version: "1.0"
defaults:
base_dir: "../worktrees"
hooks:
post_create:
- type: copy
from: ".env"
to: ".env"
- type: symlink
from: ".bin"
to: ".bin"
- type: command
command: "pnpm install --frozen-lockfile"
- type: command
command: "npm run db:setup"
Example for a repo with a pure package-lock.json setup:
version: "1.0"
defaults:
base_dir: "../worktrees"
hooks:
post_create:
- type: copy
from: ".env"
to: ".env"
- type: symlink
from: ".bin"
to: ".bin"
- type: command
command: "pnpm install"
- type: command
command: "npm run db:setup"
This is especially useful for repos that need local env files, shared tool directories, or bootstrap commands in every new worktree.
Shell integration
For interactive shells, enable completions and navigation hooks:
eval "$(wtp shell-init zsh)"
# or
# eval "$(wtp shell-init bash)"
# wtp shell-init fish | source
Then wtp cd feature/auth can switch directly in the shell, and interactive wtp add can auto-switch into the new worktree.
Useful patterns
# Return to the main worktree
wtp cd @
# Run tests in the main worktree
wtp exec @ -- npm test
# Force remove a dirty worktree only when you are sure
wtp remove --force feature/auth
# Remove worktree and force-delete its branch
wtp remove --with-branch --force-branch feature/auth
Notes
- Default generated paths are based on branch names under
../worktrees - Remote-only branches are tracked automatically when unambiguous
- If multiple remotes contain the same branch name, create a local tracking branch first
wtp cdis also useful in scripts because it prints the resolved absolute path