Merge branch 'sa/replay-atomic-ref-updates' into seen
"git replay" (experimental) learned to perform ref updates itself in a transaction by default, instead of emitting where each refs should point at and leaving the actual update to another command. Comments? * sa/replay-atomic-ref-updates: replay: make atomic ref updates the default behavior
commit
123b06ac0d
|
|
@ -9,16 +9,16 @@ git-replay - EXPERIMENTAL: Replay commits on a new base, works with bare repos t
|
||||||
SYNOPSIS
|
SYNOPSIS
|
||||||
--------
|
--------
|
||||||
[verse]
|
[verse]
|
||||||
(EXPERIMENTAL!) 'git replay' ([--contained] --onto <newbase> | --advance <branch>) <revision-range>...
|
(EXPERIMENTAL!) 'git replay' ([--contained] --onto <newbase> | --advance <branch>) [--output-commands | --allow-partial] <revision-range>...
|
||||||
|
|
||||||
DESCRIPTION
|
DESCRIPTION
|
||||||
-----------
|
-----------
|
||||||
|
|
||||||
Takes ranges of commits and replays them onto a new location. Leaves
|
Takes ranges of commits and replays them onto a new location. Leaves
|
||||||
the working tree and the index untouched, and updates no references.
|
the working tree and the index untouched, and by default updates the
|
||||||
The output of this command is meant to be used as input to
|
relevant references using atomic transactions. Use `--output-commands`
|
||||||
`git update-ref --stdin`, which would update the relevant branches
|
to get the old default behavior where update commands that can be piped
|
||||||
(see the OUTPUT section below).
|
to `git update-ref --stdin` are emitted (see the OUTPUT section below).
|
||||||
|
|
||||||
THIS COMMAND IS EXPERIMENTAL. THE BEHAVIOR MAY CHANGE.
|
THIS COMMAND IS EXPERIMENTAL. THE BEHAVIOR MAY CHANGE.
|
||||||
|
|
||||||
|
|
@ -42,6 +42,20 @@ When `--advance` is specified, the update-ref command(s) in the output
|
||||||
will update the branch passed as an argument to `--advance` to point at
|
will update the branch passed as an argument to `--advance` to point at
|
||||||
the new commits (in other words, this mimics a cherry-pick operation).
|
the new commits (in other words, this mimics a cherry-pick operation).
|
||||||
|
|
||||||
|
--output-commands::
|
||||||
|
Output update-ref commands instead of updating refs directly.
|
||||||
|
When this option is used, the output can be piped to `git update-ref --stdin`
|
||||||
|
for successive, relatively slow, ref updates. This is equivalent to the
|
||||||
|
old default behavior.
|
||||||
|
|
||||||
|
--allow-partial::
|
||||||
|
Allow some ref updates to succeed even if others fail. By default,
|
||||||
|
ref updates are atomic (all succeed or all fail). With this option,
|
||||||
|
failed updates are reported as warnings rather than causing the entire
|
||||||
|
command to fail. The command exits with code 0 only if all updates
|
||||||
|
succeed; any failures result in exit code 1. Cannot be used with
|
||||||
|
`--output-commands`.
|
||||||
|
|
||||||
<revision-range>::
|
<revision-range>::
|
||||||
Range of commits to replay. More than one <revision-range> can
|
Range of commits to replay. More than one <revision-range> can
|
||||||
be passed, but in `--advance <branch>` mode, they should have
|
be passed, but in `--advance <branch>` mode, they should have
|
||||||
|
|
@ -54,15 +68,20 @@ include::rev-list-options.adoc[]
|
||||||
OUTPUT
|
OUTPUT
|
||||||
------
|
------
|
||||||
|
|
||||||
When there are no conflicts, the output of this command is usable as
|
By default, when there are no conflicts, this command updates the relevant
|
||||||
input to `git update-ref --stdin`. It is of the form:
|
references using atomic transactions and produces no output. All ref updates
|
||||||
|
succeed or all fail (atomic behavior). Use `--allow-partial` to allow some
|
||||||
|
updates to succeed while others fail.
|
||||||
|
|
||||||
|
When `--output-commands` is used, the output is usable as input to
|
||||||
|
`git update-ref --stdin`. It is of the form:
|
||||||
|
|
||||||
update refs/heads/branch1 ${NEW_branch1_HASH} ${OLD_branch1_HASH}
|
update refs/heads/branch1 ${NEW_branch1_HASH} ${OLD_branch1_HASH}
|
||||||
update refs/heads/branch2 ${NEW_branch2_HASH} ${OLD_branch2_HASH}
|
update refs/heads/branch2 ${NEW_branch2_HASH} ${OLD_branch2_HASH}
|
||||||
update refs/heads/branch3 ${NEW_branch3_HASH} ${OLD_branch3_HASH}
|
update refs/heads/branch3 ${NEW_branch3_HASH} ${OLD_branch3_HASH}
|
||||||
|
|
||||||
where the number of refs updated depends on the arguments passed and
|
where the number of refs updated depends on the arguments passed and
|
||||||
the shape of the history being replayed. When using `--advance`, the
|
the shape of the history being replayed. When using `--advance`, the
|
||||||
number of refs updated is always one, but for `--onto`, it can be one
|
number of refs updated is always one, but for `--onto`, it can be one
|
||||||
or more (rebasing multiple branches simultaneously is supported).
|
or more (rebasing multiple branches simultaneously is supported).
|
||||||
|
|
||||||
|
|
@ -77,30 +96,50 @@ is something other than 0 or 1.
|
||||||
EXAMPLES
|
EXAMPLES
|
||||||
--------
|
--------
|
||||||
|
|
||||||
To simply rebase `mybranch` onto `target`:
|
To simply rebase `mybranch` onto `target` (default behavior):
|
||||||
|
|
||||||
------------
|
------------
|
||||||
$ git replay --onto target origin/main..mybranch
|
$ git replay --onto target origin/main..mybranch
|
||||||
update refs/heads/mybranch ${NEW_mybranch_HASH} ${OLD_mybranch_HASH}
|
|
||||||
------------
|
------------
|
||||||
|
|
||||||
To cherry-pick the commits from mybranch onto target:
|
To cherry-pick the commits from mybranch onto target:
|
||||||
|
|
||||||
------------
|
------------
|
||||||
$ git replay --advance target origin/main..mybranch
|
$ git replay --advance target origin/main..mybranch
|
||||||
update refs/heads/target ${NEW_target_HASH} ${OLD_target_HASH}
|
|
||||||
------------
|
------------
|
||||||
|
|
||||||
Note that the first two examples replay the exact same commits and on
|
Note that the first two examples replay the exact same commits and on
|
||||||
top of the exact same new base, they only differ in that the first
|
top of the exact same new base, they only differ in that the first
|
||||||
provides instructions to make mybranch point at the new commits and
|
updates mybranch to point at the new commits and the second updates
|
||||||
the second provides instructions to make target point at them.
|
target to point at them.
|
||||||
|
|
||||||
|
To get the old default behavior where update commands are emitted:
|
||||||
|
|
||||||
|
------------
|
||||||
|
$ git replay --output-commands --onto target origin/main..mybranch
|
||||||
|
update refs/heads/mybranch ${NEW_mybranch_HASH} ${OLD_mybranch_HASH}
|
||||||
|
------------
|
||||||
|
|
||||||
|
To rebase multiple branches with partial failure tolerance:
|
||||||
|
|
||||||
|
------------
|
||||||
|
$ git replay --allow-partial --contained --onto origin/main origin/main..tipbranch
|
||||||
|
------------
|
||||||
|
|
||||||
What if you have a stack of branches, one depending upon another, and
|
What if you have a stack of branches, one depending upon another, and
|
||||||
you'd really like to rebase the whole set?
|
you'd really like to rebase the whole set?
|
||||||
|
|
||||||
------------
|
------------
|
||||||
$ git replay --contained --onto origin/main origin/main..tipbranch
|
$ git replay --contained --onto origin/main origin/main..tipbranch
|
||||||
|
------------
|
||||||
|
|
||||||
|
This automatically finds and rebases all branches contained within the
|
||||||
|
`origin/main..tipbranch` range.
|
||||||
|
|
||||||
|
Or if you want to see the old default behavior where update commands are emitted:
|
||||||
|
|
||||||
|
------------
|
||||||
|
$ git replay --output-commands --contained --onto origin/main origin/main..tipbranch
|
||||||
update refs/heads/branch1 ${NEW_branch1_HASH} ${OLD_branch1_HASH}
|
update refs/heads/branch1 ${NEW_branch1_HASH} ${OLD_branch1_HASH}
|
||||||
update refs/heads/branch2 ${NEW_branch2_HASH} ${OLD_branch2_HASH}
|
update refs/heads/branch2 ${NEW_branch2_HASH} ${OLD_branch2_HASH}
|
||||||
update refs/heads/tipbranch ${NEW_tipbranch_HASH} ${OLD_tipbranch_HASH}
|
update refs/heads/tipbranch ${NEW_tipbranch_HASH} ${OLD_tipbranch_HASH}
|
||||||
|
|
@ -108,10 +147,19 @@ update refs/heads/tipbranch ${NEW_tipbranch_HASH} ${OLD_tipbranch_HASH}
|
||||||
|
|
||||||
When calling `git replay`, one does not need to specify a range of
|
When calling `git replay`, one does not need to specify a range of
|
||||||
commits to replay using the syntax `A..B`; any range expression will
|
commits to replay using the syntax `A..B`; any range expression will
|
||||||
do:
|
do. Here's an example where you explicitly specify which branches to rebase:
|
||||||
|
|
||||||
------------
|
------------
|
||||||
$ git replay --onto origin/main ^base branch1 branch2 branch3
|
$ git replay --onto origin/main ^base branch1 branch2 branch3
|
||||||
|
------------
|
||||||
|
|
||||||
|
This gives you explicit control over exactly which branches are rebased,
|
||||||
|
unlike the previous `--contained` example which automatically discovers them.
|
||||||
|
|
||||||
|
To see the update commands that would be executed:
|
||||||
|
|
||||||
|
------------
|
||||||
|
$ git replay --output-commands --onto origin/main ^base branch1 branch2 branch3
|
||||||
update refs/heads/branch1 ${NEW_branch1_HASH} ${OLD_branch1_HASH}
|
update refs/heads/branch1 ${NEW_branch1_HASH} ${OLD_branch1_HASH}
|
||||||
update refs/heads/branch2 ${NEW_branch2_HASH} ${OLD_branch2_HASH}
|
update refs/heads/branch2 ${NEW_branch2_HASH} ${OLD_branch2_HASH}
|
||||||
update refs/heads/branch3 ${NEW_branch3_HASH} ${OLD_branch3_HASH}
|
update refs/heads/branch3 ${NEW_branch3_HASH} ${OLD_branch3_HASH}
|
||||||
|
|
|
||||||
114
builtin/replay.c
114
builtin/replay.c
|
|
@ -284,6 +284,28 @@ static struct commit *pick_regular_commit(struct repository *repo,
|
||||||
return create_commit(repo, result->tree, pickme, replayed_base);
|
return create_commit(repo, result->tree, pickme, replayed_base);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int add_ref_to_transaction(struct ref_transaction *transaction,
|
||||||
|
const char *refname,
|
||||||
|
const struct object_id *new_oid,
|
||||||
|
const struct object_id *old_oid,
|
||||||
|
struct strbuf *err)
|
||||||
|
{
|
||||||
|
return ref_transaction_update(transaction, refname, new_oid, old_oid,
|
||||||
|
NULL, NULL, 0, "git replay", err);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void print_rejected_update(const char *refname,
|
||||||
|
const struct object_id *old_oid UNUSED,
|
||||||
|
const struct object_id *new_oid UNUSED,
|
||||||
|
const char *old_target UNUSED,
|
||||||
|
const char *new_target UNUSED,
|
||||||
|
enum ref_transaction_error err,
|
||||||
|
void *cb_data UNUSED)
|
||||||
|
{
|
||||||
|
const char *reason = ref_transaction_error_msg(err);
|
||||||
|
warning(_("failed to update %s: %s"), refname, reason);
|
||||||
|
}
|
||||||
|
|
||||||
int cmd_replay(int argc,
|
int cmd_replay(int argc,
|
||||||
const char **argv,
|
const char **argv,
|
||||||
const char *prefix,
|
const char *prefix,
|
||||||
|
|
@ -294,6 +316,8 @@ int cmd_replay(int argc,
|
||||||
struct commit *onto = NULL;
|
struct commit *onto = NULL;
|
||||||
const char *onto_name = NULL;
|
const char *onto_name = NULL;
|
||||||
int contained = 0;
|
int contained = 0;
|
||||||
|
int output_commands = 0;
|
||||||
|
int allow_partial = 0;
|
||||||
|
|
||||||
struct rev_info revs;
|
struct rev_info revs;
|
||||||
struct commit *last_commit = NULL;
|
struct commit *last_commit = NULL;
|
||||||
|
|
@ -302,12 +326,15 @@ int cmd_replay(int argc,
|
||||||
struct merge_result result;
|
struct merge_result result;
|
||||||
struct strset *update_refs = NULL;
|
struct strset *update_refs = NULL;
|
||||||
kh_oid_map_t *replayed_commits;
|
kh_oid_map_t *replayed_commits;
|
||||||
|
struct ref_transaction *transaction = NULL;
|
||||||
|
struct strbuf transaction_err = STRBUF_INIT;
|
||||||
|
int commits_processed = 0;
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
const char * const replay_usage[] = {
|
const char *const replay_usage[] = {
|
||||||
N_("(EXPERIMENTAL!) git replay "
|
N_("(EXPERIMENTAL!) git replay "
|
||||||
"([--contained] --onto <newbase> | --advance <branch>) "
|
"([--contained] --onto <newbase> | --advance <branch>) "
|
||||||
"<revision-range>..."),
|
"[--output-commands | --allow-partial] <revision-range>..."),
|
||||||
NULL
|
NULL
|
||||||
};
|
};
|
||||||
struct option replay_options[] = {
|
struct option replay_options[] = {
|
||||||
|
|
@ -319,6 +346,10 @@ int cmd_replay(int argc,
|
||||||
N_("replay onto given commit")),
|
N_("replay onto given commit")),
|
||||||
OPT_BOOL(0, "contained", &contained,
|
OPT_BOOL(0, "contained", &contained,
|
||||||
N_("advance all branches contained in revision-range")),
|
N_("advance all branches contained in revision-range")),
|
||||||
|
OPT_BOOL(0, "output-commands", &output_commands,
|
||||||
|
N_("output update commands instead of updating refs")),
|
||||||
|
OPT_BOOL(0, "allow-partial", &allow_partial,
|
||||||
|
N_("allow some ref updates to succeed even if others fail")),
|
||||||
OPT_END()
|
OPT_END()
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
@ -330,9 +361,12 @@ int cmd_replay(int argc,
|
||||||
usage_with_options(replay_usage, replay_options);
|
usage_with_options(replay_usage, replay_options);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (advance_name_opt && contained)
|
die_for_incompatible_opt2(!!advance_name_opt, "--advance",
|
||||||
die(_("options '%s' and '%s' cannot be used together"),
|
contained, "--contained");
|
||||||
"--advance", "--contained");
|
|
||||||
|
die_for_incompatible_opt2(allow_partial, "--allow-partial",
|
||||||
|
output_commands, "--output-commands");
|
||||||
|
|
||||||
advance_name = xstrdup_or_null(advance_name_opt);
|
advance_name = xstrdup_or_null(advance_name_opt);
|
||||||
|
|
||||||
repo_init_revisions(repo, &revs, prefix);
|
repo_init_revisions(repo, &revs, prefix);
|
||||||
|
|
@ -389,6 +423,17 @@ int cmd_replay(int argc,
|
||||||
determine_replay_mode(repo, &revs.cmdline, onto_name, &advance_name,
|
determine_replay_mode(repo, &revs.cmdline, onto_name, &advance_name,
|
||||||
&onto, &update_refs);
|
&onto, &update_refs);
|
||||||
|
|
||||||
|
if (!output_commands) {
|
||||||
|
unsigned int transaction_flags = allow_partial ? REF_TRANSACTION_ALLOW_FAILURE : 0;
|
||||||
|
transaction = ref_store_transaction_begin(get_main_ref_store(repo),
|
||||||
|
transaction_flags,
|
||||||
|
&transaction_err);
|
||||||
|
if (!transaction) {
|
||||||
|
ret = error(_("failed to begin ref transaction: %s"), transaction_err.buf);
|
||||||
|
goto cleanup;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (!onto) /* FIXME: Should handle replaying down to root commit */
|
if (!onto) /* FIXME: Should handle replaying down to root commit */
|
||||||
die("Replaying down to root commit is not supported yet!");
|
die("Replaying down to root commit is not supported yet!");
|
||||||
|
|
||||||
|
|
@ -407,6 +452,8 @@ int cmd_replay(int argc,
|
||||||
khint_t pos;
|
khint_t pos;
|
||||||
int hr;
|
int hr;
|
||||||
|
|
||||||
|
commits_processed = 1;
|
||||||
|
|
||||||
if (!commit->parents)
|
if (!commit->parents)
|
||||||
die(_("replaying down to root commit is not supported yet!"));
|
die(_("replaying down to root commit is not supported yet!"));
|
||||||
if (commit->parents->next)
|
if (commit->parents->next)
|
||||||
|
|
@ -434,10 +481,18 @@ int cmd_replay(int argc,
|
||||||
if (decoration->type == DECORATION_REF_LOCAL &&
|
if (decoration->type == DECORATION_REF_LOCAL &&
|
||||||
(contained || strset_contains(update_refs,
|
(contained || strset_contains(update_refs,
|
||||||
decoration->name))) {
|
decoration->name))) {
|
||||||
printf("update %s %s %s\n",
|
if (output_commands) {
|
||||||
decoration->name,
|
printf("update %s %s %s\n",
|
||||||
oid_to_hex(&last_commit->object.oid),
|
decoration->name,
|
||||||
oid_to_hex(&commit->object.oid));
|
oid_to_hex(&last_commit->object.oid),
|
||||||
|
oid_to_hex(&commit->object.oid));
|
||||||
|
} else if (add_ref_to_transaction(transaction, decoration->name,
|
||||||
|
&last_commit->object.oid,
|
||||||
|
&commit->object.oid,
|
||||||
|
&transaction_err) < 0) {
|
||||||
|
ret = error(_("failed to add ref update to transaction: %s"), transaction_err.buf);
|
||||||
|
goto cleanup;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
decoration = decoration->next;
|
decoration = decoration->next;
|
||||||
}
|
}
|
||||||
|
|
@ -445,10 +500,33 @@ int cmd_replay(int argc,
|
||||||
|
|
||||||
/* In --advance mode, advance the target ref */
|
/* In --advance mode, advance the target ref */
|
||||||
if (result.clean == 1 && advance_name) {
|
if (result.clean == 1 && advance_name) {
|
||||||
printf("update %s %s %s\n",
|
if (output_commands) {
|
||||||
advance_name,
|
printf("update %s %s %s\n",
|
||||||
oid_to_hex(&last_commit->object.oid),
|
advance_name,
|
||||||
oid_to_hex(&onto->object.oid));
|
oid_to_hex(&last_commit->object.oid),
|
||||||
|
oid_to_hex(&onto->object.oid));
|
||||||
|
} else if (add_ref_to_transaction(transaction, advance_name,
|
||||||
|
&last_commit->object.oid,
|
||||||
|
&onto->object.oid,
|
||||||
|
&transaction_err) < 0) {
|
||||||
|
ret = error(_("failed to add ref update to transaction: %s"), transaction_err.buf);
|
||||||
|
goto cleanup;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Commit the ref transaction if we have one */
|
||||||
|
if (transaction && result.clean == 1) {
|
||||||
|
if (ref_transaction_commit(transaction, &transaction_err)) {
|
||||||
|
if (allow_partial) {
|
||||||
|
warning(_("some ref updates failed: %s"), transaction_err.buf);
|
||||||
|
ref_transaction_for_each_rejected_update(transaction,
|
||||||
|
print_rejected_update, NULL);
|
||||||
|
ret = 0; /* Set failure even with allow_partial */
|
||||||
|
} else {
|
||||||
|
ret = error(_("failed to update refs: %s"), transaction_err.buf);
|
||||||
|
goto cleanup;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
merge_finalize(&merge_opt, &result);
|
merge_finalize(&merge_opt, &result);
|
||||||
|
|
@ -457,9 +535,17 @@ int cmd_replay(int argc,
|
||||||
strset_clear(update_refs);
|
strset_clear(update_refs);
|
||||||
free(update_refs);
|
free(update_refs);
|
||||||
}
|
}
|
||||||
ret = result.clean;
|
|
||||||
|
/* Handle empty ranges: if no commits were processed, treat as success */
|
||||||
|
if (!commits_processed)
|
||||||
|
ret = 1; /* Success - no commits to replay is not an error */
|
||||||
|
else
|
||||||
|
ret = result.clean;
|
||||||
|
|
||||||
cleanup:
|
cleanup:
|
||||||
|
if (transaction)
|
||||||
|
ref_transaction_free(transaction);
|
||||||
|
strbuf_release(&transaction_err);
|
||||||
release_revisions(&revs);
|
release_revisions(&revs);
|
||||||
free(advance_name);
|
free(advance_name);
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -52,7 +52,7 @@ test_expect_success 'setup bare' '
|
||||||
'
|
'
|
||||||
|
|
||||||
test_expect_success 'using replay to rebase two branches, one on top of other' '
|
test_expect_success 'using replay to rebase two branches, one on top of other' '
|
||||||
git replay --onto main topic1..topic2 >result &&
|
git replay --output-commands --onto main topic1..topic2 >result &&
|
||||||
|
|
||||||
test_line_count = 1 result &&
|
test_line_count = 1 result &&
|
||||||
|
|
||||||
|
|
@ -67,9 +67,30 @@ test_expect_success 'using replay to rebase two branches, one on top of other' '
|
||||||
test_cmp expect result
|
test_cmp expect result
|
||||||
'
|
'
|
||||||
|
|
||||||
|
test_expect_success 'using replay with default atomic behavior (no output)' '
|
||||||
|
# Create a test branch that wont interfere with others
|
||||||
|
git branch atomic-test topic2 &&
|
||||||
|
git rev-parse atomic-test >atomic-test-old &&
|
||||||
|
|
||||||
|
# Default behavior: atomic ref updates (no output)
|
||||||
|
git replay --onto main topic1..atomic-test >output &&
|
||||||
|
test_must_be_empty output &&
|
||||||
|
|
||||||
|
# Verify the branch was updated
|
||||||
|
git rev-parse atomic-test >atomic-test-new &&
|
||||||
|
! test_cmp atomic-test-old atomic-test-new &&
|
||||||
|
|
||||||
|
# Verify the history is correct
|
||||||
|
git log --format=%s atomic-test >actual &&
|
||||||
|
test_write_lines E D M L B A >expect &&
|
||||||
|
test_cmp expect actual
|
||||||
|
'
|
||||||
|
|
||||||
test_expect_success 'using replay on bare repo to rebase two branches, one on top of other' '
|
test_expect_success 'using replay on bare repo to rebase two branches, one on top of other' '
|
||||||
git -C bare replay --onto main topic1..topic2 >result-bare &&
|
git -C bare replay --output-commands --onto main topic1..topic2 >result-bare &&
|
||||||
test_cmp expect result-bare
|
|
||||||
|
# The result should match what we got from the regular repo
|
||||||
|
test_cmp result result-bare
|
||||||
'
|
'
|
||||||
|
|
||||||
test_expect_success 'using replay to rebase with a conflict' '
|
test_expect_success 'using replay to rebase with a conflict' '
|
||||||
|
|
@ -86,7 +107,7 @@ test_expect_success 'using replay to perform basic cherry-pick' '
|
||||||
# 2nd field of result is refs/heads/main vs. refs/heads/topic2
|
# 2nd field of result is refs/heads/main vs. refs/heads/topic2
|
||||||
# 4th field of result is hash for main instead of hash for topic2
|
# 4th field of result is hash for main instead of hash for topic2
|
||||||
|
|
||||||
git replay --advance main topic1..topic2 >result &&
|
git replay --output-commands --advance main topic1..topic2 >result &&
|
||||||
|
|
||||||
test_line_count = 1 result &&
|
test_line_count = 1 result &&
|
||||||
|
|
||||||
|
|
@ -102,7 +123,7 @@ test_expect_success 'using replay to perform basic cherry-pick' '
|
||||||
'
|
'
|
||||||
|
|
||||||
test_expect_success 'using replay on bare repo to perform basic cherry-pick' '
|
test_expect_success 'using replay on bare repo to perform basic cherry-pick' '
|
||||||
git -C bare replay --advance main topic1..topic2 >result-bare &&
|
git -C bare replay --output-commands --advance main topic1..topic2 >result-bare &&
|
||||||
test_cmp expect result-bare
|
test_cmp expect result-bare
|
||||||
'
|
'
|
||||||
|
|
||||||
|
|
@ -115,7 +136,7 @@ test_expect_success 'replay fails when both --advance and --onto are omitted' '
|
||||||
'
|
'
|
||||||
|
|
||||||
test_expect_success 'using replay to also rebase a contained branch' '
|
test_expect_success 'using replay to also rebase a contained branch' '
|
||||||
git replay --contained --onto main main..topic3 >result &&
|
git replay --output-commands --contained --onto main main..topic3 >result &&
|
||||||
|
|
||||||
test_line_count = 2 result &&
|
test_line_count = 2 result &&
|
||||||
cut -f 3 -d " " result >new-branch-tips &&
|
cut -f 3 -d " " result >new-branch-tips &&
|
||||||
|
|
@ -139,12 +160,12 @@ test_expect_success 'using replay to also rebase a contained branch' '
|
||||||
'
|
'
|
||||||
|
|
||||||
test_expect_success 'using replay on bare repo to also rebase a contained branch' '
|
test_expect_success 'using replay on bare repo to also rebase a contained branch' '
|
||||||
git -C bare replay --contained --onto main main..topic3 >result-bare &&
|
git -C bare replay --output-commands --contained --onto main main..topic3 >result-bare &&
|
||||||
test_cmp expect result-bare
|
test_cmp expect result-bare
|
||||||
'
|
'
|
||||||
|
|
||||||
test_expect_success 'using replay to rebase multiple divergent branches' '
|
test_expect_success 'using replay to rebase multiple divergent branches' '
|
||||||
git replay --onto main ^topic1 topic2 topic4 >result &&
|
git replay --output-commands --onto main ^topic1 topic2 topic4 >result &&
|
||||||
|
|
||||||
test_line_count = 2 result &&
|
test_line_count = 2 result &&
|
||||||
cut -f 3 -d " " result >new-branch-tips &&
|
cut -f 3 -d " " result >new-branch-tips &&
|
||||||
|
|
@ -168,7 +189,7 @@ test_expect_success 'using replay to rebase multiple divergent branches' '
|
||||||
'
|
'
|
||||||
|
|
||||||
test_expect_success 'using replay on bare repo to rebase multiple divergent branches, including contained ones' '
|
test_expect_success 'using replay on bare repo to rebase multiple divergent branches, including contained ones' '
|
||||||
git -C bare replay --contained --onto main ^main topic2 topic3 topic4 >result &&
|
git -C bare replay --output-commands --contained --onto main ^main topic2 topic3 topic4 >result &&
|
||||||
|
|
||||||
test_line_count = 4 result &&
|
test_line_count = 4 result &&
|
||||||
cut -f 3 -d " " result >new-branch-tips &&
|
cut -f 3 -d " " result >new-branch-tips &&
|
||||||
|
|
@ -217,4 +238,131 @@ test_expect_success 'merge.directoryRenames=false' '
|
||||||
--onto rename-onto rename-onto..rename-from
|
--onto rename-onto rename-onto..rename-from
|
||||||
'
|
'
|
||||||
|
|
||||||
|
# Tests for new default atomic behavior and options
|
||||||
|
|
||||||
|
test_expect_success 'replay default behavior should not produce output when successful' '
|
||||||
|
git replay --onto main topic1..topic3 >output &&
|
||||||
|
test_must_be_empty output
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'replay with --output-commands produces traditional output' '
|
||||||
|
git replay --output-commands --onto main topic1..topic3 >output &&
|
||||||
|
test_line_count = 1 output &&
|
||||||
|
grep "^update refs/heads/topic3 " output
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'replay with --allow-partial should not produce output when successful' '
|
||||||
|
git replay --allow-partial --onto main topic1..topic3 >output &&
|
||||||
|
test_must_be_empty output
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'replay fails when --output-commands and --allow-partial are used together' '
|
||||||
|
test_must_fail git replay --output-commands --allow-partial --onto main topic1..topic2 2>error &&
|
||||||
|
grep "cannot be used together" error
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'replay with --contained updates multiple branches atomically' '
|
||||||
|
# Create fresh test branches based on the original structure
|
||||||
|
# contained-topic1 should be contained within the range to contained-topic3
|
||||||
|
git branch contained-base main &&
|
||||||
|
git checkout -b contained-topic1 contained-base &&
|
||||||
|
test_commit ContainedC &&
|
||||||
|
git checkout -b contained-topic3 contained-topic1 &&
|
||||||
|
test_commit ContainedG &&
|
||||||
|
test_commit ContainedH &&
|
||||||
|
git checkout main &&
|
||||||
|
|
||||||
|
# Store original states
|
||||||
|
git rev-parse contained-topic1 >contained-topic1-old &&
|
||||||
|
git rev-parse contained-topic3 >contained-topic3-old &&
|
||||||
|
|
||||||
|
# Use --contained to update multiple branches - this should update both
|
||||||
|
git replay --contained --onto main contained-base..contained-topic3 &&
|
||||||
|
|
||||||
|
# Verify both branches were updated
|
||||||
|
git rev-parse contained-topic1 >contained-topic1-new &&
|
||||||
|
git rev-parse contained-topic3 >contained-topic3-new &&
|
||||||
|
! test_cmp contained-topic1-old contained-topic1-new &&
|
||||||
|
! test_cmp contained-topic3-old contained-topic3-new
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'replay atomic behavior: all refs updated or none' '
|
||||||
|
# Store original state
|
||||||
|
git rev-parse topic4 >topic4-old &&
|
||||||
|
|
||||||
|
# Default atomic behavior
|
||||||
|
git replay --onto main main..topic4 &&
|
||||||
|
|
||||||
|
# Verify ref was updated
|
||||||
|
git rev-parse topic4 >topic4-new &&
|
||||||
|
! test_cmp topic4-old topic4-new &&
|
||||||
|
|
||||||
|
# Verify no partial state
|
||||||
|
git log --format=%s topic4 >actual &&
|
||||||
|
test_write_lines J I M L B A >expect &&
|
||||||
|
test_cmp expect actual
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'replay works correctly with bare repositories' '
|
||||||
|
# Test atomic behavior in bare repo (important for Gitaly)
|
||||||
|
git checkout -b bare-test topic1 &&
|
||||||
|
test_commit BareTest &&
|
||||||
|
|
||||||
|
# Test with bare repo - replay the commits from main..bare-test to get the full history
|
||||||
|
git -C bare fetch .. bare-test:bare-test &&
|
||||||
|
git -C bare replay --onto main main..bare-test &&
|
||||||
|
|
||||||
|
# Verify the bare repo was updated correctly (no output)
|
||||||
|
git -C bare log --format=%s bare-test >actual &&
|
||||||
|
test_write_lines BareTest F C M L B A >expect &&
|
||||||
|
test_cmp expect actual
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'replay --allow-partial with no failures produces no output' '
|
||||||
|
git checkout -b partial-test topic1 &&
|
||||||
|
test_commit PartialTest &&
|
||||||
|
|
||||||
|
# Should succeed silently even with partial mode
|
||||||
|
git replay --allow-partial --onto main topic1..partial-test >output &&
|
||||||
|
test_must_be_empty output
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'replay maintains ref update consistency' '
|
||||||
|
# Test that traditional vs atomic produce equivalent results
|
||||||
|
git checkout -b method1-test topic2 &&
|
||||||
|
git checkout -b method2-test topic2 &&
|
||||||
|
|
||||||
|
# Both methods should update refs to point to the same replayed commits
|
||||||
|
git replay --output-commands --onto main topic1..method1-test >update-commands &&
|
||||||
|
git update-ref --stdin <update-commands &&
|
||||||
|
git log --format=%s method1-test >traditional-result &&
|
||||||
|
|
||||||
|
# Direct atomic method should produce same commit history
|
||||||
|
git replay --onto main topic1..method2-test &&
|
||||||
|
git log --format=%s method2-test >atomic-result &&
|
||||||
|
|
||||||
|
# Both methods should produce identical commit histories
|
||||||
|
test_cmp traditional-result atomic-result
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'replay error messages are helpful and clear' '
|
||||||
|
# Test that error messages are clear
|
||||||
|
test_must_fail git replay --output-commands --allow-partial --onto main topic1..topic2 2>error &&
|
||||||
|
grep "cannot be used together" error
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'replay with empty range produces no output and no changes' '
|
||||||
|
# Create a test branch for empty range testing
|
||||||
|
git checkout -b empty-test topic1 &&
|
||||||
|
git rev-parse empty-test >empty-test-before &&
|
||||||
|
|
||||||
|
# Empty range should succeed but do nothing
|
||||||
|
git replay --onto main empty-test..empty-test >output &&
|
||||||
|
test_must_be_empty output &&
|
||||||
|
|
||||||
|
# Branch should be unchanged
|
||||||
|
git rev-parse empty-test >empty-test-after &&
|
||||||
|
test_cmp empty-test-before empty-test-after
|
||||||
|
'
|
||||||
|
|
||||||
test_done
|
test_done
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue