TL;DR:
mkdir project
cd project
git clone --bare <repository> .bare
echo "gitdir: ./.bare" > .git
git worktree add main
git worktree add work
I came across this article from matklad about using git worktrees. I have had several occasions of needing to stash work and switch branches, and it really can be a good bit of mental gymnastics to context switch like that, and then come back to what I had stashed and pick up where I left off. I appreciate their idea of being able to work concurrently on multiple aspects of a project at once using git woktrees.
I have a directory structure that I really like, modeled on how Go structures
directories. Each project goes into ~/Work/src/<upstream>/<repository>
, which
makes it very easy to find projects on the fly. For example, I was recently
working with
koxudaxi/datamodel-code-generator.
While I was exploring the project, I kept a copy at
~/Work/src/github.com/koxudaxi/datamodel-code-generator
. But later I forked
the project in order to make some contributions. My fork goes into
~/Work/src/github.com/andrew-womeldorf/datamodel-code-generator
.
So I wanted to come up with the workflow that would work best with my preferred directory structure. I came across a few different workflows that each had their own quirks that I didn’t really like, until I finally found one I liked.
Neighboring directories ๐
The most common advice I found when reading about worktrees was to just make neighboring directories for each worktree:
git worktree add ../myfeature
However, if I did that, my github directory structure would now contain:
.
โโโ andrew-womeldorf
โโโ datamodel-code-generator
โย ย โโโ .git
โย ย โย ย โโโ branches
โย ย โย ย โโโ COMMIT_EDITMSG
โย ย โย ย โโโ config
โย ย โย ย โโโ description
โย ย โย ย โโโ HEAD
โย ย โย ย โโโ hooks
โย ย โย ย โโโ index
โย ย โย ย โโโ info
โย ย โย ย โโโ logs
โย ย โย ย โโโ objects
โย ย โย ย โโโ refs
โย ย โย ย โโโ worktrees
โย ย โโโ README.md
โโโ myfeature
โโโ .git
โโโ README.md
That doesn’t work well for me, because if I was looking through my directories,
and saw myfeature
, my initial assumption would be that I have a project at
github.com/andrew-womeldorf/myfeature
.
Nested directories ๐
For my structure, it’s better if myfeature
was contained in
~/Work/src/github.com/andrew-womeldorf/datamodel-code-generator
. The problem
with that is that I’d be nesting a duplicate of all project files inside the
project files.
git worktree add myfeature
.
โโโ andrew-womeldorf
โโโ datamodel-code-generator
โโโ .git
โย ย โโโ branches
โย ย โโโ COMMIT_EDITMSG
โย ย โโโ config
โย ย โโโ description
โย ย โโโ HEAD
โย ย โโโ hooks
โย ย โโโ index
โย ย โโโ info
โย ย โโโ logs
โย ย โโโ objects
โย ย โโโ refs
โย ย โโโ worktrees
โโโ myfeature
โย ย โโโ .git
โย ย โโโ README.md
โโโ README.md
That leaves too much opportunity to really screw up and add a ton of unnecessary files.
Neighboring directories nested in a non-git directory ๐
This is getting closer to what I want to see, at least visually. However, being
in andrew-womeldorf/datamodel-code-generator
doesn’t allow me to do anything
with worktrees. I need to go down into the worktrees to do anything, which can
be confusing.
.
โโโ andrew-womeldorf
โโโ datamodel-code-generator
โโโ main
โย ย โโโ .git
โย ย โย ย โโโ branches
โย ย โย ย โโโ COMMIT_EDITMSG
โย ย โย ย โโโ config
โย ย โย ย โโโ description
โย ย โย ย โโโ HEAD
โย ย โย ย โโโ hooks
โย ย โย ย โโโ index
โย ย โย ย โโโ info
โย ย โย ย โโโ logs
โย ย โย ย โโโ objects
โย ย โย ย โโโ refs
โย ย โย ย โโโ worktrees
โย ย โโโ README.md
โโโ myfeature
โโโ .git
โโโ README.md
Bare Repository ๐
Another option is to use a bare git repo.
git clone --bare git@github.com:andrew-womeldorf/datamodel-code-generator
cd datamodel-code-generator
git worktree add main
git worktree add myfeature
.
โโโ andrew-womeldorf
โโโ datamodel-code-generator
โโโ branches
โโโ config
โโโ description
โโโ HEAD
โโโ hooks
โย ย โโโ applypatch-msg.sample
โย ย โโโ commit-msg.sample
โย ย โโโ fsmonitor-watchman.sample
โย ย โโโ post-update.sample
โย ย โโโ pre-applypatch.sample
โย ย โโโ pre-commit.sample
โย ย โโโ pre-merge-commit.sample
โย ย โโโ prepare-commit-msg.sample
โย ย โโโ pre-push.sample
โย ย โโโ pre-rebase.sample
โย ย โโโ pre-receive.sample
โย ย โโโ push-to-checkout.sample
โย ย โโโ update.sample
โโโ info
โย ย โโโ exclude
โโโ main
โย ย โโโ .git
โย ย โโโ README.md
โโโ myfeature
โย ย โโโ .git
โย ย โโโ README.md
โโโ objects
โย ย โโโ 10
โย ย โโโ 4b
โย ย โโโ fa
โย ย โโโ info
โย ย โโโ pack
โโโ packed-refs
โโโ refs
โย ย โโโ heads
โย ย โโโ tags
โโโ worktrees
โโโ main
โโโ myfeature
Ugly.
Bare repository with a reference file ๐
Finally, I found a post by
tomups that had my answer, which
is to use a bare repository in a hidden directory with a text file for .git
that points to the bare repository. I didn’t know you could do this with git,
but it’s pretty cool.
mkdir datamodel-code-generator
cd datamodel-code-generator
git clone --bare git@github.com:andrew-womeldorf/datamodel-code-generator .bare
echo "gitdir: ./.bare" > .git
git worktree add main
git worktree add myfeature
.
โโโ andrew-womeldorf
โโโ datamodel-code-generator
โโโ .bare
โย ย โโโ branches
โย ย โโโ config
โย ย โโโ description
โย ย โโโ HEAD
โย ย โโโ hooks
โย ย โโโ info
โย ย โโโ objects
โย ย โโโ packed-refs
โย ย โโโ refs
โย ย โโโ worktrees
โโโ .git
โโโ main
โย ย โโโ .git
โย ย โโโ README.md
โโโ myfeature
โโโ .git
โโโ README.md
So now each non-hidden directory in the
andrew-womeldorf/datamodel-code-generator
directory is a separate worktree. I
can run git worktree commands from andrew-womeldorf/datamodel-code-generator
.
My preferred directory structure is minimlally impacted. And I don’t have files
nested oddly inside of worktrees just to make worktrees work.