1#!/bin/sh 2# 3# Copyright (c) 2005 Junio C Hamano. 4# 5 6USAGE='[-v] [--onto <newbase>] <upstream> [<branch>]' 7LONG_USAGE='git-rebase replaces <branch> with a new branch of the 8same name. When the --onto option is provided the new branch starts 9out with a HEAD equal to <newbase>, otherwise it is equal to <upstream> 10It then attempts to create a new commit for each commit from the original 11<branch> that does not exist in the <upstream> branch. 12 13It is possible that a merge failure will prevent this process from being 14completely automatic. You will have to resolve any such merge failure 15and run git rebase --continue. Another option is to bypass the commit 16that caused the merge failure with git rebase --skip. To restore the 17original <branch> and remove the .dotest working files, use the command 18git rebase --abort instead. 19 20Note that if <branch> is not specified on the command line, the 21currently checked out branch is used. You must be in the top 22directory of your project to start (or continue) a rebase. 23 24Example: git-rebase master~1 topic 25 26 A---B---C topic A'\''--B'\''--C'\'' topic 27 / --> / 28 D---E---F---G master D---E---F---G master 29' 30. git-sh-setup 31 32RESOLVEMSG=" 33When you have resolved this problem run\"git rebase --continue\". 34If you would prefer to skip this patch, instead run\"git rebase --skip\". 35To restore the original branch and stop rebasing run\"git rebase --abort\". 36" 37unset newbase 38strategy=recursive 39do_merge= 40dotest=$GIT_DIR/.dotest-merge 41prec=4 42verbose= 43 44continue_merge () { 45test -n"$prev_head"|| die "prev_head must be defined" 46test -d"$dotest"|| die "$dotestdirectory does not exist" 47 48 unmerged=$(git-ls-files -u) 49iftest -n"$unmerged" 50then 51echo"You still have unmerged paths in your index" 52echo"did you forget update-index?" 53 die "$RESOLVEMSG" 54fi 55 56iftest -n"`git-diff-index HEAD`" 57then 58if! git-commit -C"`cat$dotest/current`" 59then 60echo"Commit failed, please do not call\"git commit\"" 61echo"directly, but instead do one of the following: " 62 die "$RESOLVEMSG" 63fi 64printf"Committed: %0${prec}d"$msgnum 65else 66printf"Already applied: %0${prec}d"$msgnum 67fi 68echo' '`git-rev-list --pretty=oneline -1 HEAD | \ 69 sed 's/^[a-f0-9]\+ //'` 70 71 prev_head=`git-rev-parse HEAD^0` 72# save the resulting commit so we can read-tree on it later 73echo"$prev_head">"$dotest/prev_head" 74 75# onto the next patch: 76 msgnum=$(($msgnum + 1)) 77echo"$msgnum">"$dotest/msgnum" 78} 79 80call_merge () { 81 cmt="$(cat $dotest/cmt.$1)" 82echo"$cmt">"$dotest/current" 83 git-merge-$strategy"$cmt^"-- HEAD "$cmt" 84 rv=$? 85case"$rv"in 860) 87return 88;; 891) 90test -d"$GIT_DIR/rr-cache"&& git-rerere 91 die "$RESOLVEMSG" 92;; 932) 94echo"Strategy:$rv$strategyfailed, try another"1>&2 95 die "$RESOLVEMSG" 96;; 97*) 98 die "Unknown exit code ($rv) from command:" \ 99"git-merge-$strategy$cmt^ -- HEAD$cmt" 100;; 101esac 102} 103 104finish_rb_merge () { 105rm-r"$dotest" 106echo"All done." 107} 108 109while case"$#"in0)break;;esac 110do 111case"$1"in 112--continue) 113diff=$(git-diff-files) 114case"$diff"in 115 ?*)echo"You must edit all merge conflicts and then" 116echo"mark them as resolved using git update-index" 117exit1 118;; 119esac 120iftest -d"$dotest" 121then 122 prev_head="`cat$dotest/prev_head`" 123 end="`cat$dotest/end`" 124 msgnum="`cat$dotest/msgnum`" 125 onto="`cat$dotest/onto`" 126 continue_merge 127whiletest"$msgnum"-le"$end" 128do 129 call_merge "$msgnum" 130 continue_merge 131done 132 finish_rb_merge 133exit 134fi 135 git am --resolved --3way --resolvemsg="$RESOLVEMSG" \ 136--reflog-action=rebase 137exit 138;; 139--skip) 140iftest -d"$dotest" 141then 142iftest -d"$GIT_DIR/rr-cache" 143then 144 git-rerereclear 145fi 146 prev_head="`cat$dotest/prev_head`" 147 end="`cat$dotest/end`" 148 msgnum="`cat$dotest/msgnum`" 149 msgnum=$(($msgnum + 1)) 150 onto="`cat$dotest/onto`" 151whiletest"$msgnum"-le"$end" 152do 153 call_merge "$msgnum" 154 continue_merge 155done 156 finish_rb_merge 157exit 158fi 159 git am -3 --skip --resolvemsg="$RESOLVEMSG" \ 160--reflog-action=rebase 161exit 162;; 163--abort) 164iftest -d"$GIT_DIR/rr-cache" 165then 166 git-rerereclear 167fi 168iftest -d"$dotest" 169then 170rm-r"$dotest" 171eliftest -d .dotest 172then 173rm-r .dotest 174else 175 die "No rebase in progress?" 176fi 177 git reset--hard ORIG_HEAD 178exit 179;; 180--onto) 181test2-le"$#"|| usage 182 newbase="$2" 183shift 184;; 185-M|-m|--m|--me|--mer|--merg|--merge) 186 do_merge=t 187;; 188-s=*|--s=*|--st=*|--str=*|--stra=*|--strat=*|--strate=*|\ 189--strateg=*|--strategy=*|\ 190-s|--s|--st|--str|--stra|--strat|--strate|--strateg|--strategy) 191case"$#,$1"in 192*,*=*) 193 strategy=`expr "z$1" : 'z-[^=]*=\(.*\)'`;; 1941,*) 195 usage ;; 196*) 197 strategy="$2" 198shift;; 199esac 200 do_merge=t 201;; 202-v|--verbose) 203 verbose=t 204;; 205-*) 206 usage 207;; 208*) 209break 210;; 211esac 212shift 213done 214 215# Make sure we do not have .dotest 216iftest -z"$do_merge" 217then 218ifmkdir .dotest 219then 220rmdir .dotest 221else 222echo>&2' 223It seems that I cannot create a .dotest directory, and I wonder if you 224are in the middle of patch application or another rebase. If that is not 225the case, please rm -fr .dotest and run me again. I am stopping in case 226you still have something valuable there.' 227exit1 228fi 229else 230iftest -d"$dotest" 231then 232 die "previous dotest directory$doteststill exists." \ 233'try git-rebase < --continue | --abort >' 234fi 235fi 236 237# The tree must be really really clean. 238git-update-index --refresh||exit 239diff=$(git-diff-index --cached --name-status -r HEAD) 240case"$diff"in 241?*)echo"$diff" 242exit1 243;; 244esac 245 246# The upstream head must be given. Make sure it is valid. 247upstream_name="$1" 248upstream=`git rev-parse --verify "${upstream_name}^0"`|| 249 die "invalid upstream$upstream_name" 250 251# If a hook exists, give it a chance to interrupt 252iftest -x"$GIT_DIR/hooks/pre-rebase" 253then 254"$GIT_DIR/hooks/pre-rebase"${1+"$@"}|| { 255echo>&2"The pre-rebase hook refused to rebase." 256exit1 257} 258fi 259 260# If the branch to rebase is given, first switch to it. 261case"$#"in 2622) 263 branch_name="$2" 264 git-checkout"$2"|| usage 265;; 266*) 267 branch_name=`git symbolic-ref HEAD`|| die "No current branch" 268 branch_name=`expr "z$branch_name" : 'zrefs/heads/\(.*\)'` 269;; 270esac 271branch=$(git-rev-parse --verify "${branch_name}^0")||exit 272 273# Make sure the branch to rebase onto is valid. 274onto_name=${newbase-"$upstream_name"} 275onto=$(git-rev-parse --verify "${onto_name}^0")||exit 276 277# Now we are rebasing commits $upstream..$branch on top of $onto 278 279# Check if we are already based on $onto, but this should be 280# done only when upstream and onto are the same. 281mb=$(git-merge-base "$onto" "$branch") 282iftest"$upstream"="$onto"&&test"$mb"="$onto" 283then 284echo>&2"Current branch$branch_nameis up to date." 285exit0 286fi 287 288iftest -n"$verbose" 289then 290echo"Changes from$mbto$onto:" 291 git-diff-tree --stat --summary"$mb""$onto" 292fi 293 294# Rewind the head to "$onto"; this saves our current head in ORIG_HEAD. 295echo"First, rewinding head to replay your work on top of it..." 296git-reset --hard"$onto" 297 298# If the $onto is a proper descendant of the tip of the branch, then 299# we just fast forwarded. 300iftest"$mb"="$branch" 301then 302echo>&2"Fast-forwarded$branch_nameto$onto_name." 303exit0 304fi 305 306iftest -z"$do_merge" 307then 308 git-format-patch -k --stdout --full-index --ignore-if-in-upstream"$upstream"..ORIG_HEAD | 309 git am --binary -3 -k --resolvemsg="$RESOLVEMSG" \ 310--reflog-action=rebase 311exit $? 312fi 313 314# start doing a rebase with git-merge 315# this is rename-aware if the recursive (default) strategy is used 316 317mkdir-p"$dotest" 318echo"$onto">"$dotest/onto" 319prev_head=`git-rev-parse HEAD^0` 320echo"$prev_head">"$dotest/prev_head" 321 322msgnum=0 323for cmt in`git-rev-list --no-merges "$upstream"..ORIG_HEAD \ 324 | @@PERL@@ -e 'print reverse <>'` 325do 326 msgnum=$(($msgnum + 1)) 327echo"$cmt">"$dotest/cmt.$msgnum" 328done 329 330echo1>"$dotest/msgnum" 331echo$msgnum>"$dotest/end" 332 333end=$msgnum 334msgnum=1 335 336whiletest"$msgnum"-le"$end" 337do 338 call_merge "$msgnum" 339 continue_merge 340done 341 342finish_rb_merge