A streamlined workflow for developing in large Bazel + IntelliJ monorepos using Git worktrees.
Enables instant IntelliJ context switching between worktrees—no re-imports, no re-indexing—and scales to support parallel development by humans and AI agents alike.
Git worktrees let you work on multiple branches in parallel, but IntelliJ treats each worktree as a separate project, requiring expensive Bazel syncs and index rebuilds every time you switch to a new worktree.
This toolkit makes IntelliJ context switching instant by:
- Symlink trick: IntelliJ always opens the same path; switching worktrees looks like a branch checkout → incremental refresh in seconds, not minutes
- Metadata vault: IDE project metadata (
.ijwb,.idea,.vscode, etc.) is stored externally and automatically installed into every new worktree—no manual IDE setup needed - Safe worktree management: Automatic stash/restore, branch creation, and cleanup of merged branches
- Parallel development at scale: Works for humans and AI agents alike
📊 See the presentation slides for a visual walkthrough.
# Install (interactive prompts for configuration)
./install.sh
# Reload shell
source ~/.zshrc
# Use
wt helpThe installer will:
- Copy the toolkit to
~/.wt/ - Add sourcing to your shell rc file
- Prompt for workspace paths (main repo, worktrees, metadata vault)
- Create required directories
- Optionally migrate existing repo to worktree structure
- Optionally export project metadata to the vault
- Optionally set up a nightly cron job to refresh Bazel IDE metadata
The directory structure expected (controlled by environment variables, can be overwritten):
~/Development/
├── java -> java-master # Symlink (IntelliJ opens this)
├── java-master/ # Main repository
├── java-worktrees/ # Worktrees go here
└── idea-project-files/ # Project metadata vault
┌─────────────────────────────────────────────┐
│ External Project Metadata Vault │
│ ~/Development/idea-project-files │
│ (IDE configs: .ijwb, .idea, etc.) │
└──────────▲───────────────┬──────────────────┘
│ │
│ │
┌──wt metadata-export─┘ └──wt metadata-import─┐
│ │
┌──────────┴───────────────────────┐ ┌─────────────▼──────────────────────┐
│ Main Repository │ │ Worktrees │
│ ~/Development/java-master │ wt add │ ~/Development/java-worktrees/... │
│ • master branch │ ──────────────────► │ • feature/foo │
│ • safe stash/pull/restore │(calls metadata-imp) │ • bugfix/bar │
│ • never removed │ │ • agent-task-123 │
└───────────────┬──────────────────┘ └─────────┬──────────────────────────┘
│ │
wt switch wt remove
│ │
┌───────────▼──────────────────┐ ┌─────────▼────────────┐
│ Stable IntelliJ Project Dir │ │ Safe cleanup with │
│ ~/Development/java │ │ confirmation prompt │
│ (symlink updated per switch) │ └──────────────────────┘
└───────────▲──────────────────┘
│
IntelliJ auto-refresh
│
┌──────────▼───────────────────┐
│ IntelliJ loads worktree │
│ instantly (no import needed) │
└──────────────────────────────┘
# Existing branch
wt add feature/foo
# New branch (from latest master)
wt add -b feature/fooWhen creating with -b, the script:
- Stashes uncommitted changes
- Switches to master, pulls latest
- Creates branch + worktree
- Imports project metadata from vault
- Restores original state
# Interactive
wt switch
# Direct
wt switch ~/Development/java-worktrees/feature/fooUpdates the symlink so IntelliJ instantly loads the new worktree.
# Interactive cd
wt cd
# Direct cd
wt cd ~/Development/java-worktrees/feature/foowt listShows all worktrees with status indicators:
*= Currently linked worktree[main]= Main repository root[linked]= Active symlink target[dirty]= Has uncommitted changes[↑N]/[↓N]= Commits ahead/behind upstream
# Interactive
wt remove
# Direct (with confirmation)
wt remove ~/Development/java-worktrees/feature/foo
# Skip confirmation (unless uncommitted changes exist)
wt remove -y ~/Development/java-worktrees/feature/foo
# Remove all worktrees with branches merged into base branch
wt remove --merged
# Auto-remove merged without prompts (skips worktrees with uncommitted changes)
wt remove --merged -ySafety features:
- Warns if the worktree is currently linked (symlink will be switched to main repo)
- Warns if there are uncommitted changes (shows summary)
- Always prompts for confirmation if uncommitted changes exist, even with
-y --mergedmode: automatically finds and removes all worktrees whose branches are merged
# Export metadata from main repo to vault (run after setting up new IDE projects)
wt metadata-export
# Import metadata into a worktree (interactive selection if target omitted)
wt metadata-import
wt metadata-import ~/Development/java-worktrees/feature/foo
# Skip confirmation prompts (useful in scripts)
wt metadata-export -y
wt metadata-import -y ~/Development/java-worktrees/feature/fooWhen most development work is done in worktrees, the Bazel IDE directories (.ijwb, .aswb, .clwb) in the main repository can become stale (targets files don't reflect new Bazel targets).
The lib/wt-metadata-refresh script is designed to run as a cron job to keep metadata current.
Note: When IntelliJ has derive_targets_from_directories: true in .bazelproject (the default), it queries Bazel fresh on every sync. The targets-* file serves as a cache for initial project imports and may improve import speed.
Note: The installer (install.sh) offers to set up this cron job automatically (default: yes).
To set it up manually:
# Create log directory
mkdir -p ~/.wt/logs
# Edit crontab
crontab -e
# Add this line to run nightly at 2am (uses login shell for full PATH):
0 2 * * * /bin/zsh -lc '~/.wt/lib/wt-metadata-refresh' >> ~/.wt/logs/metadata-refresh.log 2>&1You can also run the script manually:
# Refresh all Bazel IDE directories and re-export to vault
~/.wt/lib/wt-metadata-refresh
# Preview what would be refreshed (dry run)
~/.wt/lib/wt-metadata-refresh --dry-run
# Refresh targets files only (skip re-export step)
~/.wt/lib/wt-metadata-refresh --no-exportThe refresh script:
- Uses
bazel queryto regeneratetargets/targets-*files in each Bazel IDE directory - Supports all Bazel patterns configured in WT_METADATA_PATTERNS (
.ijwb,.aswb,.clwb) - Parses
.bazelprojectto determine which directories to include in the query - Preserves existing targets file hashes (IntelliJ may reference them)
- Re-exports all metadata to the vault (including non-Bazel patterns)
- Logs timestamped output for monitoring
- Returns exit codes: 0=success, 1=error, 2=partial success
The scripts rely on a few environment variables to know where your main repository, worktrees, and IntelliJ metadata live.
These environment variables are set in wt-common with built-in defaults.
If set in your shell configuration, they take precedence over the built-in defaults.
| Variable | Default | Purpose |
|---|---|---|
WT_MAIN_REPO_ROOT |
~/Development/java-master |
Main repository root |
WT_WORKTREES_BASE |
~/Development/java-worktrees |
Where worktrees are created |
WT_IDEA_FILES_BASE |
~/Development/idea-project-files |
IntelliJ metadata vault |
WT_ACTIVE_WORKTREE |
~/Development/java |
Symlink to active worktree |
WT_BASE_BRANCH |
master |
Default branch for new worktrees |
Path to your primary git repository clone.
Default: ~/Development/java-master
export WT_MAIN_REPO_ROOT="$HOME/Development/java-master"Used by:
- wt-add (for stash/restore & base branch operations)
- wt-choose (listing worktrees)
- wt-switch (default symlink target)
- wt-remove (safety check to prevent removing main repo)
Directory where new worktrees are created by default.
Default: ~/Development/java-worktrees
export WT_WORKTREES_BASE="$HOME/Development/java-worktrees"Canonical metadata vault storing project metadata (IDE configs, etc.).
Default: ~/Development/idea-project-files
export WT_IDEA_FILES_BASE="$HOME/Development/idea-project-files"Used by:
- wt-metadata-import
- wt-metadata-export
- wt-metadata-refresh
- wt-add (when installing metadata)
Symlink path that points to the currently active worktree. This is where IntelliJ should open the project.
Default: ~/Development/java
export WT_ACTIVE_WORKTREE="$HOME/Development/java"Used by:
- wt-switch (updates this symlink)
- wt-remove (warns if removing the linked worktree)
Name of the mainline branch to branch from.
Default: master
export WT_BASE_BRANCH="master"A 10-minute overview presentation is available in the presentation/ directory:
slides.md— Marp markdown sourceslides.pdf— Generated PDF
To regenerate the PDF from the markdown:
npx @marp-team/marp-cli presentation/slides.md -o presentation/slides.pdfwt/
├── wt.sh # Entry point (source this)
├── presentation/ # Overview slides
├── bin/ # Executable commands
│ ├── wt-add
│ ├── wt-cd
│ ├── wt-list
│ ├── wt-remove
│ ├── wt-switch
│ ├── wt-metadata-import
│ └── wt-metadata-export
├── lib/ # Shared libraries
│ ├── wt-common # Configuration and helpers
│ ├── wt-choose # Interactive worktree selection
│ ├── wt-help # Help text for wt command
│ ├── wt-completion # Shell completion for wt command
│ └── wt-metadata-refresh # Cron script to refresh Bazel IDE metadata
├── completion/ # Shell completions for wt-* scripts
│ ├── wt.zsh
│ └── wt.bash
├── install.sh
└── README.md
You can also run the underlying scripts directly:
wt-add, wt-switch, wt-remove, wt-list, wt-cd, wt-metadata-export, wt-metadata-importThese are located in bin/ and work identically to the wt subcommands.
The lib/wt-metadata-refresh script is designed for cron jobs and can be run directly from its location.
| Resource | Description |
|---|---|
| CODEOWNERS | Project lead(s) |
| GOVERNANCE.md | Project governance |
| LICENSE | Apache License, Version 2.0 |