git-rebase.shon commit parse-options: multi-word argh should use dash to separate words (e703d71)
   1#!/bin/sh
   2#
   3# Copyright (c) 2005 Junio C Hamano.
   4#
   5
   6SUBDIRECTORY_OK=Yes
   7OPTIONS_KEEPDASHDASH=
   8OPTIONS_STUCKLONG=t
   9OPTIONS_SPEC="\
  10git rebase [-i] [options] [--exec <cmd>] [--onto <newbase>] [<upstream>] [<branch>]
  11git rebase [-i] [options] [--exec <cmd>] [--onto <newbase>] --root [<branch>]
  12git-rebase --continue | --abort | --skip | --edit-todo
  13--
  14 Available options are
  15v,verbose!         display a diffstat of what changed upstream
  16q,quiet!           be quiet. implies --no-stat
  17autostash!         automatically stash/stash pop before and after
  18fork-point         use 'merge-base --fork-point' to refine upstream
  19onto=!             rebase onto given branch instead of upstream
  20p,preserve-merges! try to recreate merges instead of ignoring them
  21s,strategy=!       use the given merge strategy
  22no-ff!             cherry-pick all commits, even if unchanged
  23m,merge!           use merging strategies to rebase
  24i,interactive!     let the user edit the list of commits to rebase
  25x,exec=!           add exec lines after each commit of the editable list
  26k,keep-empty       preserve empty commits during rebase
  27f,force-rebase!    force rebase even if branch is up to date
  28X,strategy-option=! pass the argument through to the merge strategy
  29stat!              display a diffstat of what changed upstream
  30n,no-stat!         do not show diffstat of what changed upstream
  31verify             allow pre-rebase hook to run
  32rerere-autoupdate  allow rerere to update index with resolved conflicts
  33root!              rebase all reachable commits up to the root(s)
  34autosquash         move commits that begin with squash!/fixup! under -i
  35committer-date-is-author-date! passed to 'git am'
  36ignore-date!       passed to 'git am'
  37whitespace=!       passed to 'git apply'
  38ignore-whitespace! passed to 'git apply'
  39C=!                passed to 'git apply'
  40S,gpg-sign?        GPG-sign commits
  41 Actions:
  42continue!          continue
  43abort!             abort and check out the original branch
  44skip!              skip current patch and continue
  45edit-todo!         edit the todo list during an interactive rebase
  46"
  47. git-sh-setup
  48. git-sh-i18n
  49set_reflog_action rebase
  50require_work_tree_exists
  51cd_to_toplevel
  52
  53LF='
  54'
  55ok_to_skip_pre_rebase=
  56resolvemsg="
  57$(gettext 'When you have resolved this problem, run "git rebase --continue".
  58If you prefer to skip this patch, run "git rebase --skip" instead.
  59To check out the original branch and stop rebasing, run "git rebase --abort".')
  60"
  61unset onto
  62cmd=
  63strategy=
  64strategy_opts=
  65do_merge=
  66merge_dir="$GIT_DIR"/rebase-merge
  67apply_dir="$GIT_DIR"/rebase-apply
  68verbose=
  69diffstat=
  70test "$(git config --bool rebase.stat)" = true && diffstat=t
  71autostash="$(git config --bool rebase.autostash || echo false)"
  72fork_point=auto
  73git_am_opt=
  74rebase_root=
  75force_rebase=
  76allow_rerere_autoupdate=
  77# Non-empty if a rebase was in progress when 'git rebase' was invoked
  78in_progress=
  79# One of {am, merge, interactive}
  80type=
  81# One of {"$GIT_DIR"/rebase-apply, "$GIT_DIR"/rebase-merge}
  82state_dir=
  83# One of {'', continue, skip, abort}, as parsed from command line
  84action=
  85preserve_merges=
  86autosquash=
  87keep_empty=
  88test "$(git config --bool rebase.autosquash)" = "true" && autosquash=t
  89gpg_sign_opt=
  90
  91read_basic_state () {
  92        test -f "$state_dir/head-name" &&
  93        test -f "$state_dir/onto" &&
  94        head_name=$(cat "$state_dir"/head-name) &&
  95        onto=$(cat "$state_dir"/onto) &&
  96        # We always write to orig-head, but interactive rebase used to write to
  97        # head. Fall back to reading from head to cover for the case that the
  98        # user upgraded git with an ongoing interactive rebase.
  99        if test -f "$state_dir"/orig-head
 100        then
 101                orig_head=$(cat "$state_dir"/orig-head)
 102        else
 103                orig_head=$(cat "$state_dir"/head)
 104        fi &&
 105        GIT_QUIET=$(cat "$state_dir"/quiet) &&
 106        test -f "$state_dir"/verbose && verbose=t
 107        test -f "$state_dir"/strategy && strategy="$(cat "$state_dir"/strategy)"
 108        test -f "$state_dir"/strategy_opts &&
 109                strategy_opts="$(cat "$state_dir"/strategy_opts)"
 110        test -f "$state_dir"/allow_rerere_autoupdate &&
 111                allow_rerere_autoupdate="$(cat "$state_dir"/allow_rerere_autoupdate)"
 112        test -f "$state_dir"/gpg_sign_opt &&
 113                gpg_sign_opt="$(cat "$state_dir"/gpg_sign_opt)"
 114}
 115
 116write_basic_state () {
 117        echo "$head_name" > "$state_dir"/head-name &&
 118        echo "$onto" > "$state_dir"/onto &&
 119        echo "$orig_head" > "$state_dir"/orig-head &&
 120        echo "$GIT_QUIET" > "$state_dir"/quiet &&
 121        test t = "$verbose" && : > "$state_dir"/verbose
 122        test -n "$strategy" && echo "$strategy" > "$state_dir"/strategy
 123        test -n "$strategy_opts" && echo "$strategy_opts" > \
 124                "$state_dir"/strategy_opts
 125        test -n "$allow_rerere_autoupdate" && echo "$allow_rerere_autoupdate" > \
 126                "$state_dir"/allow_rerere_autoupdate
 127        test -n "$gpg_sign_opt" && echo "$gpg_sign_opt" > "$state_dir"/gpg_sign_opt
 128}
 129
 130output () {
 131        case "$verbose" in
 132        '')
 133                output=$("$@" 2>&1 )
 134                status=$?
 135                test $status != 0 && printf "%s\n" "$output"
 136                return $status
 137                ;;
 138        *)
 139                "$@"
 140                ;;
 141        esac
 142}
 143
 144move_to_original_branch () {
 145        case "$head_name" in
 146        refs/*)
 147                message="rebase finished: $head_name onto $onto"
 148                git update-ref -m "$message" \
 149                        $head_name $(git rev-parse HEAD) $orig_head &&
 150                git symbolic-ref \
 151                        -m "rebase finished: returning to $head_name" \
 152                        HEAD $head_name ||
 153                die "$(gettext "Could not move back to $head_name")"
 154                ;;
 155        esac
 156}
 157
 158finish_rebase () {
 159        if test -f "$state_dir/autostash"
 160        then
 161                stash_sha1=$(cat "$state_dir/autostash")
 162                if git stash apply $stash_sha1 2>&1 >/dev/null
 163                then
 164                        echo "$(gettext 'Applied autostash.')"
 165                else
 166                        git stash store -m "autostash" -q $stash_sha1 ||
 167                        die "$(eval_gettext "Cannot store \$stash_sha1")"
 168                        gettext 'Applying autostash resulted in conflicts.
 169Your changes are safe in the stash.
 170You can run "git stash pop" or "git stash drop" at any time.
 171'
 172                fi
 173        fi
 174        git gc --auto &&
 175        rm -rf "$state_dir"
 176}
 177
 178run_specific_rebase_internal () {
 179        if [ "$interactive_rebase" = implied ]; then
 180                GIT_EDITOR=:
 181                export GIT_EDITOR
 182                autosquash=
 183        fi
 184        # On FreeBSD, the shell's "return" returns from the current
 185        # function, not from the current file inclusion.
 186        # run_specific_rebase_internal has the file inclusion as a
 187        # last statement, so POSIX and FreeBSD's return will do the
 188        # same thing.
 189        . git-rebase--$type
 190}
 191
 192run_specific_rebase () {
 193        run_specific_rebase_internal
 194        ret=$?
 195        if test $ret -eq 0
 196        then
 197                finish_rebase
 198        fi
 199        exit $ret
 200}
 201
 202run_pre_rebase_hook () {
 203        if test -z "$ok_to_skip_pre_rebase" &&
 204           test -x "$GIT_DIR/hooks/pre-rebase"
 205        then
 206                "$GIT_DIR/hooks/pre-rebase" ${1+"$@"} ||
 207                die "$(gettext "The pre-rebase hook refused to rebase.")"
 208        fi
 209}
 210
 211test -f "$apply_dir"/applying &&
 212        die "$(gettext "It looks like git-am is in progress. Cannot rebase.")"
 213
 214if test -d "$apply_dir"
 215then
 216        type=am
 217        state_dir="$apply_dir"
 218elif test -d "$merge_dir"
 219then
 220        if test -f "$merge_dir"/interactive
 221        then
 222                type=interactive
 223                interactive_rebase=explicit
 224        else
 225                type=merge
 226        fi
 227        state_dir="$merge_dir"
 228fi
 229test -n "$type" && in_progress=t
 230
 231total_argc=$#
 232while test $# != 0
 233do
 234        case "$1" in
 235        --no-verify)
 236                ok_to_skip_pre_rebase=yes
 237                ;;
 238        --verify)
 239                ok_to_skip_pre_rebase=
 240                ;;
 241        --continue|--skip|--abort|--edit-todo)
 242                test $total_argc -eq 2 || usage
 243                action=${1##--}
 244                ;;
 245        --onto=*)
 246                onto="${1#--onto=}"
 247                ;;
 248        --exec=*)
 249                cmd="${cmd}exec ${1#--exec=}${LF}"
 250                ;;
 251        --interactive)
 252                interactive_rebase=explicit
 253                ;;
 254        --keep-empty)
 255                keep_empty=yes
 256                ;;
 257        --preserve-merges)
 258                preserve_merges=t
 259                test -z "$interactive_rebase" && interactive_rebase=implied
 260                ;;
 261        --autosquash)
 262                autosquash=t
 263                ;;
 264        --no-autosquash)
 265                autosquash=
 266                ;;
 267        --fork-point)
 268                fork_point=t
 269                ;;
 270        --no-fork-point)
 271                fork_point=
 272                ;;
 273        --merge)
 274                do_merge=t
 275                ;;
 276        --strategy-option=*)
 277                strategy_opts="$strategy_opts $(git rev-parse --sq-quote "--${1#--strategy-option=}")"
 278                do_merge=t
 279                test -z "$strategy" && strategy=recursive
 280                ;;
 281        --strategy=*)
 282                strategy="${1#--strategy=}"
 283                do_merge=t
 284                ;;
 285        --no-stat)
 286                diffstat=
 287                ;;
 288        --stat)
 289                diffstat=t
 290                ;;
 291        --autostash)
 292                autostash=true
 293                ;;
 294        --verbose)
 295                verbose=t
 296                diffstat=t
 297                GIT_QUIET=
 298                ;;
 299        --quiet)
 300                GIT_QUIET=t
 301                git_am_opt="$git_am_opt -q"
 302                verbose=
 303                diffstat=
 304                ;;
 305        --whitespace=*)
 306                git_am_opt="$git_am_opt --whitespace=${1#--whitespace=}"
 307                case "${1#--whitespace=}" in
 308                fix|strip)
 309                        force_rebase=t
 310                        ;;
 311                esac
 312                ;;
 313        --ignore-whitespace)
 314                git_am_opt="$git_am_opt $1"
 315                ;;
 316        --committer-date-is-author-date|--ignore-date)
 317                git_am_opt="$git_am_opt $1"
 318                force_rebase=t
 319                ;;
 320        -C*)
 321                git_am_opt="$git_am_opt $1"
 322                ;;
 323        --root)
 324                rebase_root=t
 325                ;;
 326        --force-rebase|--no-ff)
 327                force_rebase=t
 328                ;;
 329        --rerere-autoupdate|--no-rerere-autoupdate)
 330                allow_rerere_autoupdate="$1"
 331                ;;
 332        --gpg-sign)
 333                gpg_sign_opt=-S
 334                ;;
 335        --gpg-sign=*)
 336                gpg_sign_opt="-S${1#--gpg-sign=}"
 337                ;;
 338        --)
 339                shift
 340                break
 341                ;;
 342        esac
 343        shift
 344done
 345test $# -gt 2 && usage
 346
 347if test -n "$cmd" &&
 348   test "$interactive_rebase" != explicit
 349then
 350        die "$(gettext "The --exec option must be used with the --interactive option")"
 351fi
 352
 353if test -n "$action"
 354then
 355        test -z "$in_progress" && die "$(gettext "No rebase in progress?")"
 356        # Only interactive rebase uses detailed reflog messages
 357        if test "$type" = interactive && test "$GIT_REFLOG_ACTION" = rebase
 358        then
 359                GIT_REFLOG_ACTION="rebase -i ($action)"
 360                export GIT_REFLOG_ACTION
 361        fi
 362fi
 363
 364if test "$action" = "edit-todo" && test "$type" != "interactive"
 365then
 366        die "$(gettext "The --edit-todo action can only be used during interactive rebase.")"
 367fi
 368
 369case "$action" in
 370continue)
 371        # Sanity check
 372        git rev-parse --verify HEAD >/dev/null ||
 373                die "$(gettext "Cannot read HEAD")"
 374        git update-index --ignore-submodules --refresh &&
 375        git diff-files --quiet --ignore-submodules || {
 376                echo "$(gettext "You must edit all merge conflicts and then
 377mark them as resolved using git add")"
 378                exit 1
 379        }
 380        read_basic_state
 381        run_specific_rebase
 382        ;;
 383skip)
 384        output git reset --hard HEAD || exit $?
 385        read_basic_state
 386        run_specific_rebase
 387        ;;
 388abort)
 389        git rerere clear
 390        read_basic_state
 391        case "$head_name" in
 392        refs/*)
 393                git symbolic-ref -m "rebase: aborting" HEAD $head_name ||
 394                die "$(eval_gettext "Could not move back to \$head_name")"
 395                ;;
 396        esac
 397        output git reset --hard $orig_head
 398        finish_rebase
 399        exit
 400        ;;
 401edit-todo)
 402        run_specific_rebase
 403        ;;
 404esac
 405
 406# Make sure no rebase is in progress
 407if test -n "$in_progress"
 408then
 409        state_dir_base=${state_dir##*/}
 410        cmd_live_rebase="git rebase (--continue | --abort | --skip)"
 411        cmd_clear_stale_rebase="rm -fr \"$state_dir\""
 412        die "
 413$(eval_gettext 'It seems that there is already a $state_dir_base directory, and
 414I wonder if you are in the middle of another rebase.  If that is the
 415case, please try
 416        $cmd_live_rebase
 417If that is not the case, please
 418        $cmd_clear_stale_rebase
 419and run me again.  I am stopping in case you still have something
 420valuable there.')"
 421fi
 422
 423if test -n "$rebase_root" && test -z "$onto"
 424then
 425        test -z "$interactive_rebase" && interactive_rebase=implied
 426fi
 427
 428if test -n "$interactive_rebase"
 429then
 430        type=interactive
 431        state_dir="$merge_dir"
 432elif test -n "$do_merge"
 433then
 434        type=merge
 435        state_dir="$merge_dir"
 436else
 437        type=am
 438        state_dir="$apply_dir"
 439fi
 440
 441if test -z "$rebase_root"
 442then
 443        case "$#" in
 444        0)
 445                if ! upstream_name=$(git rev-parse --symbolic-full-name \
 446                        --verify -q @{upstream} 2>/dev/null)
 447                then
 448                        . git-parse-remote
 449                        error_on_missing_default_upstream "rebase" "rebase" \
 450                                "against" "git rebase <branch>"
 451                fi
 452
 453                test "$fork_point" = auto && fork_point=t
 454                ;;
 455        *)      upstream_name="$1"
 456                shift
 457                ;;
 458        esac
 459        upstream=$(peel_committish "${upstream_name}") ||
 460        die "$(eval_gettext "invalid upstream \$upstream_name")"
 461        upstream_arg="$upstream_name"
 462else
 463        if test -z "$onto"
 464        then
 465                empty_tree=`git hash-object -t tree /dev/null`
 466                onto=`git commit-tree $empty_tree </dev/null`
 467                squash_onto="$onto"
 468        fi
 469        unset upstream_name
 470        unset upstream
 471        test $# -gt 1 && usage
 472        upstream_arg=--root
 473fi
 474
 475# Make sure the branch to rebase onto is valid.
 476onto_name=${onto-"$upstream_name"}
 477case "$onto_name" in
 478*...*)
 479        if      left=${onto_name%...*} right=${onto_name#*...} &&
 480                onto=$(git merge-base --all ${left:-HEAD} ${right:-HEAD})
 481        then
 482                case "$onto" in
 483                ?*"$LF"?*)
 484                        die "$(eval_gettext "\$onto_name: there are more than one merge bases")"
 485                        ;;
 486                '')
 487                        die "$(eval_gettext "\$onto_name: there is no merge base")"
 488                        ;;
 489                esac
 490        else
 491                die "$(eval_gettext "\$onto_name: there is no merge base")"
 492        fi
 493        ;;
 494*)
 495        onto=$(peel_committish "$onto_name") ||
 496        die "$(eval_gettext "Does not point to a valid commit: \$onto_name")"
 497        ;;
 498esac
 499
 500# If the branch to rebase is given, that is the branch we will rebase
 501# $branch_name -- branch being rebased, or HEAD (already detached)
 502# $orig_head -- commit object name of tip of the branch before rebasing
 503# $head_name -- refs/heads/<that-branch> or "detached HEAD"
 504switch_to=
 505case "$#" in
 5061)
 507        # Is it "rebase other $branchname" or "rebase other $commit"?
 508        branch_name="$1"
 509        switch_to="$1"
 510
 511        if git show-ref --verify --quiet -- "refs/heads/$1" &&
 512           orig_head=$(git rev-parse -q --verify "refs/heads/$1")
 513        then
 514                head_name="refs/heads/$1"
 515        elif orig_head=$(git rev-parse -q --verify "$1")
 516        then
 517                head_name="detached HEAD"
 518        else
 519                die "$(eval_gettext "fatal: no such branch: \$branch_name")"
 520        fi
 521        ;;
 5220)
 523        # Do not need to switch branches, we are already on it.
 524        if branch_name=`git symbolic-ref -q HEAD`
 525        then
 526                head_name=$branch_name
 527                branch_name=`expr "z$branch_name" : 'zrefs/heads/\(.*\)'`
 528        else
 529                head_name="detached HEAD"
 530                branch_name=HEAD ;# detached
 531        fi
 532        orig_head=$(git rev-parse --verify HEAD) || exit
 533        ;;
 534*)
 535        die "BUG: unexpected number of arguments left to parse"
 536        ;;
 537esac
 538
 539if test "$fork_point" = t
 540then
 541        new_upstream=$(git merge-base --fork-point "$upstream_name" \
 542                        "${switch_to:-HEAD}")
 543        if test -n "$new_upstream"
 544        then
 545                upstream=$new_upstream
 546        fi
 547fi
 548
 549if test "$autostash" = true && ! (require_clean_work_tree) 2>/dev/null
 550then
 551        stash_sha1=$(git stash create "autostash") ||
 552        die "$(gettext 'Cannot autostash')"
 553
 554        mkdir -p "$state_dir" &&
 555        echo $stash_sha1 >"$state_dir/autostash" &&
 556        stash_abbrev=$(git rev-parse --short $stash_sha1) &&
 557        echo "$(eval_gettext 'Created autostash: $stash_abbrev')" &&
 558        git reset --hard
 559fi
 560
 561require_clean_work_tree "rebase" "$(gettext "Please commit or stash them.")"
 562
 563# Now we are rebasing commits $upstream..$orig_head (or with --root,
 564# everything leading up to $orig_head) on top of $onto
 565
 566# Check if we are already based on $onto with linear history,
 567# but this should be done only when upstream and onto are the same
 568# and if this is not an interactive rebase.
 569mb=$(git merge-base "$onto" "$orig_head")
 570if test "$type" != interactive && test "$upstream" = "$onto" &&
 571        test "$mb" = "$onto" &&
 572        # linear history?
 573        ! (git rev-list --parents "$onto".."$orig_head" | sane_grep " .* ") > /dev/null
 574then
 575        if test -z "$force_rebase"
 576        then
 577                # Lazily switch to the target branch if needed...
 578                test -z "$switch_to" ||
 579                GIT_REFLOG_ACTION="$GIT_REFLOG_ACTION: checkout $switch_to" \
 580                        git checkout "$switch_to" --
 581                say "$(eval_gettext "Current branch \$branch_name is up to date.")"
 582                finish_rebase
 583                exit 0
 584        else
 585                say "$(eval_gettext "Current branch \$branch_name is up to date, rebase forced.")"
 586        fi
 587fi
 588
 589# If a hook exists, give it a chance to interrupt
 590run_pre_rebase_hook "$upstream_arg" "$@"
 591
 592if test -n "$diffstat"
 593then
 594        if test -n "$verbose"
 595        then
 596                echo "$(eval_gettext "Changes from \$mb to \$onto:")"
 597        fi
 598        # We want color (if set), but no pager
 599        GIT_PAGER='' git diff --stat --summary "$mb" "$onto"
 600fi
 601
 602test "$type" = interactive && run_specific_rebase
 603
 604# Detach HEAD and reset the tree
 605say "$(gettext "First, rewinding head to replay your work on top of it...")"
 606
 607GIT_REFLOG_ACTION="$GIT_REFLOG_ACTION: checkout $onto_name" \
 608        git checkout -q "$onto^0" || die "could not detach HEAD"
 609git update-ref ORIG_HEAD $orig_head
 610
 611# If the $onto is a proper descendant of the tip of the branch, then
 612# we just fast-forwarded.
 613if test "$mb" = "$orig_head"
 614then
 615        say "$(eval_gettext "Fast-forwarded \$branch_name to \$onto_name.")"
 616        move_to_original_branch
 617        finish_rebase
 618        exit 0
 619fi
 620
 621if test -n "$rebase_root"
 622then
 623        revisions="$onto..$orig_head"
 624else
 625        revisions="$upstream..$orig_head"
 626fi
 627
 628run_specific_rebase