@@ -1204,6 +1204,7 @@ void ref_transaction_free(struct ref_transaction *transaction)
free(transaction->updates[i]->committer_info);
free((char *)transaction->updates[i]->new_target);
free((char *)transaction->updates[i]->old_target);
+ strbuf_release(&transaction->updates[i]->rejection_err);
free(transaction->updates[i]);
}
string_list_clear(&transaction->refnames, 0);
@@ -1211,6 +1212,14 @@ void ref_transaction_free(struct ref_transaction *transaction)
free(transaction);
}
+void ref_transaction_add_rejection(struct ref_transaction *transaction,
+ size_t update_idx, struct strbuf *err)
+{
+ struct ref_update *update = transaction->updates[update_idx];
+ update->rejected = 1;
+ strbuf_addbuf(&update->rejection_err, err);
+}
+
struct ref_update *ref_transaction_add_update(
struct ref_transaction *transaction,
const char *refname, unsigned int flags,
@@ -1237,6 +1246,8 @@ struct ref_update *ref_transaction_add_update(
update->flags = flags;
+ strbuf_init(&update->rejection_err, 0);
+
update->new_target = xstrdup_or_null(new_target);
update->old_target = xstrdup_or_null(old_target);
if ((flags & REF_HAVE_NEW) && new_oid)
@@ -2676,6 +2687,27 @@ void ref_transaction_for_each_queued_update(struct ref_transaction *transaction,
}
}
+void ref_transaction_for_each_rejected_update(struct ref_transaction *transaction,
+ ref_transaction_for_each_rejected_update_fn cb,
+ void *cb_data)
+{
+ if (!(transaction->flags & REF_TRANSACTION_ALLOW_PARTIAL))
+ return;
+
+ for (size_t i = 0; i < transaction->nr; i++) {
+ struct ref_update *update = transaction->updates[i];
+
+ if (!update->rejected)
+ continue;
+
+ cb(update->refname,
+ (update->flags & REF_HAVE_OLD) ? &update->old_oid : NULL,
+ (update->flags & REF_HAVE_NEW) ? &update->new_oid : NULL,
+ update->old_target, update->new_target,
+ &update->rejection_err, cb_data);
+ }
+}
+
int refs_delete_refs(struct ref_store *refs, const char *logmsg,
struct string_list *refnames, unsigned int flags)
{
@@ -638,6 +638,13 @@ enum ref_transaction_flag {
* either be absent or null_oid.
*/
REF_TRANSACTION_FLAG_INITIAL = (1 << 0),
+
+ /*
+ * The transaction mechanism by default fails all updates if any conflict
+ * is detected. This flag allows transactions to partially apply updates
+ * while rejecting updates which do not match the expected state.
+ */
+ REF_TRANSACTION_ALLOW_PARTIAL = (1 << 1),
};
/*
@@ -889,6 +896,21 @@ void ref_transaction_for_each_queued_update(struct ref_transaction *transaction,
ref_transaction_for_each_queued_update_fn cb,
void *cb_data);
+/*
+ * Execute the given callback function for each of the reference updates which
+ * have been rejected in the given transaction.
+ */
+typedef void ref_transaction_for_each_rejected_update_fn(const char *refname,
+ const struct object_id *old_oid,
+ const struct object_id *new_oid,
+ const char *old_target,
+ const char *new_target,
+ const struct strbuf *reason,
+ void *cb_data);
+void ref_transaction_for_each_rejected_update(struct ref_transaction *transaction,
+ ref_transaction_for_each_rejected_update_fn cb,
+ void *cb_data);
+
/*
* Free `*transaction` and all associated data.
*/
@@ -2852,8 +2852,18 @@ static int files_transaction_prepare(struct ref_store *ref_store,
ret = lock_ref_for_update(refs, update, transaction,
head_ref, err);
- if (ret)
+ if (ret) {
+ if (transaction->flags & REF_TRANSACTION_ALLOW_PARTIAL) {
+ ref_transaction_add_rejection(transaction, i, err);
+
+ strbuf_setlen(err, 0);
+ ret = 0;
+
+ continue;
+ }
goto cleanup;
+ }
+
if (update->flags & REF_DELETING &&
!(update->flags & REF_LOG_ONLY) &&
@@ -1313,9 +1313,10 @@ static int packed_ref_store_remove_on_disk(struct ref_store *ref_store,
* remain locked when it is done.
*/
static int write_with_updates(struct packed_ref_store *refs,
- struct string_list *updates,
+ struct ref_transaction *transaction,
struct strbuf *err)
{
+ struct string_list *updates = &transaction->refnames;
struct ref_iterator *iter = NULL;
size_t i;
int ok;
@@ -1393,6 +1394,13 @@ static int write_with_updates(struct packed_ref_store *refs,
strbuf_addf(err, "cannot update ref '%s': "
"reference already exists",
update->refname);
+
+ if (transaction->flags & REF_TRANSACTION_ALLOW_PARTIAL) {
+ ref_transaction_add_rejection(transaction, i, err);
+ strbuf_setlen(err, 0);
+ continue;
+ }
+
goto error;
} else if (!oideq(&update->old_oid, iter->oid)) {
strbuf_addf(err, "cannot update ref '%s': "
@@ -1400,6 +1408,13 @@ static int write_with_updates(struct packed_ref_store *refs,
update->refname,
oid_to_hex(iter->oid),
oid_to_hex(&update->old_oid));
+
+ if (transaction->flags & REF_TRANSACTION_ALLOW_PARTIAL) {
+ ref_transaction_add_rejection(transaction, i, err);
+ strbuf_setlen(err, 0);
+ continue;
+ }
+
goto error;
}
}
@@ -1434,6 +1449,13 @@ static int write_with_updates(struct packed_ref_store *refs,
"reference is missing but expected %s",
update->refname,
oid_to_hex(&update->old_oid));
+
+ if (transaction->flags & REF_TRANSACTION_ALLOW_PARTIAL) {
+ ref_transaction_add_rejection(transaction, i, err);
+ strbuf_setlen(err, 0);
+ continue;
+ }
+
goto error;
}
}
@@ -1657,7 +1679,7 @@ static int packed_transaction_prepare(struct ref_store *ref_store,
data->own_lock = 1;
}
- if (write_with_updates(refs, &transaction->refnames, err))
+ if (write_with_updates(refs, transaction, err))
goto failure;
transaction->state = REF_TRANSACTION_PREPARED;
@@ -3,6 +3,7 @@
#include "refs.h"
#include "iterator.h"
+#include "strbuf.h"
#include "string-list.h"
struct fsck_options;
@@ -123,6 +124,13 @@ struct ref_update {
*/
unsigned int index;
+ /*
+ * Used in partial transactions to mark a given update as rejected,
+ * with rejection reason.
+ */
+ unsigned int rejected;
+ struct strbuf rejection_err;
+
/*
* If this ref_update was split off of a symref update via
* split_symref_update(), then this member points at that
@@ -142,6 +150,13 @@ int refs_read_raw_ref(struct ref_store *ref_store, const char *refname,
struct object_id *oid, struct strbuf *referent,
unsigned int *type, int *failure_errno);
+/*
+ * Mark a given update as rejected with a given reason. To be used in conjuction
+ * with the `REF_TRANSACTION_ALLOW_PARTIAL` flag to allow partial transactions.
+ */
+void ref_transaction_add_rejection(struct ref_transaction *transaction,
+ size_t update_idx, struct strbuf *err);
+
/*
* Add a ref_update with the specified properties to transaction, and
* return a pointer to the new object. This function does not verify
@@ -1364,8 +1364,18 @@ static int reftable_be_transaction_prepare(struct ref_store *ref_store,
transaction, be,
transaction->updates[i], head_type,
&head_referent, &referent, err);
- if (ret)
+
+ if (ret) {
+ if (transaction->flags & REF_TRANSACTION_ALLOW_PARTIAL) {
+ ref_transaction_add_rejection(transaction, i, err);
+
+ strbuf_setlen(err, 0);
+ ret = 0;
+
+ continue;
+ }
goto done;
+ }
}
transaction->backend_data = tx_data;
Git's reference transactions are all-or-nothing: either all updates succeed, or none do. While this atomic behavior is generally desirable, it can be suboptimal when using the reftable backend, where batching multiple reference updates into a single transaction is more efficient than performing them sequentially. Introduce partial transaction support through a new flag `REF_TRANSACTION_ALLOW_PARTIAL`. When this flag is set, individual reference updates that would normally fail the entire transaction are instead marked as rejected while allowing other updates to proceed. This provides more flexibility while maintaining transactional integrity where needed. The implementation introduces several key components: - Add 'rejected' and 'rejection_err' fields to struct `ref_update` to track failed updates and their failure reasons. - Modify reference backends (files, packed, reftable) to handle partial transactions by using `ref_transaction_add_rejection()` instead of failing the entire transaction when `REF_TRANSACTION_ALLOW_PARTIAL` is set. - Add `ref_transaction_for_each_rejected_update()` to let callers examine which updates were rejected and why. This foundational change enables partial transaction support throughout the reference subsystem. The next commit will expose this capability to users by adding a `--allow-partial` flag to 'git-update-ref(1)', providing both a user-facing feature and a testable implementation. Signed-off-by: Karthik Nayak <karthik.188@gmail.com> --- refs.c | 32 ++++++++++++++++++++++++++++++++ refs.h | 22 ++++++++++++++++++++++ refs/files-backend.c | 12 +++++++++++- refs/packed-backend.c | 26 ++++++++++++++++++++++++-- refs/refs-internal.h | 15 +++++++++++++++ refs/reftable-backend.c | 12 +++++++++++- 6 files changed, 115 insertions(+), 4 deletions(-)