A git worktree manager built for AI agent workflows.
Spin up isolated worktrees for Claude Code sessions.
Switch between them instantly with fzf.
See which agents are busy, waiting, or idle.
Documentation
Running multiple Claude Code sessions on the same repo means constant context-switching, stashing, and branch juggling. Willow fixes this by giving every task its own isolated directory via git worktrees, then adding fzf-based switching and live agent status tracking on top.
~/.willow/
βββ repos/
β βββ myrepo.git/ # bare clone (shared git database)
βββ worktrees/
β βββ myrepo/
β βββ main/ # each branch = isolated directory
β βββ auth-refactor/ # Claude Code running here
β βββ payments/ # another agent running here
βββ status/
βββ myrepo/
βββ auth-refactor/
β βββ <session_id>.json # {"status": "BUSY", ...}
βββ payments/
βββ <session_id>.json # {"status": "WAIT", ...}
brew install iamrajjoshi/tap/willowgo install github.com/iamrajjoshi/willow/cmd/willow@latest# Add to .bashrc / .zshrc
eval "$(willow shell-init)"
# fish
willow shell-init | sourceThis gives you:
| Command | Description |
|---|---|
ww <cmd> |
Alias for willow |
ww sw |
fzf worktree switcher (cd's into selection) |
ww new <branch> |
Create worktree + cd into it (tmux-aware) |
ww checkout <branch> |
Smart checkout + cd (switch or create, tmux-aware) |
wwn <branch> |
Shorthand for ww new |
wwc <branch> |
Shorthand for ww checkout |
www |
cd to ~/.willow/worktrees/ |
Optional: Set terminal tab title to the current worktree name:
eval "$(willow shell-init --tab-title)"Teach Claude Code how to use willow automatically:
# Option 1: npx
npx skills add https://github.com/iamrajjoshi/willow --skill willow
# Option 2: git clone
git clone https://github.com/iamrajjoshi/willow ~/.claude/skills/willowOnce installed, Claude Code will use ww checkout, ww sync, and other willow commands automatically when you ask it to work on branches, PRs, or parallel tasks.
ww cc-setupInstalls hooks into ~/.claude/settings.json that write per-session agent status (BUSY / DONE / WAIT / IDLE) to ~/.willow/status/. Supports multiple Claude sessions per worktree. This powers the status column in ww ls, ww sw, ww status, and ww dashboard.
# Clone a repo (one-time)
ww clone git@github.com:org/myrepo.git
# Create a worktree and cd into it
ww new auth-refactor
# Start Claude Code
claude
# In another terminal β create a second worktree
ww new payments-fix
claude
# Switch between worktrees (fzf picker with agent status)
ww sw
# Check on all agents
ww status
# Clean up when done
ww rm auth-refactorBare-clone a repo and create an initial worktree on the default branch. Required entry point.
ww clone git@github.com:org/repo.git
ww clone git@github.com:org/repo.git myrepo # custom name
ww clone git@github.com:org/repo.git --force # re-clone from scratchCreate a new worktree with a new branch, an existing branch, or a GitHub PR.
ww new feature/auth # create worktree
ww new feature/auth -b develop # fork from specific branch
ww new -e existing-branch # use existing branch
ww new -e # pick from remote branches (fzf)
ww new --pr 123 # checkout PR #123
ww new https://github.com/org/repo/pull/123 # checkout a PR by URL
ww new feature/auth -r myrepo # target a specific repo
ww new feature/auth # auto-cd via shell integration (tmux-aware)| Flag | Description |
|---|---|
-b, --base |
Base branch to fork from |
-r, --repo |
Target repo by name |
-e, --existing |
Use an existing branch (or pick from fzf if no branch given) |
--pr |
GitHub PR number or URL |
--no-fetch |
Skip fetching from remote |
--cd |
Print only the path (for scripting) |
Smart switch-or-create. If a worktree exists for the branch, switch to it. If the branch exists on the remote, create a worktree for it. Otherwise, create a new branch and worktree. Merged worktrees show a [merged] indicator in ww ls and the tmux picker.
ww checkout auth-refactor # switch if exists, create if not
ww checkout --pr 123 # checkout PR #123
ww checkout https://github.com/org/repo/pull/123 # checkout a PR by URL
ww checkout brand-new-feature # creates new branch + worktree
ww checkout brand-new -b develop # new branch from develop
ww checkout auth-refactor # auto-cd via shell integration (tmux-aware)| Flag | Description |
|---|---|
-r, --repo |
Target repo by name |
-b, --base |
Base branch (only when creating a new branch) |
--pr |
GitHub PR number or URL |
--no-fetch |
Skip fetching from remote |
--cd |
Print only the path (for scripting) |
Create stacked branches with --base:
ww new feature-a -b main # start a stack
ww new feature-b -b feature-a # stack on top
ww new feature-c -b feature-b # third layerStacked branches are shown as a tree in ww ls and the tmux picker. Parent relationships are tracked in branches.json per repo.
Show CI, review, and merge status for every PR in a stack at a glance. Fetches all PR data in a single gh pr list call.
ww stack status # current repo
ww stack status -r myrepo # target a specific repo
ww stack status --json # JSON output feature-a #42 β CI β Review MERGEABLE +100 -20
ββ feature-b #43 β CI β― Review CONFLICTING +50 -10
ββ feature-c (no PR)
| Flag | Description |
|---|---|
-r, --repo |
Target repo by name |
--json |
JSON output |
Requires the GitHub CLI (gh).
Rebase stacked worktrees onto their parents in topological order.
ww sync # sync all stacks in current repo
ww sync feature-b # sync feature-b and its descendants only
ww sync --abort # abort any in-progress rebases| Flag | Description |
|---|---|
-r, --repo |
Target repo by name |
--no-fetch |
Skip fetching from remote |
--abort |
Abort in-progress rebases |
Switch worktrees via fzf. Shows Claude Code agent status per worktree, sorted by activity.
π€ BUSY auth-refactor ~/.willow/worktrees/repo/auth-refactor
β
DONE api-cleanup ~/.willow/worktrees/repo/api-cleanup
β³ WAIT payments ~/.willow/worktrees/repo/payments
π‘ IDLE main ~/.willow/worktrees/repo/main
-- old-feature ~/.willow/worktrees/repo/old-feature
Remove a worktree. Without arguments, opens fzf picker with multi-select (TAB to toggle, Ctrl-A to select all).
ww rm auth-refactor # direct removal
ww rm # fzf picker
ww rm auth-refactor --force # skip safety checks
ww rm auth-refactor --prune # also run git worktree prune| Flag | Description |
|---|---|
-f, --force |
Skip safety checks |
--keep-branch |
Keep the local branch |
--prune |
Run git worktree prune after |
List worktrees with status.
| Flag | Description |
|---|---|
--json |
JSON output |
--path-only |
Paths only (one per line) |
Rich view of Claude Code agent status. Shows per-session rows when multiple agents run in the same worktree, with unread indicators (β) for completed sessions you haven't reviewed.
| Flag | Description |
|---|---|
--json |
JSON output |
--cost |
Show estimated token cost per session |
Live-refreshing TUI showing all Claude Code sessions across all repos. Includes diff stats, unread counts, per-session activity, a timeline sparkline showing agent status transitions over the last 60 minutes, and estimated token cost. Press c to toggle the cost column.
ww dashboard # default 2s refresh
ww dash -i 5 # 5s refresh interval
ww dash --no-timeline # hide the timeline column
ww dash --no-cost # hide cost column| Key | Action |
|---|---|
j/k |
Navigate rows |
Enter |
Switch to tmux session |
t |
Toggle timeline column |
r |
Refresh |
q |
Quit |
Show activity log of worktree events (creates, removes, syncs).
ww log # last 20 events
ww log --branch auth-refactor # filter by branch
ww log --repo myrepo # filter by repo
ww log --since 7d # events from last 7 days
ww log -n 50 # last 50 events
ww log --json # raw JSON output| Flag | Description |
|---|---|
--branch |
Filter by branch name |
-r, --repo |
Filter by repo name |
--since |
Show events after duration (e.g. 7d, 24h) |
-n, --limit |
Max events to show (default 20) |
--json |
JSON output |
Desktop notifications for agent status changes. Runs a background daemon that polls agent statuses and fires macOS Notification Center alerts when agents finish or need input.
ww notify on # start background daemon
ww notify on --interval 5 # custom poll interval
ww notify off # stop daemon
ww notify status # check if runningDesktop notifications are off by default. Enable with "notify": {"desktop": true} in config. This applies to both ww notify and the tmux status bar widget.
Create a worktree and launch Claude Code with a prompt. From the terminal, Claude runs interactively in the foreground. From the tmux picker (Ctrl-G), it launches in a background session.
ww dispatch "Fix the login validation bug" # auto-name branch
ww dispatch "Add retry logic" --name add-retries # explicit branch name
ww dispatch "Write tests for auth" --base feature/auth # stacked on a branch
ww dispatch "Refactor payments" --repo myrepo # target specific repo| Flag | Description |
|---|---|
--name |
Worktree/branch name (default: auto-generated from prompt) |
-r, --repo |
Target repo by name |
-b, --base |
Base branch to fork from |
--no-fetch |
Skip fetching from remote |
--yolo |
Run Claude with --dangerously-skip-permissions |
One-time hook installation for Claude Code status tracking.
Check your willow setup for common issues. Verifies git version, optional tools (gh, tmux), Claude Code hooks, willow directories, stale sessions, and config validity.
ww doctorView, edit, and initialize willow configuration.
ww config show # show merged config with sources
ww config show --json # raw JSON output
ww config edit # open global config in $EDITOR
ww config edit --local # open local (per-repo) config
ww config init # create default global config
ww config init --local # create default local configPrint shell integration script.
| Flag | Description |
|---|---|
--tab-title |
Include terminal tab title hook (sets tab to repo/branch) |
After running ww cc-setup, Claude Code automatically reports its state:
| Icon | Status | Meaning |
|---|---|---|
| π€ | BUSY |
Agent is actively working |
| β | DONE |
Agent finished its turn |
| β³ | WAIT |
Agent is waiting for user input |
| π‘ | IDLE |
Agent session ended |
-- |
No activity detected |
Status appears in ww ls, ww sw, ww status, and ww dashboard. Stale BUSY/DONE status (>5 min) automatically degrades to IDLE. Completed sessions show a β unread indicator until you switch to that worktree via ww sw.
Config merges two tiers (local wins):
| Priority | Path | Scope |
|---|---|---|
| 1 | ~/.config/willow/config.json |
Global defaults |
| 2 | ~/.willow/repos/<repo>.git/willow.json |
Per-repo |
Willow collects anonymous usage telemetry via Sentry to help improve the tool. This includes command names, execution times, and error reports. No repo contents, branch names, file paths, or personally identifiable information is sent. Each machine is identified by a hashed hostname only.
Opt out:
# Environment variable
export WILLOW_TELEMETRY=off
# Or in config (persistent)
# ~/.config/willow/config.json
{ "telemetry": false }See the configuration docs for all options.
Use --tab-title to automatically set your terminal tab to the worktree name:
eval "$(willow shell-init --tab-title)"Each tab shows repo/branch (e.g. myrepo/auth-refactor) when inside a willow worktree.
Recommended Ghostty layout per worktree:
βββββββββββββββββββββββββββββββββββββββ
β Tab: myrepo/auth-refactor β
ββββββββββββββββββββ¬βββββββββββββββββββ€
β claude β claude β
β (agent 1) β (agent 2) β
ββββββββββββββββββββ΄βββββββββββββββββββ€
β shell (git diff, tests, etc.) β
βββββββββββββββββββββββββββββββββββββββ
# Build
go build -o bin/willow ./cmd/willow
# Test
go test ./...Requires Go 1.25+ and fzf.
The docs site is built with VitePress.
cd website
pnpm install
pnpm dev # localhost:5173
pnpm build # production buildDeployed automatically to GitHub Pages on push to main when website/ changes.
Releases are automated via GoReleaser and GitHub Actions.





{ "baseBranch": "main", "branchPrefix": "alice", "postCheckoutHook": ".husky/post-checkout", "setup": ["npm install"], "teardown": [], "defaults": { "fetch": true, "autoSetupRemote": true }, "tmux": { "layout": ["split-window -h", "select-layout even-horizontal"], "panes": [ { "command": "cd website" }, { "command": "cd website" } ] }, "cost": { "inputRate": 3.0, // $/M tokens (default: Sonnet 4) "outputRate": 15.0 // $/M tokens (default: Sonnet 4) } }