Git: rebase onto development branch from upstream

前端 未结 3 908
日久生厌
日久生厌 2021-02-05 22:08

I have local master and develop branches. I do all my work on develop and then merge them into master for releases. There is

3条回答
  •  小鲜肉
    小鲜肉 (楼主)
    2021-02-05 22:18

    The simplest (most obvious to everyone) method is to update your master branch first, then rebase onto the updated master which is now exactly the same as origin/master:

    $ git fetch origin                    # Get updates from remote.
    $ git checkout master                 # Now bring master into sync:
    $ git merge --ff-only origin/master   # if this fails you have stuff
                                          # in your master that they don't
                                          # have in theirs, and you need
                                          # to decide what to do about it
    

    At this point, if all went well, master and origin/master are the same (as you can see with graphical viewers like gitk, or with git log --graph --oneline --decorate), and it should be clear how git rebase master will work.

    But you don't actually have to do that. You can just git rebase origin/master, while being on develop. (This will leave your master un-forward-ed—presumably at some point you will want it forward-ed—so it's not really saving you much. But it is easier to do now.)


    The long boring "why" part: git rebase takes, in its long form, three points, which the documentation describes as newbase, upstream, and branch:

    git rebase ... [--onto ] [] []
    

    If you specify exactly one argument, as in git rebase master, that names the upstream and both newbase and branch are computed. The branch is the current branch (i.e., HEAD, which must not be detached). If you omit --onto, the newbase is taken as the upstream argument. So if you're on develop now, and you run git rebase X, the branch is develop and both newbase and upstream are X.

    The rebase method is, in effect (there are various internal optimizations and the effect on the reflog is a bit different):

    1. check out branch
    2. save a reference (ORIG_HEAD) to the commit to which branch points
    3. reset it (a la git reset --hard) to newbase
    4. for every commit in upstream..ORIG_HEAD1 (in oldest to newest order), git cherry-pick that commit to add it to the just-reset branch.

    Thus, as in the documentation:

       Assume the following history exists and the current branch is "topic":
    
                     A---B---C  HEAD=topic
                    /
               D---E---F---G    master
    

    when you git rebase master you get:

                       A---B---C         ORIG_HEAD
                      /
                     /       A'--B'--C'  HEAD=topic
                    /       /
               D---E---F---G             master
    

    (all I did here was take the example in the man page and add the ORIG_HEAD label and the HEAD=, to show that the original commits are "still in there", and that HEAD is a reference to topic).

    So, what happens if you have your develop and master and they have their master which has a few extra changes in it? Let's draw that:

    A -- B -- C                         master
              | \
              |   D                     origin/master
              |
              E -- F                    HEAD=develop
    

    Now you git rebase origin/master:

    A -- B -- C                         master
              | \
              |   D                     origin/master
              |     \
              |       E' -- F'          HEAD=develop
              |
              E -- F                    ORIG_HEAD
    

    At some point, you eventually move your own master to point to commit D too (and you drop ORIG_HEAD) giving:

    A -- B -- C -- D                    master, origin/master
                     \
                       E' - F'          HEAD=develop
    

    which is the same thing with some of the labels moved around.

    That's all that the branch labels are, they are just labels. Each label points to one (single) commit. The commits themselves point back to previous commits, which is what builds the commit tree (or "commit DAG", really).


    1Git's X..Y syntax hides a lot of "deep magic". It means "all commits reachable from label Y that are not reachable from label X. Which is exactly the set of commits that need to be cherry-picked, as those are the commits that were on branch, and were not on upstream, before the "rebase" op. It "looks like" a time-based sequence at first blush, and that generally works in people's heads, but it's based on the commit graph topology. Sometimes this trips people up though: A..B, where A is not related to B at all (because of multiple commit trees in the repo), means "every revision reachable from B.

提交回复
热议问题