137 lines
5.1 KiB
Markdown
137 lines
5.1 KiB
Markdown
---
|
||
title: git format-patch is your Friend
|
||
date: 2014-05-21
|
||
excerpt: An explanation of the "git format-patch" command, and how it could be used in Drupal's Git workflow.
|
||
tags:
|
||
- drupal
|
||
- drupal-planet
|
||
- git
|
||
- patches
|
||
---
|
||
|
||
An explanation of the "git format-patch" command, and how it could be used in
|
||
Drupal's Git workflow.
|
||
|
||
## The Problem
|
||
|
||
As an active contributor to the [Drupal](http://drupal.org) project, I spend a
|
||
lot of time working with other peoples’ modules and themes, and occassionally
|
||
have to fix a bug or add some new functionality.
|
||
|
||
In the Drupal community, we use a patch based workflow where any changes that I
|
||
make get exported to a file detailing the differences. The patch file (\*.patch)
|
||
is attached to an item in an issue queue on Drupal.org, applied by the
|
||
maintainer to their local copy of the code and reviewed, and hopefully
|
||
committed.
|
||
|
||
There is an option that the maintainer can add to the end of their commit
|
||
message.
|
||
|
||
For example:
|
||
|
||
```bash
|
||
--author="opdavies <opdavies@381388.no-reply.drupal.org>"
|
||
```
|
||
|
||
This differs slightly different for each Drupal user, and the code can be found
|
||
on their Drupal.org profile page.
|
||
|
||
If this is added to the end of the commit message, the resulting commit will
|
||
show that it was committed by the maintainer but authored by a different user.
|
||
This will then display on Drupal.org that you’ve made a commit to that project.
|
||
|
||
![A screenshot of a commit that was authored by rli but committed by opdavies](/images/blog/git-format-patch.png)
|
||
|
||
The problem is that some project maintainers either don’t know about this option
|
||
or occasionally forget to add it. [Dreditor](http://dreditor.org) can suggest a
|
||
commit message and assign an author, but it is optional and, of course, not all
|
||
maintainers use Dreditor (although they probably should).
|
||
|
||
The `git format-patch` command seems to be the answer, and will be my preferred
|
||
method for generating patch files in the future rather than `git diff`.
|
||
|
||
## What does it do Differently?
|
||
|
||
From the [manual page](http://git-scm.com/docs/git-format-patch):
|
||
|
||
> Prepare each commit with its patch in one file per commit, formatted to
|
||
> resemble UNIX mailbox format. The output of this command is convenient for
|
||
> e-mail submission or for use with git am.
|
||
|
||
Here is a section of a patch that I created for the
|
||
[Metatag module](http://drupal.org/project/metatag) using `git format-patch`:
|
||
|
||
```bash
|
||
From 80c8fa14de7f4a83c2e70367aab0aedcadf4f3b0 Mon Sep 17 00:00:00 2001
|
||
From: Oliver Davies <oliver@oliverdavies.co.uk>
|
||
Subject: [PATCH] Exclude comment entities when checking if this is the page,
|
||
otherwise comment_fragment.module will break metatag
|
||
---
|
||
```
|
||
|
||
As mentioned above, the patch is structured in an email format. The commit
|
||
message is used as the subject line, and the date that the commit was made
|
||
locally is used for the date. What we’re interested in is the “From” value. This
|
||
contains your name and email address from your `~/.gitconfig` file and is used
|
||
to author the patch automatically.
|
||
|
||
Everything below this is the same as a standard patch file, the same as if was
|
||
generated with `git diff`.
|
||
|
||
The full patch file can be found at
|
||
<https://drupal.org/files/issues/metatag-comment-fragment-conflict-2265447-4.patch>.
|
||
|
||
## The Process
|
||
|
||
How did I create this patch? Here are the steps that I took:
|
||
|
||
1. Clone the source repository using
|
||
`$ git clone --branch 7.x-1.x http://git.drupal.org/project/metatag.git` and
|
||
move into that directory.
|
||
2. Create a branch for this patch using
|
||
`$ git checkout -b 2265447-comment-fragment-conflict`.
|
||
3. Add and commit any changes as normal.
|
||
4. Generate the patch file using
|
||
`$ git format-patch 7.x-1.x --stdout > metatag-comment-fragment-conflict-2265447-4.patch`.
|
||
|
||
_Note:_ I am defining 7.x-1.x in the last step as the original branch to compare
|
||
(i.e. the original branch that we forked to make our issue branch). This will
|
||
change depending on the project that you are patching, and it’s version number.
|
||
Also, commits should always be made against the development branch and not the
|
||
stable release.
|
||
|
||
By default, a separate patch file will be created for each commit that we’ve
|
||
made. This is overridden by the `--stdout` option which combines all of the
|
||
patches into a single file. This is the recommended approach when uploading to
|
||
Drupal.org.
|
||
|
||
The resulting patch file can be uploaded onto a Drupal.org issue queue, reviewed
|
||
by the Testbot and applied by a module maintainer, and you automatically get the
|
||
commit attributed. Problem solved.
|
||
|
||
## Committing the Patch
|
||
|
||
If you need to commit a patch that was created using `git format-patch`, the
|
||
best command to do this with is the `git am` command.
|
||
|
||
For example, within your repository, run:
|
||
|
||
```bash
|
||
$ git am /path/to/file
|
||
$ git am ~/Code/metatag-comment-fragment-conflict-2265447-4.patch
|
||
```
|
||
|
||
You should end up with some output similar to the following:
|
||
|
||
```bash
|
||
Applying: #2272799 Added supporters section
|
||
Applying: #2272799 Added navigation tabs
|
||
Applying: #2272799 Fixed indentation
|
||
Applying: #2272799 Replaced URL
|
||
```
|
||
|
||
Each line is the commit message associated with that patch.
|
||
|
||
Assuming that there are no errors, you can go ahead and push your updated code
|
||
into your remote repository.
|