refs: support rejection in batch updates during F/D checks

The `refs_verify_refnames_available()` is used to batch check refnames
for F/D conflicts. While this is the more performant alternative than
its individual version, it does not provide rejection capabilities on a
single update level. For batched updates, this would mean a rejection of
the entire transaction whenever one reference has a F/D conflict.

Modify the function to call `ref_transaction_maybe_set_rejected()` to
check if a single update can be rejected. Since this function is only
internally used within 'refs/' and we want to pass in a `struct
ref_transaction *` as a variable. We also move and mark
`refs_verify_refnames_available()` to 'refs-internal.h' to be an
internal function.

Signed-off-by: Karthik Nayak <karthik.188@gmail.com>
Acked-by: Patrick Steinhardt <ps@pks.im>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
Karthik Nayak
2025-04-08 10:51:11 +02:00
committed by Junio C Hamano
parent 23fc8e4f61
commit 31726bb90d
5 changed files with 76 additions and 27 deletions

37
refs.c
View File

@@ -2540,6 +2540,7 @@ enum ref_transaction_error refs_verify_refnames_available(struct ref_store *refs
const struct string_list *refnames, const struct string_list *refnames,
const struct string_list *extras, const struct string_list *extras,
const struct string_list *skip, const struct string_list *skip,
struct ref_transaction *transaction,
unsigned int initial_transaction, unsigned int initial_transaction,
struct strbuf *err) struct strbuf *err)
{ {
@@ -2547,6 +2548,7 @@ enum ref_transaction_error refs_verify_refnames_available(struct ref_store *refs
struct strbuf referent = STRBUF_INIT; struct strbuf referent = STRBUF_INIT;
struct string_list_item *item; struct string_list_item *item;
struct ref_iterator *iter = NULL; struct ref_iterator *iter = NULL;
struct strset conflicting_dirnames;
struct strset dirnames; struct strset dirnames;
int ret = REF_TRANSACTION_ERROR_NAME_CONFLICT; int ret = REF_TRANSACTION_ERROR_NAME_CONFLICT;
@@ -2557,9 +2559,11 @@ enum ref_transaction_error refs_verify_refnames_available(struct ref_store *refs
assert(err); assert(err);
strset_init(&conflicting_dirnames);
strset_init(&dirnames); strset_init(&dirnames);
for_each_string_list_item(item, refnames) { for_each_string_list_item(item, refnames) {
const size_t *update_idx = (size_t *)item->util;
const char *refname = item->string; const char *refname = item->string;
const char *extra_refname; const char *extra_refname;
struct object_id oid; struct object_id oid;
@@ -2597,14 +2601,30 @@ enum ref_transaction_error refs_verify_refnames_available(struct ref_store *refs
continue; continue;
if (!initial_transaction && if (!initial_transaction &&
!refs_read_raw_ref(refs, dirname.buf, &oid, &referent, (strset_contains(&conflicting_dirnames, dirname.buf) ||
&type, &ignore_errno)) { !refs_read_raw_ref(refs, dirname.buf, &oid, &referent,
&type, &ignore_errno))) {
if (transaction && ref_transaction_maybe_set_rejected(
transaction, *update_idx,
REF_TRANSACTION_ERROR_NAME_CONFLICT)) {
strset_remove(&dirnames, dirname.buf);
strset_add(&conflicting_dirnames, dirname.buf);
continue;
}
strbuf_addf(err, _("'%s' exists; cannot create '%s'"), strbuf_addf(err, _("'%s' exists; cannot create '%s'"),
dirname.buf, refname); dirname.buf, refname);
goto cleanup; goto cleanup;
} }
if (extras && string_list_has_string(extras, dirname.buf)) { if (extras && string_list_has_string(extras, dirname.buf)) {
if (transaction && ref_transaction_maybe_set_rejected(
transaction, *update_idx,
REF_TRANSACTION_ERROR_NAME_CONFLICT)) {
strset_remove(&dirnames, dirname.buf);
continue;
}
strbuf_addf(err, _("cannot process '%s' and '%s' at the same time"), strbuf_addf(err, _("cannot process '%s' and '%s' at the same time"),
refname, dirname.buf); refname, dirname.buf);
goto cleanup; goto cleanup;
@@ -2637,6 +2657,11 @@ enum ref_transaction_error refs_verify_refnames_available(struct ref_store *refs
string_list_has_string(skip, iter->refname)) string_list_has_string(skip, iter->refname))
continue; continue;
if (transaction && ref_transaction_maybe_set_rejected(
transaction, *update_idx,
REF_TRANSACTION_ERROR_NAME_CONFLICT))
continue;
strbuf_addf(err, _("'%s' exists; cannot create '%s'"), strbuf_addf(err, _("'%s' exists; cannot create '%s'"),
iter->refname, refname); iter->refname, refname);
goto cleanup; goto cleanup;
@@ -2648,6 +2673,11 @@ enum ref_transaction_error refs_verify_refnames_available(struct ref_store *refs
extra_refname = find_descendant_ref(dirname.buf, extras, skip); extra_refname = find_descendant_ref(dirname.buf, extras, skip);
if (extra_refname) { if (extra_refname) {
if (transaction && ref_transaction_maybe_set_rejected(
transaction, *update_idx,
REF_TRANSACTION_ERROR_NAME_CONFLICT))
continue;
strbuf_addf(err, _("cannot process '%s' and '%s' at the same time"), strbuf_addf(err, _("cannot process '%s' and '%s' at the same time"),
refname, extra_refname); refname, extra_refname);
goto cleanup; goto cleanup;
@@ -2659,6 +2689,7 @@ enum ref_transaction_error refs_verify_refnames_available(struct ref_store *refs
cleanup: cleanup:
strbuf_release(&referent); strbuf_release(&referent);
strbuf_release(&dirname); strbuf_release(&dirname);
strset_clear(&conflicting_dirnames);
strset_clear(&dirnames); strset_clear(&dirnames);
ref_iterator_free(iter); ref_iterator_free(iter);
return ret; return ret;
@@ -2679,7 +2710,7 @@ enum ref_transaction_error refs_verify_refname_available(
}; };
return refs_verify_refnames_available(refs, &refnames, extras, skip, return refs_verify_refnames_available(refs, &refnames, extras, skip,
initial_transaction, err); NULL, initial_transaction, err);
} }
struct do_for_each_reflog_help { struct do_for_each_reflog_help {

12
refs.h
View File

@@ -141,18 +141,6 @@ enum ref_transaction_error refs_verify_refname_available(struct ref_store *refs,
unsigned int initial_transaction, unsigned int initial_transaction,
struct strbuf *err); struct strbuf *err);
/*
* Same as `refs_verify_refname_available()`, but checking for a list of
* refnames instead of only a single item. This is more efficient in the case
* where one needs to check multiple refnames.
*/
enum ref_transaction_error refs_verify_refnames_available(struct ref_store *refs,
const struct string_list *refnames,
const struct string_list *extras,
const struct string_list *skip,
unsigned int initial_transaction,
struct strbuf *err);
int refs_ref_exists(struct ref_store *refs, const char *refname); int refs_ref_exists(struct ref_store *refs, const char *refname);
int should_autocreate_reflog(enum log_refs_config log_all_ref_updates, int should_autocreate_reflog(enum log_refs_config log_all_ref_updates,

View File

@@ -677,16 +677,18 @@ static void unlock_ref(struct ref_lock *lock)
* - Generate informative error messages in the case of failure * - Generate informative error messages in the case of failure
*/ */
static enum ref_transaction_error lock_raw_ref(struct files_ref_store *refs, static enum ref_transaction_error lock_raw_ref(struct files_ref_store *refs,
const char *refname, struct ref_update *update,
size_t update_idx,
int mustexist, int mustexist,
struct string_list *refnames_to_check, struct string_list *refnames_to_check,
const struct string_list *extras, const struct string_list *extras,
struct ref_lock **lock_p, struct ref_lock **lock_p,
struct strbuf *referent, struct strbuf *referent,
unsigned int *type,
struct strbuf *err) struct strbuf *err)
{ {
enum ref_transaction_error ret = REF_TRANSACTION_ERROR_GENERIC; enum ref_transaction_error ret = REF_TRANSACTION_ERROR_GENERIC;
const char *refname = update->refname;
unsigned int *type = &update->type;
struct ref_lock *lock; struct ref_lock *lock;
struct strbuf ref_file = STRBUF_INIT; struct strbuf ref_file = STRBUF_INIT;
int attempts_remaining = 3; int attempts_remaining = 3;
@@ -785,6 +787,8 @@ retry:
if (files_read_raw_ref(&refs->base, refname, &lock->old_oid, referent, if (files_read_raw_ref(&refs->base, refname, &lock->old_oid, referent,
type, &failure_errno)) { type, &failure_errno)) {
struct string_list_item *item;
if (failure_errno == ENOENT) { if (failure_errno == ENOENT) {
if (mustexist) { if (mustexist) {
/* Garden variety missing reference. */ /* Garden variety missing reference. */
@@ -864,7 +868,9 @@ retry:
* make sure there is no existing packed ref that conflicts * make sure there is no existing packed ref that conflicts
* with refname. This check is deferred so that we can batch it. * with refname. This check is deferred so that we can batch it.
*/ */
string_list_append(refnames_to_check, refname); item = string_list_append(refnames_to_check, refname);
item->util = xmalloc(sizeof(update_idx));
memcpy(item->util, &update_idx, sizeof(update_idx));
} }
ret = 0; ret = 0;
@@ -2547,6 +2553,7 @@ struct files_transaction_backend_data {
*/ */
static enum ref_transaction_error lock_ref_for_update(struct files_ref_store *refs, static enum ref_transaction_error lock_ref_for_update(struct files_ref_store *refs,
struct ref_update *update, struct ref_update *update,
size_t update_idx,
struct ref_transaction *transaction, struct ref_transaction *transaction,
const char *head_ref, const char *head_ref,
struct string_list *refnames_to_check, struct string_list *refnames_to_check,
@@ -2575,9 +2582,9 @@ static enum ref_transaction_error lock_ref_for_update(struct files_ref_store *re
if (lock) { if (lock) {
lock->count++; lock->count++;
} else { } else {
ret = lock_raw_ref(refs, update->refname, mustexist, ret = lock_raw_ref(refs, update, update_idx, mustexist,
refnames_to_check, &transaction->refnames, refnames_to_check, &transaction->refnames,
&lock, &referent, &update->type, err); &lock, &referent, err);
if (ret) { if (ret) {
char *reason; char *reason;
@@ -2849,7 +2856,7 @@ static int files_transaction_prepare(struct ref_store *ref_store,
for (i = 0; i < transaction->nr; i++) { for (i = 0; i < transaction->nr; i++) {
struct ref_update *update = transaction->updates[i]; struct ref_update *update = transaction->updates[i];
ret = lock_ref_for_update(refs, update, transaction, ret = lock_ref_for_update(refs, update, i, transaction,
head_ref, &refnames_to_check, head_ref, &refnames_to_check,
err); err);
if (ret) { if (ret) {
@@ -2905,7 +2912,8 @@ static int files_transaction_prepare(struct ref_store *ref_store,
* So instead, we accept the race for now. * So instead, we accept the race for now.
*/ */
if (refs_verify_refnames_available(refs->packed_ref_store, &refnames_to_check, if (refs_verify_refnames_available(refs->packed_ref_store, &refnames_to_check,
&transaction->refnames, NULL, 0, err)) { &transaction->refnames, NULL, transaction,
0, err)) {
ret = REF_TRANSACTION_ERROR_NAME_CONFLICT; ret = REF_TRANSACTION_ERROR_NAME_CONFLICT;
goto cleanup; goto cleanup;
} }
@@ -2951,7 +2959,7 @@ static int files_transaction_prepare(struct ref_store *ref_store,
cleanup: cleanup:
free(head_ref); free(head_ref);
string_list_clear(&refnames_to_check, 0); string_list_clear(&refnames_to_check, 1);
if (ret) if (ret)
files_transaction_cleanup(refs, transaction); files_transaction_cleanup(refs, transaction);
@@ -3097,7 +3105,8 @@ static int files_transaction_finish_initial(struct files_ref_store *refs,
} }
if (refs_verify_refnames_available(&refs->base, &refnames_to_check, if (refs_verify_refnames_available(&refs->base, &refnames_to_check,
&affected_refnames, NULL, 1, err)) { &affected_refnames, NULL, transaction,
1, err)) {
packed_refs_unlock(refs->packed_ref_store); packed_refs_unlock(refs->packed_ref_store);
ret = REF_TRANSACTION_ERROR_NAME_CONFLICT; ret = REF_TRANSACTION_ERROR_NAME_CONFLICT;
goto cleanup; goto cleanup;

View File

@@ -806,4 +806,20 @@ enum ref_transaction_error ref_update_check_old_target(const char *referent,
*/ */
int ref_update_expects_existing_old_ref(struct ref_update *update); int ref_update_expects_existing_old_ref(struct ref_update *update);
/*
* Same as `refs_verify_refname_available()`, but checking for a list of
* refnames instead of only a single item. This is more efficient in the case
* where one needs to check multiple refnames.
*
* If using batched updates, then individual updates are marked rejected,
* reference backends are then in charge of not committing those updates.
*/
enum ref_transaction_error refs_verify_refnames_available(struct ref_store *refs,
const struct string_list *refnames,
const struct string_list *extras,
const struct string_list *skip,
struct ref_transaction *transaction,
unsigned int initial_transaction,
struct strbuf *err);
#endif /* REFS_REFS_INTERNAL_H */ #endif /* REFS_REFS_INTERNAL_H */

View File

@@ -1074,6 +1074,7 @@ static enum ref_transaction_error prepare_single_update(struct reftable_ref_stor
struct ref_transaction *transaction, struct ref_transaction *transaction,
struct reftable_backend *be, struct reftable_backend *be,
struct ref_update *u, struct ref_update *u,
size_t update_idx,
struct string_list *refnames_to_check, struct string_list *refnames_to_check,
unsigned int head_type, unsigned int head_type,
struct strbuf *head_referent, struct strbuf *head_referent,
@@ -1149,6 +1150,7 @@ static enum ref_transaction_error prepare_single_update(struct reftable_ref_stor
if (ret < 0) if (ret < 0)
return REF_TRANSACTION_ERROR_GENERIC; return REF_TRANSACTION_ERROR_GENERIC;
if (ret > 0 && !ref_update_expects_existing_old_ref(u)) { if (ret > 0 && !ref_update_expects_existing_old_ref(u)) {
struct string_list_item *item;
/* /*
* The reference does not exist, and we either have no * The reference does not exist, and we either have no
* old object ID or expect the reference to not exist. * old object ID or expect the reference to not exist.
@@ -1158,7 +1160,9 @@ static enum ref_transaction_error prepare_single_update(struct reftable_ref_stor
* can output a proper error message instead of failing * can output a proper error message instead of failing
* at a later point. * at a later point.
*/ */
string_list_append(refnames_to_check, u->refname); item = string_list_append(refnames_to_check, u->refname);
item->util = xmalloc(sizeof(update_idx));
memcpy(item->util, &update_idx, sizeof(update_idx));
/* /*
* There is no need to write the reference deletion * There is no need to write the reference deletion
@@ -1368,7 +1372,7 @@ static int reftable_be_transaction_prepare(struct ref_store *ref_store,
for (i = 0; i < transaction->nr; i++) { for (i = 0; i < transaction->nr; i++) {
ret = prepare_single_update(refs, tx_data, transaction, be, ret = prepare_single_update(refs, tx_data, transaction, be,
transaction->updates[i], transaction->updates[i], i,
&refnames_to_check, head_type, &refnames_to_check, head_type,
&head_referent, &referent, err); &head_referent, &referent, err);
if (ret) { if (ret) {
@@ -1384,6 +1388,7 @@ static int reftable_be_transaction_prepare(struct ref_store *ref_store,
ret = refs_verify_refnames_available(ref_store, &refnames_to_check, ret = refs_verify_refnames_available(ref_store, &refnames_to_check,
&transaction->refnames, NULL, &transaction->refnames, NULL,
transaction,
transaction->flags & REF_TRANSACTION_FLAG_INITIAL, transaction->flags & REF_TRANSACTION_FLAG_INITIAL,
err); err);
if (ret < 0) if (ret < 0)
@@ -1402,7 +1407,7 @@ done:
} }
strbuf_release(&referent); strbuf_release(&referent);
strbuf_release(&head_referent); strbuf_release(&head_referent);
string_list_clear(&refnames_to_check, 0); string_list_clear(&refnames_to_check, 1);
return ret; return ret;
} }