I've spent hours this week helping students fix problems with their git repos. I've now seen the same types of things happen frequently enough, that I thought I'd offer some advice on how to avoid falling down a git-sized hole. Git does exactly what you tell it to do, regardless of whether or not you meant it to do that. It's worth being aware of what you're saying when you talk to it.
git addfiles using
*or any other wildcard. I know why people do this: it feels like it takes forever to type or copy/paste all the file paths you want to add. Why not just say
git add .and have it add everything that's changed? The problem is, that 1 in every 10 times you do this, you'll add more than you meant to, for example: a submodule change. Whatever amount of time you think you save by not explicitly typing out all the paths you're adding, I promise you that you'll lose it when you have to go back and try to figure out some mess because you accidentally added things you should not have.
git commiton your
masterbranch. Branches in git are cheap, and you should make dozens of them. Make hundreds of them. Don't worry about having too many branches. Doing a bug and you want to try some small thing, make a branch off your branch. Want to try something else? Make a branch of that branch. You should be branching all the time: at least once per bug, feature, etc. You want to commit on every branch except your
masterbranch. Here, you want to only pull changes from upstream and merge them in, ideally with a
fast-fowardmerge. Here's your go-to pattern:
$ git remote add upstream <url-to-upstream-repo> ... $ git checkout master $ git pull upstream master $ git checkout -b fix-bug-123
Before you start any bug fix or experiment, get your
master branch up to date first, then branch off and start your work there.
- Never merge
masterdirectly into your current branch. This seems like a good idea at the time: there are changes on
masterthat you need in order to keep working, or to test your code fully; but doing so ends up causing headaches later when you want to land your code on
master. Imagine you're working on a bug on branch
fix-bug-123and you need to also include what's on
master, what do you do? Let's explore some options:
Option 1: rebase
$ git checkout master $ git pull upstream master $ git checkout fix-bug-123 $ git rebase master
Option 2: make a new temporary branch and merge with
$ git checkout master $ git pull upstream master $ git checkout -b fix-bug-123-master fix-bug-123 $ git merge master
In both cases you will potentially have to deal with merge conflicts. The way you do that depends on whether you're doing a rebase or a merge, but the effect is the same. What's nice about doing the rebase is that you'll probably want to do this later anyway; what's nice about doing the merge on a temporary branch, where you can try things out, is that you don't need to try and keep this work in sync: it's just a way to test things right now while you're working.
What if you make a mistake and do 1) or 2) above, how do you clean things up? Depending on how big a mess you're in, I'd recommend one of the following:
- If committed work on your
masterbranch, it's probably best to reset (note the
-bbelow) your branch to the upstream, so you can pull changes without conflicts. Before you do, drop a branch at your current
masterso you can get at old work:
$ git checkout -b broken-master master $ git fetch upstream $ git checkout -B master upstream/master
master branch has been reset to what the upstream repo is on, and your current work is on
- If you need to get some commits from an old branch, but don't want everything from that branch due to a failed merge, you can
cherry-pickcommits. When you
cherry-pickyou tell git to selectively "copy" commits onto your current branch, one by one. If you have a branch with 3 good commits, and you want them all on a new branch off
master, you could do:
$ git checkout master $ git pull upstream master $ git checkout -b new-branch $ git cherry-pick <first-commit-sha-you-want-from-old-branch> $ git cherry-pick <second-commit-sha-you-want-from-old-branch> $ git cherry-pick <third-commit-sha-you-want-from-old-branch>
- If you want the effect of some commits on a broken branch, but not the commits themselves, you can try creating a
git applyit on a new branch:
$ git checkout master $ git pull upstream master $ git checkout -b bad-branch $ git diff master HEAD > bad-branch.diff
Now you can hand-edit the file
bad-branch.diff to remove any hunks you don't want (e.g., changes to files that shouldn't be there) and save it. Then you do this:
$ git checkout -b new-branch master $ git apply bad-branch.diff
You'll end-up with a new commit that consolidates all the changes you had on the other branch.
- If you want to keep going with a current PR, but need to blow away what's there (e.g., you merged
masterand wish you didn't), you can fix in one of the ways I mention above, then do this:
$ git checkout -b bad-branch-bak bad-branch $ git checkout bad-branch ...fix bad-branch somehow... $ git push origin bad-branch -f
To summarize what I'm doing above:
git checkout -b <branch-name> [base-commit | HEAD]when you do a
-byou are saying: "Create a new branch named
branch-namebased on commit/branch
base-commit, or use HEAD (current commit), and check it out for me.
git checkout -b new-branchmakes a branch,
new-branchoff the current HEAD commit.
git checkout -b new-branch HEADmakes a branch,
new-branchoff the current HEAD commit. Identical to the above, but explicitly says HEAD
git checkout -b new-branch another-branchmakes a branch,
new-branchoff same commit that
git checkout -B new-branch another-branchmakes or resets a branch,
new-branchoff same commit that
another-branchis on. If
new-branchdidn't exist, it gets created; if
new-branchdid exist, it gets moved. Want to blow away your
masterbranch locally and make it match what's upstream?
git reset --hard HEAD && git fetch upstream && git checkout -B master upstream/master
git fetch upstreamdownloads commits, branches, etc. but does not merge what is in the upstream repo. It's like forking, but into your current fork.
git pull upstream masterdoes a fetch of what's in the upstream
masterbranch and then merges it with your current branch. Want to try doing a merge with what's in the upstream
masterwith your current bug, do a
pullinto a new temporary branch:
git checkout -b merge-attempt && git pull upstream master. Now
merge-attemptis your current branch and the upstream
master. Want to go back?
git checkout -which tells git to go to your previous branch (like doing
cd -in bash).