Skip to content

sQVe/grove

Folders and files

NameName
Last commit message
Last commit date

Latest commit

Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 

Repository files navigation

🌳 Grove

GitHub release License

A fast, intuitive CLI that makes Git worktrees as simple as switching branches.

Without Grove:

git clone --bare [email protected]:org/repo.git .bare
echo "gitdir: ./.bare" > .git
git worktree add main
cd main
cp ../other-worktree/.env .  # Don't forget this
npm install                  # Or this
# Later: cd ../feat-auth to switch

With Grove:

grove clone [email protected]:org/repo
grove add feat/auth --switch  # .env copied, hooks run automatically
grove switch main             # Just like git checkout, but each branch keeps its own directory
demo.mp4

Note

Grove is under active development. Core functionality is stable, but APIs may change between major versions.


New to worktrees?

Git worktrees let you work on multiple branches simultaneously in separate directories. No stashing. No "wrong branch" mistakes. Your work stays exactly where you left it.

The catch: git worktree is clunky. Grove makes it feel like git checkout β€” but each branch gets its own persistent directory.

πŸ“¦ Installation

Quick install (Linux/macOS)

curl -fsSL https://raw.githubusercontent.com/sQVe/grove/main/install.sh | sh

Go install

go install github.com/sqve/grove/cmd/grove@latest

From source

git clone https://github.com/sQVe/grove && cd grove
go build -o grove ./cmd/grove
sudo mv grove /usr/local/bin/

Package managers

Download .deb or .rpm packages from GitHub Releases.

Optional: GitHub CLI

Grove works without additional dependencies, but installing the GitHub CLI (gh) enables enhanced features:

  • PR worktrees: Create worktrees from pull requests with grove add --pr 123 or grove clone https://github.com/owner/repo/pull/123
  • Squash-merge detection: grove prune accurately detects branches merged via GitHub's squash-and-merge, even with multiple commits. Without gh, only single-commit squash merges are detected via git.

See GitHub CLI installation for setup instructions.

πŸ”§ Setup

Shell Integration (Required)

Without this, grove switch only prints a path β€” you'd have to cd manually. Add to your shell config:

# bash (~/.bashrc) or zsh (~/.zshrc)
eval "$(grove switch shell-init)"

# fish (~/.config/fish/config.fish)
grove switch shell-init | source

# powershell ($PROFILE)
grove switch shell-init | Invoke-Expression

Shell Completion

Tab completion for commands, flags, and worktree names.

# bash
eval "$(grove completion bash)"

# zsh
eval "$(grove completion zsh)"

# fish
grove completion fish | source

# powershell
grove completion powershell | Invoke-Expression

πŸš€ Quick Start

# Clone a repository
grove clone https://github.com/owner/repo
cd repo

# Start a feature
grove add feat/auth --switch

# Check your worktrees
grove list

# Switch back to main
grove switch main

# Clean up when done
grove remove feat/auth --branch

πŸ“— Commands

grove clone <url> [directory]

Clone a repository into a Grove workspace.

Flags:

  • --branches <list> β€” Comma-separated branches to create worktrees for
  • --shallow β€” Shallow clone (depth=1)
  • -v, --verbose β€” Show git output

Examples:

grove clone https://github.com/owner/repo
grove clone https://github.com/owner/repo my-project
grove clone https://github.com/owner/repo --branches main,develop
grove clone https://github.com/owner/repo/pull/123 # Clone and checkout PR
grove init new [directory] / grove init convert

Initialize a Grove workspace.

Subcommands:

  • new [directory] β€” Create new workspace
  • convert β€” Convert existing repository

Flags (convert):

  • --branches <list> β€” Additional branches to create worktrees for
  • -v, --verbose β€” Show git output

Examples:

grove init new my-project
grove init convert
grove init convert --branches develop,staging
grove add [branch|PR-URL|ref]

Add a worktree for a branch, pull request, or ref.

Flags:

  • -s, --switch β€” Switch to worktree after creating
  • --base <branch> β€” Create new branch from base instead of HEAD
  • --name <name> β€” Custom directory name
  • -d, --detach β€” Detached HEAD state
  • --pr <number> β€” Create worktree for a pull request

Examples:

grove add feat/auth
grove add feat/auth --switch
grove add --base main feat/auth
grove add --pr 123            # PR by number
grove add --detach v1.0.0     # Tag in detached HEAD
grove switch <worktree>

Switch to a worktree by directory or branch name.

Requires shell integration (see Setup section).

Examples:

grove switch main
grove switch feat-auth
grove switch feat/auth
grove list

List all worktrees with status.

Flags:

  • --fast β€” Skip remote sync checks
  • --filter <status> β€” Filter by: dirty, ahead, behind, gone, locked
  • --json β€” JSON output
  • -v, --verbose β€” Show paths and upstreams

