+++
title = "wt remove"
description = "Remove worktree; delete branch if merged. Defaults to the current worktree."
weight = 12
[extra]
group = "Commands"
+++
Remove worktree; delete branch if merged. Defaults to the current worktree.
## Examples
Remove current worktree:
{{ terminal(cmd="wt remove") }}
Remove specific worktrees / branches:
Keep the branch:
{{ terminal(cmd="wt remove --no-delete-branch feature-branch") }}
Force-delete an unmerged branch:
{{ terminal(cmd="wt remove -D experimental") }}
## Branch cleanup
By default, branches are deleted when they would add no changes to the default branch if merged. This works with both unchanged git histories, and squash-merge or rebase workflows where commit history differs but file changes match.
Worktrunk checks six conditions (in order of cost):
1. **Same commit** — Branch HEAD equals the default branch. Shows `_` in `wt list`.
2. **Ancestor** — Branch is in target's history (fast-forward or rebase case). Shows `⊂`.
3. **No added changes** — Three-dot diff (`target...branch`) is empty. Shows `⊂`.
4. **Trees match** — Branch tree SHA equals target tree SHA. Shows `⊂`.
5. **Merge adds nothing** — Simulated merge produces the same tree as target. Handles squash-merged branches where target has advanced with changes to different files. Shows `⊂`.
6. **Patch-id match** — Branch's entire diff matches a single squash-merge commit on target. Fallback for when the simulated merge conflicts because target later modified the same files the branch touched. Shows `⊂`.
The 'same commit' check uses the local default branch; for other checks, 'target' means the default branch, or its upstream (e.g., `origin/main`) when strictly ahead.
Branches matching these conditions and with empty working trees are dimmed in `wt list` as safe to delete.
## Force flags
Worktrunk has two force flags for different situations:
| `--force` (`-f`) | Worktree | Worktree has untracked files |
| `--force-delete` (`-D`) | Branch | Branch has unmerged commits |
Without `--force`, removal fails if the worktree contains untracked files. Without `--force-delete`, removal keeps branches with unmerged changes. Use `--no-delete-branch` to keep the branch regardless of merge status.
## Background removal
Removal runs in the background by default — the command returns immediately. The worktree is renamed into `.git/wt/trash/` (instant same-filesystem rename), git metadata is pruned, the branch is deleted, and a detached `rm -rf` finishes cleanup. Cross-filesystem worktrees fall back to `git worktree remove`. Logs: `.git/wt/logs/{branch}/internal/remove.log`. Use `--foreground` to run in the foreground.
After each `wt remove`, entries in `.git/wt/trash/` older than 24 hours are swept by a detached `rm -rf` — eventual cleanup for directories orphaned when a previous background removal was interrupted (SIGKILL, reboot, disk full).
## Hooks
`pre-remove` hooks run before the worktree is deleted (with access to worktree files). `post-remove` hooks run after removal. See [`wt hook`](@/hook.md) for configuration.
## Detached HEAD worktrees
Detached worktrees have no branch name. Pass the worktree path instead: `wt remove /path/to/worktree`.
## See also
- [`wt merge`](@/merge.md) — Remove worktree after merging
- [`wt list`](@/list.md) — View all worktrees
## Command reference
{% terminal() %}
wt remove - Remove worktree; delete branch if merged
Defaults to the current worktree.
Usage: <b><span class=c>wt remove</span></b> <span class=c>[OPTIONS]</span> <span class=c>[BRANCHES]...</span>
<b><span class=g>Arguments:</span></b>
<span class=c>[BRANCHES]...</span>
Branch name [default: current]
<b><span class=g>Options:</span></b>
<b><span class=c>--no-delete-branch</span></b>
Keep branch after removal
<b><span class=c>-D</span></b>, <b><span class=c>--force-delete</span></b>
Delete unmerged branches
<b><span class=c>--foreground</span></b>
Run removal in foreground (block until complete)
<b><span class=c>-f</span></b>, <b><span class=c>--force</span></b>
Force worktree removal
Remove worktrees even if they contain untracked files (like build artifacts). Without this
flag, removal fails if untracked files exist.
<b><span class=c>-h</span></b>, <b><span class=c>--help</span></b>
Print help (see a summary with '-h')
<b><span class=g>Automation:</span></b>
<b><span class=c>--no-hooks</span></b>
Skip hooks
<b><span class=c>--format</span></b><span class=c> <FORMAT></span>
Output format
JSON prints structured result to stdout after removal completes.
Possible values:
- <b><span class=c>text</span></b>: Human-readable text output
- <b><span class=c>json</span></b>: JSON output
[default: text]
<b><span class=g>Global Options:</span></b>
<b><span class=c>-C</span></b><span class=c> <path></span>
Working directory for this command
<b><span class=c>--config</span></b><span class=c> <path></span>
User config file path
<b><span class=c>-v</span></b>, <b><span class=c>--verbose</span></b><span class=c>...</span>
Verbose output (-v: info logs + hook/alias template variable & output; -vv: debug logs +
diagnostic report + trace.log/output.log under .git/wt/logs/)
<b><span class=c>-y</span></b>, <b><span class=c>--yes</span></b>
Skip approval prompts
{% end %}