Were using GitHub flow for our opensource project, but I don't understand how to handle hotfixes when there are multiple releases branches containing the bug.
Lets say we have versions 0.0.1
, 0.0.2
released and supported by me and another developer.
Were currently working on a new version 0.0.3
which is in master
.
master
is stable but not ready since we have a few more features to implement before we can commit to version 0.0.3
.
Say we found a critical bug on 0.0.1
that applies for 0.0.2
and 0.0.3
(which hasn't been deployed yet) too.
What is the steps to apply hotfix for this kind of scenario ?
Do we branch out of master
and merge the fix to every release ?
Do we version bump each release (v0.0.1b for example) ?
Thanks !
Here's a basic plan. Switch to
v0.0.1
detached, make the needed fixes, add and commit. Tag that commit, e.g.v0.0.1b
. Now, in turn, switch to each of the other release tags and cherry-pickv0.0.1b
onto it and tag it as well. Finally, switch tomain
and any other existing branches that need this fix and cherry-pickv0.0.1b
onto each of those as well.That was the TL:DR. Now let's talk about why this could be a thing to do, and why it's so obvious when you know what Git is, what a branch is, what a tag is, and what a merge is.
Git is a directed acyclic graph consisting entirely of commits. Each commit is a snapshot of the entire state of the project at a given moment. Each commit has at least one parent, which is what gives the graph its directionality. Git does not contain any diffs or even really any history; the history story is simply a spontaneous traversal of the graph, starting at any given commit and moving to its parent, the parent of that, the parent of that, and so on — the parent chain.
A branch is nothing but a name for one commit. It does have two rather remarkable properties, though: (1) if you are "on" a branch (i.e. that branch is HEAD) and you make a new commit, the branch name is automatically transferred to that commit; and (2) a commit named by a branch will never die, and neither will its parent, nor its parent, and so on. In other words, branches are what keeps the commits in the repo alive.
A tag is similar to a branch but without the first magical property; it's a name for a commit, it keeps that commit alive (along with its parent chain), but it never moves from its commit (unless you move it, but you probably wouldn't do that, as the purpose of the tag is mark forever a place in the history).
A merge is a commit that Git itself makes. It does this by applying merge logic to two commits, the one that is HEAD at the time of the merge and the one that is being merged. This merge logic, simply put, involves calculating the diffs from an earlier parent, the merge base, and applying both diffs simultaneously to the HEAD.
Now then! You've got your
v0.0.1
commit, and guess what, it's faulty. So you need a place to create a fix. You are going to have construct a piece of the history tree yourself, in a controlled and meaningful way. How should the new topology look? Well, in the first place, you want to keepv0.0.1
but you also want a new commit which is just like it except that it incorporates what you now know are the needed fixes. So you switch tov0.0.1
, which is possibly only by asking for a detached head, make the changes, and add-and-commit. You now have an excellent commit, but guess what? If you don't do something to preserve it, that commit will die, because it is not named by a branch or a tag and it has no child — all of the things that would keep a commit alive. So you tag it, and now it will live forever. I'll call the tagv0.0.1b
but it's up to you how you want to handle this naming.Now let's turn to your next release,
v0.0.2
. You want to make the same changes against it as well. That's merge logic! In particular, you probably want tocherry-pick
the commit that you have just created and tagged, the hot fix commit. Why? Well, the merge base used during a cherry-pick is the parent of the cherry-picked commit. So thecherry-pick
will create a new commit that expresses the difference betweenv0.0.1
and the newly created and tagged commit that you made as its child,v0.0.1b
. In other words, it will contain the fixes you want! So switch tov0.0.2
(which requires a detached head), cherry-pickv0.0.1b
, and don't forget to tag it to keep it alive; let's call this onev0.0.2b
.But we are now waving our hands. Do that for all you pre-existing releases. And then, don't forget to do it for
main
as well! You want it to have these fixes too, obviously. The only difference is that when you switch tomain
you are not detached, so you can just cherry-pickv0.0.1b
and stop; the job is done. Now go on with life as usual.