README
¶
git-go-patch
git-go-patch is a tool that makes it easier to work with a "patched submodule fork" workflow.
It includes several subcommands that help with specific parts of the process.
The Microsoft build of Go repository uses this tool, and it's currently the main reason the tool is being developed and maintained.
A "patched submodule fork" is when you don't hit GitHub's "Fork" button, but rather maintain your own Git repository that contains the upstream repo as a submodule along with *.patch files that modify the submodule when you use git apply patches/*.patch.
For more information about why we chose this style of fork for the Microsoft build of Go repository, see /docs/fork.
Related documentation:
- set-up-repo.md - How to set up your repo to work with git-go-patch.
- microsoft/go Developer Guide - How to use this tool as part of a microsoft/go development workflow.
Installing
First, use Go to build and install the command:
go install github.com/microsoft/go-infra/cmd/git-go-patch@latest
[!NOTE] Make sure
git-go-patchis accessible in your shell'sPATHvariable. You may need to add$GOPATH/binto yourPATH. Usego env GOPATHto locate it.
Then, run the command to see the help documentation:
git go-patch -h
[!NOTE]
gitdetects that ourgit-go-patchexecutable starts withgit-and makes it available asgit go-patch. The program still works if you call it with its real name, but we think it's easier to remember and type something that looks like agitsubcommand.
Subcommands
Make changes to a patch file
Sometimes you have to fix a bug in a patch file, add a new patch file, etc., and apply, rebase, and extract can help.
- Open a terminal anywhere within the repository containing the patch files or the submodule.
- Use
git go-patch applyto apply patches onto the submodule as a series of commits. - Navigate into the submodule.
- Edit the commits as desired. We recommend using an interactive rebase (Pro Git guide) (Git docs) started by
git go-patch rebase. A few recommended editing workflows are:- Commit-then-rebase:
- Make some changes in the submodule and create commits.
- Use
git go-patch rebaseto start an interactive rebase of the commits that include the patch changes and your changes.- This command runs
git rebase -iwith with the necessary base commit. - Reorder the list to put each of your commits under the patch file that it applies to.
- For each commit, choose
squashif you want to edit the commit message orfixupif you don't. Usepickif you want to create a new patch file.
- This command runs
- Follow the usual interactive rebase process.
- Interactive rebase
edit:- Useful if you have an exact change in mind or your commits would hit rebase conflicts.
- Use
git go-patch rebaseto start an interactive rebase before you've made any changes. - Mark commits to edit with
editand save/close the file to continue. - When the rebase process stops at a commit, make your changes, use
git commit --amendto edit the commit, thengit rebase --continueto move on.
- Other
git rebasefeatures likegit commit --fixup={commit}also work as expected.
- Commit-then-rebase:
- Use
git go-patch extractto rewrite the patch files based on the changes in the submodule.
Recovering from a bad rebase
It's possible to accidentally squash a commit into the wrong patch file during a rebase. This makes the change show up in the wrong patch file. To fix this, sometimes it's simplest to start from scratch and copy changes back in manually. However, many general history rewriting methods will work. Here are a few strategies:
Go back to the pre-rebase commit in the submodule
You might be able to go back to the pre-rebase commit and try the rebase again.
The original commit might be in your terminal history: many commands log the commit hashes they operate on.
Or, try git reflog in the submodule to recover the commit hash.
If you anticipate a challenging rebase, you can also preemptively create a temporary branch in the submodule or note down the commit hash before starting the rebase. This way, you know for sure that you can get back to a known state if it goes wrong.
Reallocate your changes
Sometimes the original commits aren't worth recovering, only the sum total of all the changes you made. Then, you can create new commits to try the rebase again.
- In the submodule,
git checkout -B badto save your current state as the branchbad. - In the outer repository, check out the unchanged patch files.
- Run
git apply -fto apply the patch files to the submodule, changing the submodule'sHEAD. - In the submodule,
git checkout bad -- .to copy the changes from thebadbranch into the index (and working directory). - Split the staged changes into your desired commits.
- Try the rebase again.
The Reset Demystified chapter of the Pro Git book may be helpful to understand the state of the Git repository, the index, and the working directory during each step of the recovery process.
Review a PR's patch file changes
When reviewing a pull request that modifies patch files, it can be difficult to understand what the actual changes are by just looking at the diff in GitHub.
The git go-patch review subcommand helps by setting up the changes locally for easier review.
This command:
- Prompts for a GitHub PR URL
- Fetches PR information from GitHub
- Applies the base branch patches with a special "before" marker
- Applies the PR branch patches with a special "after" marker
- Sets up a diff in the Git stage that shows exactly what changes the PR introduces
After running the command, you can review the changes using git diff --cached in the submodule directory or your IDE's Git tools to see a clean presentation of what the PR changes in context.
For more control over each part of the process, or to review changes that aren't in a GitHub PR, use the git go-patch apply -before and -after flags, then git go-patch stage-diff.
Fix up patch files after a submodule update
Every so often, you need to update your submodule to the latest version of the upstream repo.
Just like a rebase or merge, this can generate conflicts when the patches no longer apply cleanly.
The error may look like this:
error: patch failed: src/[...].go:329
To fix this, follow the first two steps of the process to make changes to a patch file.
While running git go-patch apply, you will see the patch failure error appear, with extra instructions about how to use git am to resolve it.
Then:
- Make sure your terminal is inside the submodule.
- Resolve the conflict. There are several ways:
- Run
git am -3. This performs a 3-way merge, and leaves merge conflict markers in the files for manual or tool-assisted fixing. - Run
git am --reject. This creates a.rejfile for each file that couldn't be patched, containing the failed chunks for you to apply manually. - Use your IDE, Git GUI, or another graphical merge tool to resolve the conflict. An
amconflict behaves much like amergeconflict. - Redo the change from scratch.
- See
git amdocumentation for more information.
- Run
- Stage your fixes.
- Run
git am --continueto create the fixed-up commit. - If there are more conflicts, go back to step 2. (The
git am --continuecommand will tell you.) - Run
git go-patch extractto save the fixes to your repository's patch files.
When creating a commit with the fixed patch files, make sure not to include the submodule change.
git go-patch apply creates temporary local commits inside the submodule with unique commit hashes.
References to these hashes won't work in other clones of the repository, causing submodule initialization errors.
If you have many patch files authored by different developers and it isn't reasonable for one person to resolve all the conflicts, you can fix a few patches and run git go-patch extract to save all the fixes completed so far.
Be careful when staging your WIP patch files in the outer repo, because extract doesn't fully understand this situation and will delete the patches that haven't been fixed up yet.
The next dev to work on resolution can then check out the WIP branch and run git go-patch apply to pick up where the last dev left it.
Init submodule and apply patches with a fresh clone
git go-patch apply understands how to set up the patched submodule, so there's no need to run git submodule [...] commands after a fresh clone or checkout:
git clone https://example.org/my/project proj
cd proj
git go-patch apply
# Proj and proj's submodule are now ready to examine.
However, in build scripts, you may want to use traditional Git commands to avoid the dependency on the git-go-patch tool in production environments.
We suggest:
git submodule update --init --recursive
cd submodule
git apply ../patches/*.patch
If you are using Azure DevOps or similar CI mechanism, it may handle submodule initialization for you.
Documentation
¶
There is no documentation for this package.