165 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			Plaintext
		
	
	
			
		
		
	
	
			165 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			Plaintext
		
	
	
| From:	Junio C Hamano <gitster@pobox.com>
 | |
| To:	git@vger.kernel.org
 | |
| Cc:	Petr Baudis <pasky@suse.cz>, Linus Torvalds <torvalds@osdl.org>
 | |
| Subject: Re: sending changesets from the middle of a git tree
 | |
| Date:	Sun, 14 Aug 2005 18:37:39 -0700
 | |
| Abstract: In this article, JC talks about how he rebases the
 | |
|  public "seen" branch using the core Git tools when he updates
 | |
|  the "master" branch, and how "rebase" works.  Also discussed
 | |
|  is how this applies to individual developers who sends patches
 | |
|  upstream.
 | |
| Content-type: text/asciidoc
 | |
| 
 | |
| How to rebase from an internal branch
 | |
| =====================================
 | |
| 
 | |
| --------------------------------------
 | |
| Petr Baudis <pasky@suse.cz> writes:
 | |
| 
 | |
| > Dear diary, on Sun, Aug 14, 2005 at 09:57:13AM CEST, I got a letter
 | |
| > where Junio C Hamano <junkio@cox.net> told me that...
 | |
| >> Linus Torvalds <torvalds@osdl.org> writes:
 | |
| >>
 | |
| >> > Junio, maybe you want to talk about how you move patches from your
 | |
| >> > "seen" branch to the real branches.
 | |
| >>
 | |
| > Actually, wouldn't this be also precisely for what StGIT is intended to?
 | |
| --------------------------------------
 | |
| 
 | |
| Exactly my feeling.  I was sort of waiting for Catalin to speak
 | |
| up.  With its basing philosophical ancestry on quilt, this is
 | |
| the kind of task StGIT is designed to do.
 | |
| 
 | |
| I just have done a simpler one, this time using only the core
 | |
| Git tools.
 | |
| 
 | |
| I had a handful of commits that were ahead of master in 'seen', and I
 | |
| wanted to add some documentation bypassing my usual habit of
 | |
| placing new things in 'seen' first.  At the beginning, the commit
 | |
| ancestry graph looked like this:
 | |
| 
 | |
| 			     *"seen" head
 | |
|     master --> #1 --> #2 --> #3
 | |
| 
 | |
| So I started from master, made a bunch of edits, and committed:
 | |
| 
 | |
|     $ git checkout master
 | |
|     $ cd Documentation; ed git.txt ...
 | |
|     $ cd ..; git add Documentation/*.txt
 | |
|     $ git commit -s
 | |
| 
 | |
| After the commit, the ancestry graph would look like this:
 | |
| 
 | |
| 			      *"seen" head
 | |
|     master^ --> #1 --> #2 --> #3
 | |
|           \
 | |
|             \---> master
 | |
| 
 | |
| The old master is now master^ (the first parent of the master).
 | |
| The new master commit holds my documentation updates.
 | |
| 
 | |
| Now I have to deal with "seen" branch.
 | |
| 
 | |
| This is the kind of situation I used to have all the time when
 | |
| Linus was the maintainer and I was a contributor, when you look
 | |
| at "master" branch being the "maintainer" branch, and "seen"
 | |
| branch being the "contributor" branch.  Your work started at the
 | |
| tip of the "maintainer" branch some time ago, you made a lot of
 | |
| progress in the meantime, and now the maintainer branch has some
 | |
| other commits you do not have yet.  And "git rebase" was written
 | |
| with the explicit purpose of helping to maintain branches like
 | |
| "seen".  You _could_ merge master to 'seen' and keep going, but if you
 | |
| eventually want to cherrypick and merge some but not necessarily
 | |
| all changes back to the master branch, it often makes later
 | |
| operations for _you_ easier if you rebase (i.e. carry forward
 | |
| your changes) "seen" rather than merge.  So I ran "git rebase":
 | |
| 
 | |
|     $ git checkout seen
 | |
|     $ git rebase master seen
 | |
| 
 | |
| What this does is to pick all the commits since the current
 | |
| branch (note that I now am on "seen" branch) forked from the
 | |
| master branch, and forward port these changes.
 | |
| 
 | |
|     master^ --> #1 --> #2 --> #3
 | |
| 	  \                                  *"seen" head
 | |
|             \---> master --> #1' --> #2' --> #3'
 | |
| 
 | |
| The diff between master^ and #1 is applied to master and
 | |
| committed to create #1' commit with the commit information (log,
 | |
| author and date) taken from commit #1.  On top of that #2' and #3'
 | |
| commits are made similarly out of #2 and #3 commits.
 | |
| 
 | |
| Old #3 is not recorded in any of the .git/refs/heads/ file
 | |
| anymore, so after doing this you will have dangling commit if
 | |
| you ran fsck-cache, which is normal.  After testing "seen", you
 | |
| can run "git prune" to get rid of those original three commits.
 | |
| 
 | |
| While I am talking about "git rebase", I should talk about how
 | |
| to do cherrypicking using only the core Git tools.
 | |
| 
 | |
| Let's go back to the earlier picture, with different labels.
 | |
| 
 | |
| You, as an individual developer, cloned upstream repository and
 | |
| made a couple of commits on top of it.
 | |
| 
 | |
|                               *your "master" head
 | |
|    upstream --> #1 --> #2 --> #3
 | |
| 
 | |
| You would want changes #2 and #3 incorporated in the upstream,
 | |
| while you feel that #1 may need further improvements.  So you
 | |
| prepare #2 and #3 for e-mail submission.
 | |
| 
 | |
|     $ git format-patch master^^ master
 | |
| 
 | |
| This creates two files, 0001-XXXX.patch and 0002-XXXX.patch.  Send
 | |
| them out "To: " your project maintainer and "Cc: " your mailing
 | |
| list.  You could use contributed script git-send-email if
 | |
| your host has necessary perl modules for this, but your usual
 | |
| MUA would do as long as it does not corrupt whitespaces in the
 | |
| patch.
 | |
| 
 | |
| Then you would wait, and you find out that the upstream picked
 | |
| up your changes, along with other changes.
 | |
| 
 | |
|    where                      *your "master" head
 | |
|   upstream --> #1 --> #2 --> #3
 | |
|     used   \
 | |
|    to be     \--> #A --> #2' --> #3' --> #B --> #C
 | |
|                                                 *upstream head
 | |
| 
 | |
| The two commits #2' and #3' in the above picture record the same
 | |
| changes your e-mail submission for #2 and #3 contained, but
 | |
| probably with the new sign-off line added by the upstream
 | |
| maintainer and definitely with different committer and ancestry
 | |
| information, they are different objects from #2 and #3 commits.
 | |
| 
 | |
| You fetch from upstream, but not merge.
 | |
| 
 | |
|     $ git fetch upstream
 | |
| 
 | |
| This leaves the updated upstream head in .git/FETCH_HEAD but
 | |
| does not touch your .git/HEAD or .git/refs/heads/master.
 | |
| You run "git rebase" now.
 | |
| 
 | |
|     $ git rebase FETCH_HEAD master
 | |
| 
 | |
| Earlier, I said that rebase applies all the commits from your
 | |
| branch on top of the upstream head.  Well, I lied.  "git rebase"
 | |
| is a bit smarter than that and notices that #2 and #3 need not
 | |
| be applied, so it only applies #1.  The commit ancestry graph
 | |
| becomes something like this:
 | |
| 
 | |
|    where                     *your old "master" head
 | |
|   upstream --> #1 --> #2 --> #3
 | |
|     used   \                      your new "master" head*
 | |
|    to be     \--> #A --> #2' --> #3' --> #B --> #C --> #1'
 | |
|                                                 *upstream
 | |
|                                                 head
 | |
| 
 | |
| Again, "git prune" would discard the disused commits #1-#3 and
 | |
| you continue on starting from the new "master" head, which is
 | |
| the #1' commit.
 | |
| 
 | |
| -jc
 |