Examples:

grove list
grove list --fast
grove list --filter dirty
grove list --filter ahead,behind
grove list --json
grove status

Show current worktree status.

Flags:

  • -v, --verbose β€” Show all diagnostic sections
  • --json β€” JSON output

Examples:

grove status
grove status --verbose
grove remove <worktree>

Remove a worktree.

Flags:

  • -f, --force β€” Remove even if dirty or locked
  • --branch β€” Also delete the branch

Examples:

grove remove feat-auth
grove remove feat-auth --branch
grove remove --force wip
grove move <worktree> <new-branch>

Rename a branch and its worktree.

Examples:

grove move feat/old feat/new
grove lock <worktree>

Lock a worktree to prevent removal.

Flags:

  • --reason <text> β€” Reason for locking

Examples:

grove lock main
grove lock release --reason "Production release"
grove unlock <worktree>

Unlock a worktree.

Examples:

grove unlock feat-auth
grove prune

Remove worktrees with deleted upstream branches. Dry-run by default.

Flags:

  • --commit β€” Actually remove (default is dry-run)
  • -f, --force β€” Remove even if dirty, locked, or unpushed
  • --stale <duration> β€” Include inactive worktrees (e.g., 30d, 2w)
  • --merged β€” Include branches merged into default branch

Examples:

grove prune          # Dry-run
grove prune --commit # Actually remove
grove prune --stale 30d --commit
grove prune --merged --commit
grove exec [worktrees...] -- <command>

Execute a command in worktrees.

Flags:

  • -a, --all β€” Execute in all worktrees
  • --fail-fast β€” Stop on first failure

Examples:

grove exec --all -- npm install
grove exec main feat-auth -- git pull
grove exec --all --fail-fast -- go build
grove exec --all -- bash -c "npm install && npm test"
grove config <subcommand>

Manage configuration.

Subcommands:

  • list β€” Show all settings
  • get <key> β€” Get value
  • set <key> <value> β€” Set value (requires --shared or --global)
  • unset <key> β€” Remove setting (requires --shared or --global)
  • init β€” Create .grove.toml template

Flags:

  • --shared β€” Target .grove.toml
  • --global β€” Target git config

Examples:

grove config list
grove config get preserve.patterns
grove config set --global plain true
grove config set --shared autolock.patterns "main,release/*"
grove config init
grove doctor

Diagnose workspace issues.

Flags:

  • --fix β€” Auto-fix safe issues
  • --json β€” JSON output
  • --perf β€” Disk space analysis

Detects:

  • Broken .git pointers
  • Stale worktree entries
  • Invalid .grove.toml syntax
  • Stale lock files

Examples:

grove doctor
grove doctor --fix
grove doctor --perf

βš™ Configuration

Grove uses a two-layer configuration system:

  • .grove.toml β€” Team settings (checked into repository)
  • Git config (~/.gitconfig) β€” Personal settings

Run grove config init to create a .grove.toml template.

Default configuration
# Grove - Git worktree management
# https://github.com/sqve/grove

[preserve]
# Files to copy from the current worktree when creating a new one.
# Useful for environment files and local configuration that shouldn't be in git.
# Supports glob patterns. These patterns override git config grove.preserve.
patterns = [
  ".env",
  ".env.keys",
  ".env.local",
  ".env.*.local",
  ".envrc",
  ".grove.toml",
  "docker-compose.override.yml",
]

# Path segments to exclude from preservation.
# Files containing any of these path segments will be skipped.
exclude = [
  # Caches
  ".cache",
  "__pycache__",

  # Build output
  "build",
  "coverage",
  "dist",
  "out",
  "target",

  # Dependencies
  ".venv",
  "node_modules",
  "vendor",
  "venv",
]

[hooks]
# Shell commands to run after creating a worktree.
# Runs sequentially, stops on first failure.
# Examples: ["npm install"], ["go mod download", "make setup"]
add = []

[autolock]
# Branch patterns to automatically lock when creating worktrees.
# Locked worktrees are protected from accidental deletion.
# Supports exact matches and trailing /* wildcards (e.g., "release/*").
patterns = ["develop", "main", "master"]

# Use Nerd Font icons in output (when not in plain mode).
nerd_fonts = true

# Threshold for marking worktrees as stale (no commits within this period).
# Format: number followed by d (days), w (weeks), or m (months).
stale_threshold = "30d"

# Disable colors and symbols in output.
plain = false

# Enable debug logging.
debug = false

🀝 Contributing

Contributions welcome! See CONTRIBUTING.md for development setup and guidelines.

About

A fast, intuitive CLI that makes Git worktrees as simple as switching branches.

Resources

Contributing

Security policy

Stars

Watchers

Forks

Contributors 3

  •  
  •  
  •  

Languages