Skip to content

Commit

Permalink
rebase: apply and cleanup autostash when rebase fails to start
Browse files Browse the repository at this point in the history
If "git rebase" fails to start after stashing the user's uncommitted
changes then it fails to restore the stashed changes and remove state
directory.  To make matters worse running "git rebase --abort" to try
and cleanup the state directory fails because the state directory only
contains the "autostash" file and is missing the "head-name" and "onto"
required by read_basic_state().

Fix this by cleaning up the autostash if the pre-rebase hook or initial
checkout fail. New tests are added to check the autostash is applied and
the state directory is removed if the rebase fails to start. Checks are
also added to ensure there is no state directory left behind when a
rebase fails to start and no autostash has been created.

Reported-by: Brian Lyles <[email protected]>
Signed-off-by: Phillip Wood <[email protected]>
  • Loading branch information
phillipwood committed Aug 13, 2024
1 parent 25673b1 commit 5be505c
Show file tree
Hide file tree
Showing 4 changed files with 87 additions and 8 deletions.
36 changes: 30 additions & 6 deletions builtin/rebase.c
Original file line number Diff line number Diff line change
Expand Up @@ -526,6 +526,22 @@ static int rebase_write_basic_state(struct rebase_options *opts)
return 0;
}

static int cleanup_autostash(struct rebase_options *opts)
{
int ret;
struct strbuf dir = STRBUF_INIT;
const char *path = state_dir_path("autostash", opts);

if (!file_exists(path))
return 0;
ret = apply_autostash(path);
strbuf_addstr(&dir, opts->state_dir);
if (remove_dir_recursively(&dir, 0))
return error_errno(_("could not remove '%s'"), opts->state_dir);

return ret;
}

static int finish_rebase(struct rebase_options *opts)
{
struct strbuf dir = STRBUF_INIT;
Expand Down Expand Up @@ -1726,7 +1742,7 @@ int cmd_rebase(int argc, const char **argv, const char *prefix)
if (require_clean_work_tree(the_repository, "rebase",
_("Please commit or stash them."), 1, 1)) {
ret = -1;
goto cleanup;
goto cleanup_autostash;
}

/*
Expand All @@ -1749,7 +1765,7 @@ int cmd_rebase(int argc, const char **argv, const char *prefix)
if (options.switch_to) {
ret = checkout_up_to_date(&options);
if (ret)
goto cleanup;
goto cleanup_autostash;
}

if (!(options.flags & REBASE_NO_QUIET))
Expand All @@ -1775,8 +1791,10 @@ int cmd_rebase(int argc, const char **argv, const char *prefix)
/* If a hook exists, give it a chance to interrupt*/
if (!ok_to_skip_pre_rebase &&
run_hooks_l("pre-rebase", options.upstream_arg,
argc ? argv[0] : NULL, NULL))
die(_("The pre-rebase hook refused to rebase."));
argc ? argv[0] : NULL, NULL)) {
ret = error(_("The pre-rebase hook refused to rebase."));
goto cleanup_autostash;
}

if (options.flags & REBASE_DIFFSTAT) {
struct diff_options opts;
Expand Down Expand Up @@ -1821,8 +1839,10 @@ int cmd_rebase(int argc, const char **argv, const char *prefix)
RESET_HEAD_RUN_POST_CHECKOUT_HOOK;
ropts.head_msg = msg.buf;
ropts.default_reflog_action = options.reflog_action;
if (reset_head(the_repository, &ropts))
die(_("Could not detach HEAD"));
if (reset_head(the_repository, &ropts)) {
ret = error(_("Could not detach HEAD"));
goto cleanup_autostash;
}
strbuf_release(&msg);

/*
Expand Down Expand Up @@ -1856,4 +1876,8 @@ int cmd_rebase(int argc, const char **argv, const char *prefix)
free(squash_onto_name);
free(keep_base_onto_name);
return !!ret;

cleanup_autostash:
ret |= !!cleanup_autostash(&options);
goto cleanup;
}
15 changes: 14 additions & 1 deletion t/t3400-rebase.sh
Original file line number Diff line number Diff line change
Expand Up @@ -422,7 +422,9 @@ test_expect_success 'refuse to switch to branch checked out elsewhere' '
git checkout main &&
git worktree add wt &&
test_must_fail git -C wt rebase main main 2>err &&
test_grep "already used by worktree at" err
test_grep "already used by worktree at" err &&
test_must_fail git -C wt rebase --quit 2>err &&
test_grep "no rebase in progress" err
'

test_expect_success 'rebase when inside worktree subdirectory' '
Expand All @@ -446,4 +448,15 @@ test_expect_success 'rebase when inside worktree subdirectory' '
)
'

test_expect_success 'rebase --apply when initial checkout fails' '
git checkout main &&
mkdir D &&
echo untracked >D/A &&
test_when_finished "rm D/A" &&
test_must_fail git rebase --apply filemove 2>err &&
test_grep "Could not detach HEAD" err &&
test_must_fail git rebase --quit 2>err &&
test_grep "no rebase in progress" err
'

test_done
4 changes: 3 additions & 1 deletion t/t3413-rebase-hook.sh
Original file line number Diff line number Diff line change
Expand Up @@ -110,7 +110,9 @@ test_expect_success 'pre-rebase hook stops rebase (1)' '
git reset --hard side &&
test_must_fail git rebase main &&
test "z$(git symbolic-ref HEAD)" = zrefs/heads/test &&
test 0 = $(git rev-list HEAD...side | wc -l)
test 0 = $(git rev-list HEAD...side | wc -l) &&
test_must_fail git rebase --quit 2>err &&
test_grep "no rebase in progress" err
'

test_expect_success 'pre-rebase hook stops rebase (2)' '
Expand Down
40 changes: 40 additions & 0 deletions t/t3420-rebase-autostash.sh
Original file line number Diff line number Diff line change
Expand Up @@ -82,6 +82,46 @@ testrebase () {
type=$1
dotest=$2

test_expect_success "rebase$type: restore autostash when pre-rebase hook fails" '
git checkout -f feature-branch &&
test_hook pre-rebase <<-\EOF &&
exit 1
EOF
echo changed >file0 &&
test_must_fail git rebase $type --autostash -f HEAD^ &&
test_must_fail git rebase --quit 2>err &&
test_grep "no rebase in progress" err &&
echo changed >expect &&
test_cmp expect file0
'

test_expect_success "rebase$type: restore autostash when checkout onto fails" '
git checkout -f --detach feature-branch &&
echo uncommitted-content >file0 &&
echo untracked >file4 &&
test_when_finished "rm file4" &&
test_must_fail git rebase $type --autostash \
unrelated-onto-branch &&
test_must_fail git rebase --quit 2>err &&
test_grep "no rebase in progress" err &&
echo uncommitted-content >expect &&
test_cmp expect file0
'

test_expect_success "rebase$type: restore autostash when branch checkout fails" '
git checkout -f unrelated-onto-branch^ &&
echo uncommitted-content >file0 &&
echo untracked >file4 &&
test_when_finished "rm file4" &&
test_must_fail git rebase $type --autostash HEAD \
unrelated-onto-branch &&
test_must_fail git rebase --quit 2>err &&
test_grep "no rebase in progress" err &&
echo uncommitted-content >expect &&
test_cmp expect file0
'

test_expect_success "rebase$type: dirty worktree, --no-autostash" '
test_config rebase.autostash true &&
git reset --hard &&
Expand Down

0 comments on commit 5be505c

Please sign in to comment.