Merge branch 'rr/rebase-autostash'
* rr/rebase-autostash: rebase: implement --[no-]autostash and rebase.autostash rebase --merge: return control to caller, for housekeeping rebase -i: return control to caller, for housekeeping am: return control to caller, for housekeeping rebase: prepare to do generic housekeeping rebase -i: don't error out if $state_dir already exists am: tighten a conditional that checks for $dotestmaint
commit
45acb75928
|
|
@ -1867,6 +1867,14 @@ rebase.stat::
|
||||||
rebase.autosquash::
|
rebase.autosquash::
|
||||||
If set to true enable '--autosquash' option by default.
|
If set to true enable '--autosquash' option by default.
|
||||||
|
|
||||||
|
rebase.autostash::
|
||||||
|
When set to true, automatically create a temporary stash
|
||||||
|
before the operation begins, and apply it after the operation
|
||||||
|
ends. This means that you can run rebase on a dirty worktree.
|
||||||
|
However, use with care: the final stash application after a
|
||||||
|
successful rebase might result in non-trivial conflicts.
|
||||||
|
Defaults to false.
|
||||||
|
|
||||||
receive.autogc::
|
receive.autogc::
|
||||||
By default, git-receive-pack will run "git-gc --auto" after
|
By default, git-receive-pack will run "git-gc --auto" after
|
||||||
receiving data from git-push and updating refs. You can stop
|
receiving data from git-push and updating refs. You can stop
|
||||||
|
|
|
||||||
|
|
@ -208,6 +208,9 @@ rebase.stat::
|
||||||
rebase.autosquash::
|
rebase.autosquash::
|
||||||
If set to true enable '--autosquash' option by default.
|
If set to true enable '--autosquash' option by default.
|
||||||
|
|
||||||
|
rebase.autostash::
|
||||||
|
If set to true enable '--autostash' option by default.
|
||||||
|
|
||||||
OPTIONS
|
OPTIONS
|
||||||
-------
|
-------
|
||||||
--onto <newbase>::
|
--onto <newbase>::
|
||||||
|
|
@ -394,6 +397,13 @@ If the '--autosquash' option is enabled by default using the
|
||||||
configuration variable `rebase.autosquash`, this option can be
|
configuration variable `rebase.autosquash`, this option can be
|
||||||
used to override and disable this setting.
|
used to override and disable this setting.
|
||||||
|
|
||||||
|
--[no-]autostash::
|
||||||
|
Automatically create a temporary stash before the operation
|
||||||
|
begins, and apply it after the operation ends. This means
|
||||||
|
that you can run rebase on a dirty worktree. However, use
|
||||||
|
with care: the final stash application after a successful
|
||||||
|
rebase might result in non-trivial conflicts.
|
||||||
|
|
||||||
--no-ff::
|
--no-ff::
|
||||||
With --interactive, cherry-pick all rebased commits instead of
|
With --interactive, cherry-pick all rebased commits instead of
|
||||||
fast-forwarding over the unchanged ones. This ensures that the
|
fast-forwarding over the unchanged ones. This ensures that the
|
||||||
|
|
|
||||||
13
git-am.sh
13
git-am.sh
|
|
@ -446,6 +446,8 @@ done
|
||||||
# If the dotest directory exists, but we have finished applying all the
|
# If the dotest directory exists, but we have finished applying all the
|
||||||
# patches in them, clear it out.
|
# patches in them, clear it out.
|
||||||
if test -d "$dotest" &&
|
if test -d "$dotest" &&
|
||||||
|
test -f "$dotest/last" &&
|
||||||
|
test -f "$dotest/next" &&
|
||||||
last=$(cat "$dotest/last") &&
|
last=$(cat "$dotest/last") &&
|
||||||
next=$(cat "$dotest/next") &&
|
next=$(cat "$dotest/next") &&
|
||||||
test $# != 0 &&
|
test $# != 0 &&
|
||||||
|
|
@ -454,7 +456,7 @@ then
|
||||||
rm -fr "$dotest"
|
rm -fr "$dotest"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
if test -d "$dotest"
|
if test -d "$dotest" && test -f "$dotest/last" && test -f "$dotest/next"
|
||||||
then
|
then
|
||||||
case "$#,$skip$resolved$abort" in
|
case "$#,$skip$resolved$abort" in
|
||||||
0,*t*)
|
0,*t*)
|
||||||
|
|
@ -904,5 +906,10 @@ if test -s "$dotest"/rewritten; then
|
||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
rm -fr "$dotest"
|
# If am was called with --rebasing (from git-rebase--am), it's up to
|
||||||
git gc --auto
|
# the caller to take care of housekeeping.
|
||||||
|
if ! test -f "$dotest/rebasing"
|
||||||
|
then
|
||||||
|
rm -fr "$dotest"
|
||||||
|
git gc --auto
|
||||||
|
fi
|
||||||
|
|
|
||||||
|
|
@ -7,12 +7,12 @@ case "$action" in
|
||||||
continue)
|
continue)
|
||||||
git am --resolved --resolvemsg="$resolvemsg" &&
|
git am --resolved --resolvemsg="$resolvemsg" &&
|
||||||
move_to_original_branch
|
move_to_original_branch
|
||||||
exit
|
return
|
||||||
;;
|
;;
|
||||||
skip)
|
skip)
|
||||||
git am --skip --resolvemsg="$resolvemsg" &&
|
git am --skip --resolvemsg="$resolvemsg" &&
|
||||||
move_to_original_branch
|
move_to_original_branch
|
||||||
exit
|
return
|
||||||
;;
|
;;
|
||||||
esac
|
esac
|
||||||
|
|
||||||
|
|
@ -56,7 +56,7 @@ else
|
||||||
|
|
||||||
As a result, git cannot rebase them.
|
As a result, git cannot rebase them.
|
||||||
EOF
|
EOF
|
||||||
exit $?
|
return $?
|
||||||
fi
|
fi
|
||||||
|
|
||||||
git am $git_am_opt --rebasing --resolvemsg="$resolvemsg" <"$GIT_DIR/rebased-patches"
|
git am $git_am_opt --rebasing --resolvemsg="$resolvemsg" <"$GIT_DIR/rebased-patches"
|
||||||
|
|
@ -68,7 +68,7 @@ fi
|
||||||
if test 0 != $ret
|
if test 0 != $ret
|
||||||
then
|
then
|
||||||
test -d "$state_dir" && write_basic_state
|
test -d "$state_dir" && write_basic_state
|
||||||
exit $ret
|
return $ret
|
||||||
fi
|
fi
|
||||||
|
|
||||||
move_to_original_branch
|
move_to_original_branch
|
||||||
|
|
|
||||||
|
|
@ -628,17 +628,16 @@ do_next () {
|
||||||
"$GIT_DIR"/hooks/post-rewrite rebase < "$rewritten_list"
|
"$GIT_DIR"/hooks/post-rewrite rebase < "$rewritten_list"
|
||||||
true # we don't care if this hook failed
|
true # we don't care if this hook failed
|
||||||
fi &&
|
fi &&
|
||||||
rm -rf "$state_dir" &&
|
|
||||||
git gc --auto &&
|
|
||||||
warn "Successfully rebased and updated $head_name."
|
warn "Successfully rebased and updated $head_name."
|
||||||
|
|
||||||
exit
|
return 1 # not failure; just to break the do_rest loop
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# can only return 0, when the infinite loop breaks
|
||||||
do_rest () {
|
do_rest () {
|
||||||
while :
|
while :
|
||||||
do
|
do
|
||||||
do_next
|
do_next || break
|
||||||
done
|
done
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -805,11 +804,13 @@ first and then run 'git rebase --continue' again."
|
||||||
|
|
||||||
require_clean_work_tree "rebase"
|
require_clean_work_tree "rebase"
|
||||||
do_rest
|
do_rest
|
||||||
|
return 0
|
||||||
;;
|
;;
|
||||||
skip)
|
skip)
|
||||||
git rerere clear
|
git rerere clear
|
||||||
|
|
||||||
do_rest
|
do_rest
|
||||||
|
return 0
|
||||||
;;
|
;;
|
||||||
edit-todo)
|
edit-todo)
|
||||||
git stripspace --strip-comments <"$todo" >"$todo".new
|
git stripspace --strip-comments <"$todo" >"$todo".new
|
||||||
|
|
@ -842,7 +843,7 @@ then
|
||||||
fi
|
fi
|
||||||
|
|
||||||
orig_head=$(git rev-parse --verify HEAD) || die "No HEAD?"
|
orig_head=$(git rev-parse --verify HEAD) || die "No HEAD?"
|
||||||
mkdir "$state_dir" || die "Could not create temporary $state_dir"
|
mkdir -p "$state_dir" || die "Could not create temporary $state_dir"
|
||||||
|
|
||||||
: > "$state_dir"/interactive || die "Could not mark as interactive"
|
: > "$state_dir"/interactive || die "Could not mark as interactive"
|
||||||
write_basic_state
|
write_basic_state
|
||||||
|
|
|
||||||
|
|
@ -96,7 +96,6 @@ finish_rb_merge () {
|
||||||
"$GIT_DIR"/hooks/post-rewrite rebase <"$state_dir"/rewritten
|
"$GIT_DIR"/hooks/post-rewrite rebase <"$state_dir"/rewritten
|
||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
rm -r "$state_dir"
|
|
||||||
say All done.
|
say All done.
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -110,7 +109,7 @@ continue)
|
||||||
continue_merge
|
continue_merge
|
||||||
done
|
done
|
||||||
finish_rb_merge
|
finish_rb_merge
|
||||||
exit
|
return
|
||||||
;;
|
;;
|
||||||
skip)
|
skip)
|
||||||
read_state
|
read_state
|
||||||
|
|
@ -122,7 +121,7 @@ skip)
|
||||||
continue_merge
|
continue_merge
|
||||||
done
|
done
|
||||||
finish_rb_merge
|
finish_rb_merge
|
||||||
exit
|
return
|
||||||
;;
|
;;
|
||||||
esac
|
esac
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -13,6 +13,7 @@ git-rebase --continue | --abort | --skip | --edit-todo
|
||||||
Available options are
|
Available options are
|
||||||
v,verbose! display a diffstat of what changed upstream
|
v,verbose! display a diffstat of what changed upstream
|
||||||
q,quiet! be quiet. implies --no-stat
|
q,quiet! be quiet. implies --no-stat
|
||||||
|
autostash! automatically stash/stash pop before and after
|
||||||
onto=! rebase onto given branch instead of upstream
|
onto=! rebase onto given branch instead of upstream
|
||||||
p,preserve-merges! try to recreate merges instead of ignoring them
|
p,preserve-merges! try to recreate merges instead of ignoring them
|
||||||
s,strategy=! use the given merge strategy
|
s,strategy=! use the given merge strategy
|
||||||
|
|
@ -64,6 +65,7 @@ apply_dir="$GIT_DIR"/rebase-apply
|
||||||
verbose=
|
verbose=
|
||||||
diffstat=
|
diffstat=
|
||||||
test "$(git config --bool rebase.stat)" = true && diffstat=t
|
test "$(git config --bool rebase.stat)" = true && diffstat=t
|
||||||
|
autostash="$(git config --bool rebase.autostash || echo false)"
|
||||||
git_am_opt=
|
git_am_opt=
|
||||||
rebase_root=
|
rebase_root=
|
||||||
force_rebase=
|
force_rebase=
|
||||||
|
|
@ -143,6 +145,29 @@ move_to_original_branch () {
|
||||||
esac
|
esac
|
||||||
}
|
}
|
||||||
|
|
||||||
|
finish_rebase () {
|
||||||
|
if test -f "$state_dir/autostash"
|
||||||
|
then
|
||||||
|
stash_sha1=$(cat "$state_dir/autostash")
|
||||||
|
if git stash apply $stash_sha1 2>&1 >/dev/null
|
||||||
|
then
|
||||||
|
echo "$(gettext 'Applied autostash.')"
|
||||||
|
else
|
||||||
|
ref_stash=refs/stash &&
|
||||||
|
>>"$GIT_DIR/logs/$ref_stash" &&
|
||||||
|
git update-ref -m "autostash" $ref_stash $stash_sha1 ||
|
||||||
|
die "$(eval_gettext 'Cannot store $stash_sha1')"
|
||||||
|
|
||||||
|
gettext 'Applying autostash resulted in conflicts.
|
||||||
|
Your changes are safe in the stash.
|
||||||
|
You can run "git stash pop" or "git stash drop" it at any time.
|
||||||
|
'
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
git gc --auto &&
|
||||||
|
rm -rf "$state_dir"
|
||||||
|
}
|
||||||
|
|
||||||
run_specific_rebase () {
|
run_specific_rebase () {
|
||||||
if [ "$interactive_rebase" = implied ]; then
|
if [ "$interactive_rebase" = implied ]; then
|
||||||
GIT_EDITOR=:
|
GIT_EDITOR=:
|
||||||
|
|
@ -150,6 +175,12 @@ run_specific_rebase () {
|
||||||
autosquash=
|
autosquash=
|
||||||
fi
|
fi
|
||||||
. git-rebase--$type
|
. git-rebase--$type
|
||||||
|
ret=$?
|
||||||
|
if test $ret -eq 0
|
||||||
|
then
|
||||||
|
finish_rebase
|
||||||
|
fi
|
||||||
|
exit $ret
|
||||||
}
|
}
|
||||||
|
|
||||||
run_pre_rebase_hook () {
|
run_pre_rebase_hook () {
|
||||||
|
|
@ -241,6 +272,9 @@ do
|
||||||
--stat)
|
--stat)
|
||||||
diffstat=t
|
diffstat=t
|
||||||
;;
|
;;
|
||||||
|
--autostash)
|
||||||
|
autostash=true
|
||||||
|
;;
|
||||||
-v)
|
-v)
|
||||||
verbose=t
|
verbose=t
|
||||||
diffstat=t
|
diffstat=t
|
||||||
|
|
@ -341,7 +375,7 @@ abort)
|
||||||
;;
|
;;
|
||||||
esac
|
esac
|
||||||
output git reset --hard $orig_head
|
output git reset --hard $orig_head
|
||||||
rm -r "$state_dir"
|
finish_rebase
|
||||||
exit
|
exit
|
||||||
;;
|
;;
|
||||||
edit-todo)
|
edit-todo)
|
||||||
|
|
@ -480,6 +514,18 @@ case "$#" in
|
||||||
;;
|
;;
|
||||||
esac
|
esac
|
||||||
|
|
||||||
|
if test "$autostash" = true && ! (require_clean_work_tree) 2>/dev/null
|
||||||
|
then
|
||||||
|
stash_sha1=$(git stash create "autostash") ||
|
||||||
|
die "$(gettext 'Cannot autostash')"
|
||||||
|
|
||||||
|
mkdir -p "$state_dir" &&
|
||||||
|
echo $stash_sha1 >"$state_dir/autostash" &&
|
||||||
|
stash_abbrev=$(git rev-parse --short $stash_sha1) &&
|
||||||
|
echo "$(eval_gettext 'Created autostash: $stash_abbrev')" &&
|
||||||
|
git reset --hard
|
||||||
|
fi
|
||||||
|
|
||||||
require_clean_work_tree "rebase" "$(gettext "Please commit or stash them.")"
|
require_clean_work_tree "rebase" "$(gettext "Please commit or stash them.")"
|
||||||
|
|
||||||
# Now we are rebasing commits $upstream..$orig_head (or with --root,
|
# Now we are rebasing commits $upstream..$orig_head (or with --root,
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,148 @@
|
||||||
|
#!/bin/sh
|
||||||
|
#
|
||||||
|
# Copyright (c) 2013 Ramkumar Ramachandra
|
||||||
|
#
|
||||||
|
|
||||||
|
test_description='git rebase --autostash tests'
|
||||||
|
. ./test-lib.sh
|
||||||
|
|
||||||
|
test_expect_success setup '
|
||||||
|
echo hello-world >file0 &&
|
||||||
|
git add . &&
|
||||||
|
test_tick &&
|
||||||
|
git commit -m "initial commit" &&
|
||||||
|
git checkout -b feature-branch &&
|
||||||
|
echo another-hello >file1 &&
|
||||||
|
echo goodbye >file2 &&
|
||||||
|
git add . &&
|
||||||
|
test_tick &&
|
||||||
|
git commit -m "second commit" &&
|
||||||
|
echo final-goodbye >file3 &&
|
||||||
|
git add . &&
|
||||||
|
test_tick &&
|
||||||
|
git commit -m "third commit" &&
|
||||||
|
git checkout -b unrelated-onto-branch master &&
|
||||||
|
echo unrelated >file4 &&
|
||||||
|
git add . &&
|
||||||
|
test_tick &&
|
||||||
|
git commit -m "unrelated commit" &&
|
||||||
|
git checkout -b related-onto-branch master &&
|
||||||
|
echo conflicting-change >file2 &&
|
||||||
|
git add . &&
|
||||||
|
test_tick &&
|
||||||
|
git commit -m "related commit"
|
||||||
|
'
|
||||||
|
|
||||||
|
testrebase() {
|
||||||
|
type=$1
|
||||||
|
dotest=$2
|
||||||
|
|
||||||
|
test_expect_success "rebase$type: dirty worktree, non-conflicting rebase" '
|
||||||
|
test_config rebase.autostash true &&
|
||||||
|
git reset --hard &&
|
||||||
|
git checkout -b rebased-feature-branch feature-branch &&
|
||||||
|
test_when_finished git branch -D rebased-feature-branch &&
|
||||||
|
echo dirty >>file3 &&
|
||||||
|
git rebase$type unrelated-onto-branch &&
|
||||||
|
grep unrelated file4 &&
|
||||||
|
grep dirty file3 &&
|
||||||
|
git checkout feature-branch
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success "rebase$type: dirty index, non-conflicting rebase" '
|
||||||
|
test_config rebase.autostash true &&
|
||||||
|
git reset --hard &&
|
||||||
|
git checkout -b rebased-feature-branch feature-branch &&
|
||||||
|
test_when_finished git branch -D rebased-feature-branch &&
|
||||||
|
echo dirty >>file3 &&
|
||||||
|
git add file3 &&
|
||||||
|
git rebase$type unrelated-onto-branch &&
|
||||||
|
grep unrelated file4 &&
|
||||||
|
grep dirty file3 &&
|
||||||
|
git checkout feature-branch
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success "rebase$type: conflicting rebase" '
|
||||||
|
test_config rebase.autostash true &&
|
||||||
|
git reset --hard &&
|
||||||
|
git checkout -b rebased-feature-branch feature-branch &&
|
||||||
|
test_when_finished git branch -D rebased-feature-branch &&
|
||||||
|
echo dirty >>file3 &&
|
||||||
|
test_must_fail git rebase$type related-onto-branch &&
|
||||||
|
test_path_is_file $dotest/autostash &&
|
||||||
|
! grep dirty file3 &&
|
||||||
|
rm -rf $dotest &&
|
||||||
|
git reset --hard &&
|
||||||
|
git checkout feature-branch
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success "rebase$type: --continue" '
|
||||||
|
test_config rebase.autostash true &&
|
||||||
|
git reset --hard &&
|
||||||
|
git checkout -b rebased-feature-branch feature-branch &&
|
||||||
|
test_when_finished git branch -D rebased-feature-branch &&
|
||||||
|
echo dirty >>file3 &&
|
||||||
|
test_must_fail git rebase$type related-onto-branch &&
|
||||||
|
test_path_is_file $dotest/autostash &&
|
||||||
|
! grep dirty file3 &&
|
||||||
|
echo "conflicting-plus-goodbye" >file2 &&
|
||||||
|
git add file2 &&
|
||||||
|
git rebase --continue &&
|
||||||
|
test_path_is_missing $dotest/autostash &&
|
||||||
|
grep dirty file3 &&
|
||||||
|
git checkout feature-branch
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success "rebase$type: --skip" '
|
||||||
|
test_config rebase.autostash true &&
|
||||||
|
git reset --hard &&
|
||||||
|
git checkout -b rebased-feature-branch feature-branch &&
|
||||||
|
test_when_finished git branch -D rebased-feature-branch &&
|
||||||
|
echo dirty >>file3 &&
|
||||||
|
test_must_fail git rebase$type related-onto-branch &&
|
||||||
|
test_path_is_file $dotest/autostash &&
|
||||||
|
! grep dirty file3 &&
|
||||||
|
git rebase --skip &&
|
||||||
|
test_path_is_missing $dotest/autostash &&
|
||||||
|
grep dirty file3 &&
|
||||||
|
git checkout feature-branch
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success "rebase$type: --abort" '
|
||||||
|
test_config rebase.autostash true &&
|
||||||
|
git reset --hard &&
|
||||||
|
git checkout -b rebased-feature-branch feature-branch &&
|
||||||
|
test_when_finished git branch -D rebased-feature-branch &&
|
||||||
|
echo dirty >>file3 &&
|
||||||
|
test_must_fail git rebase$type related-onto-branch &&
|
||||||
|
test_path_is_file $dotest/autostash &&
|
||||||
|
! grep dirty file3 &&
|
||||||
|
git rebase --abort &&
|
||||||
|
test_path_is_missing $dotest/autostash &&
|
||||||
|
grep dirty file3 &&
|
||||||
|
git checkout feature-branch
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success "rebase$type: non-conflicting rebase, conflicting stash" '
|
||||||
|
test_config rebase.autostash true &&
|
||||||
|
git reset --hard &&
|
||||||
|
git checkout -b rebased-feature-branch feature-branch &&
|
||||||
|
test_when_finished git branch -D rebased-feature-branch &&
|
||||||
|
echo dirty >file4 &&
|
||||||
|
git add file4 &&
|
||||||
|
git rebase$type unrelated-onto-branch &&
|
||||||
|
test_path_is_missing $dotest &&
|
||||||
|
git reset --hard &&
|
||||||
|
grep unrelated file4 &&
|
||||||
|
! grep dirty file4 &&
|
||||||
|
git checkout feature-branch &&
|
||||||
|
git stash pop &&
|
||||||
|
grep dirty file4
|
||||||
|
'
|
||||||
|
}
|
||||||
|
|
||||||
|
testrebase "" .git/rebase-apply
|
||||||
|
testrebase " --merge" .git/rebase-merge
|
||||||
|
testrebase " --interactive" .git/rebase-merge
|
||||||
|
|
||||||
|
test_done
|
||||||
Loading…
Reference in New Issue