Merge branch 'pw/rebase-abort-clean-rewritten'

"git rebase --abort" used to leave refs/rewritten/ when concluding
"git rebase -r", which has been corrected.

* pw/rebase-abort-clean-rewritten:
  rebase --abort/--quit: cleanup refs/rewritten
  sequencer: return errors from sequencer_remove_state()
  rebase: warn if state directory cannot be removed
  rebase: fix a memory leak
maint
Junio C Hamano 2019-07-09 15:25:41 -07:00
commit 88f95e49af
3 changed files with 53 additions and 15 deletions

View File

@ -738,6 +738,7 @@ static int finish_rebase(struct rebase_options *opts)
{ {
struct strbuf dir = STRBUF_INIT; struct strbuf dir = STRBUF_INIT;
const char *argv_gc_auto[] = { "gc", "--auto", NULL }; const char *argv_gc_auto[] = { "gc", "--auto", NULL };
int ret = 0;


delete_ref(NULL, "REBASE_HEAD", NULL, REF_NO_DEREF); delete_ref(NULL, "REBASE_HEAD", NULL, REF_NO_DEREF);
apply_autostash(opts); apply_autostash(opts);
@ -747,11 +748,20 @@ static int finish_rebase(struct rebase_options *opts)
* user should see them. * user should see them.
*/ */
run_command_v_opt(argv_gc_auto, RUN_GIT_CMD); run_command_v_opt(argv_gc_auto, RUN_GIT_CMD);
strbuf_addstr(&dir, opts->state_dir); if (opts->type == REBASE_INTERACTIVE) {
remove_dir_recursively(&dir, 0); struct replay_opts replay = REPLAY_OPTS_INIT;
strbuf_release(&dir);


return 0; replay.action = REPLAY_INTERACTIVE_REBASE;
ret = sequencer_remove_state(&replay);
} else {
strbuf_addstr(&dir, opts->state_dir);
if (remove_dir_recursively(&dir, 0))
ret = error(_("could not remove '%s'"),
opts->state_dir);
strbuf_release(&dir);
}

return ret;
} }


static struct commit *peel_committish(const char *name) static struct commit *peel_committish(const char *name)
@ -1621,15 +1631,23 @@ int cmd_rebase(int argc, const char **argv, const char *prefix)
die(_("could not move back to %s"), die(_("could not move back to %s"),
oid_to_hex(&options.orig_head)); oid_to_hex(&options.orig_head));
remove_branch_state(the_repository); remove_branch_state(the_repository);
ret = finish_rebase(&options); ret = !!finish_rebase(&options);
goto cleanup; goto cleanup;
} }
case ACTION_QUIT: { case ACTION_QUIT: {
if (options.type == REBASE_INTERACTIVE) {
struct replay_opts replay = REPLAY_OPTS_INIT;

replay.action = REPLAY_INTERACTIVE_REBASE;
ret = !!sequencer_remove_state(&replay);
} else {
strbuf_reset(&buf); strbuf_reset(&buf);
strbuf_addstr(&buf, options.state_dir); strbuf_addstr(&buf, options.state_dir);
ret = !!remove_dir_recursively(&buf, 0); ret = !!remove_dir_recursively(&buf, 0);
if (ret) if (ret)
die(_("could not remove '%s'"), options.state_dir); error(_("could not remove '%s'"),
options.state_dir);
}
goto cleanup; goto cleanup;
} }
case ACTION_EDIT_TODO: case ACTION_EDIT_TODO:
@ -2141,6 +2159,7 @@ run_rebase:
ret = !!run_specific_rebase(&options, action); ret = !!run_specific_rebase(&options, action);


cleanup: cleanup:
strbuf_release(&buf);
strbuf_release(&revisions); strbuf_release(&revisions);
free(options.head_name); free(options.head_name);
free(options.gpg_sign_opt); free(options.gpg_sign_opt);

View File

@ -279,7 +279,7 @@ static const char *gpg_sign_opt_quoted(struct replay_opts *opts)
int sequencer_remove_state(struct replay_opts *opts) int sequencer_remove_state(struct replay_opts *opts)
{ {
struct strbuf buf = STRBUF_INIT; struct strbuf buf = STRBUF_INIT;
int i; int i, ret = 0;


if (is_rebase_i(opts) && if (is_rebase_i(opts) &&
strbuf_read_file(&buf, rebase_path_refs_to_delete(), 0) > 0) { strbuf_read_file(&buf, rebase_path_refs_to_delete(), 0) > 0) {
@ -288,8 +288,10 @@ int sequencer_remove_state(struct replay_opts *opts)
char *eol = strchr(p, '\n'); char *eol = strchr(p, '\n');
if (eol) if (eol)
*eol = '\0'; *eol = '\0';
if (delete_ref("(rebase -i) cleanup", p, NULL, 0) < 0) if (delete_ref("(rebase -i) cleanup", p, NULL, 0) < 0) {
warning(_("could not delete '%s'"), p); warning(_("could not delete '%s'"), p);
ret = -1;
}
if (!eol) if (!eol)
break; break;
p = eol + 1; p = eol + 1;
@ -305,10 +307,11 @@ int sequencer_remove_state(struct replay_opts *opts)


strbuf_reset(&buf); strbuf_reset(&buf);
strbuf_addstr(&buf, get_dir(opts)); strbuf_addstr(&buf, get_dir(opts));
remove_dir_recursively(&buf, 0); if (remove_dir_recursively(&buf, 0))
ret = error(_("could not remove '%s'"), buf.buf);
strbuf_release(&buf); strbuf_release(&buf);


return 0; return ret;
} }


static const char *action_name(const struct replay_opts *opts) static const char *action_name(const struct replay_opts *opts)

View File

@ -237,8 +237,24 @@ test_expect_success 'refs/rewritten/* is worktree-local' '
test_cmp_rev HEAD "$(cat wt/b)" test_cmp_rev HEAD "$(cat wt/b)"
' '


test_expect_success '--abort cleans up refs/rewritten' '
git checkout -b abort-cleans-refs-rewritten H &&
GIT_SEQUENCE_EDITOR="echo break >>" git rebase -ir @^ &&
git rev-parse --verify refs/rewritten/onto &&
git rebase --abort &&
test_must_fail git rev-parse --verify refs/rewritten/onto
'

test_expect_success '--quit cleans up refs/rewritten' '
git checkout -b quit-cleans-refs-rewritten H &&
GIT_SEQUENCE_EDITOR="echo break >>" git rebase -ir @^ &&
git rev-parse --verify refs/rewritten/onto &&
git rebase --quit &&
test_must_fail git rev-parse --verify refs/rewritten/onto
'

test_expect_success 'post-rewrite hook and fixups work for merges' ' test_expect_success 'post-rewrite hook and fixups work for merges' '
git checkout -b post-rewrite && git checkout -b post-rewrite H &&
test_commit same1 && test_commit same1 &&
git reset --hard HEAD^ && git reset --hard HEAD^ &&
test_commit same2 && test_commit same2 &&