From: Ronnie Sahlberg <sahlberg@google.com>
To: git@vger.kernel.org
Cc: Ronnie Sahlberg <sahlberg@google.com>
Subject: [PATCH v17 32/48] receive-pack.c: use a reference transaction for updating the refs
Date: Mon, 16 Jun 2014 11:04:03 -0700 [thread overview]
Message-ID: <1402941859-29354-33-git-send-email-sahlberg@google.com> (raw)
In-Reply-To: <1402941859-29354-1-git-send-email-sahlberg@google.com>
Wrap all the ref updates inside a transaction.
Signed-off-by: Ronnie Sahlberg <sahlberg@google.com>
---
builtin/receive-pack.c | 96 +++++++++++++++++++++++++++++++++-----------------
1 file changed, 63 insertions(+), 33 deletions(-)
diff --git a/builtin/receive-pack.c b/builtin/receive-pack.c
index c323081..b51f8ae 100644
--- a/builtin/receive-pack.c
+++ b/builtin/receive-pack.c
@@ -194,7 +194,7 @@ static void write_head_info(void)
struct command {
struct command *next;
- const char *error_string;
+ char *error_string;
unsigned int skip_update:1,
did_not_exist:1;
int index;
@@ -468,19 +468,18 @@ static int update_shallow_ref(struct command *cmd, struct shallow_info *si)
return 0;
}
-static const char *update(struct command *cmd, struct shallow_info *si)
+static char *update(struct command *cmd, struct shallow_info *si)
{
const char *name = cmd->ref_name;
struct strbuf namespaced_name_buf = STRBUF_INIT;
const char *namespaced_name;
unsigned char *old_sha1 = cmd->old_sha1;
unsigned char *new_sha1 = cmd->new_sha1;
- struct ref_lock *lock;
/* only refs/... are allowed */
if (!starts_with(name, "refs/") || check_refname_format(name + 5, 0)) {
rp_error("refusing to create funny ref '%s' remotely", name);
- return "funny refname";
+ return xstrdup("funny refname");
}
strbuf_addf(&namespaced_name_buf, "%s%s", get_git_namespace(), name);
@@ -498,20 +497,20 @@ static const char *update(struct command *cmd, struct shallow_info *si)
rp_error("refusing to update checked out branch: %s", name);
if (deny_current_branch == DENY_UNCONFIGURED)
refuse_unconfigured_deny();
- return "branch is currently checked out";
+ return xstrdup("branch is currently checked out");
}
}
if (!is_null_sha1(new_sha1) && !has_sha1_file(new_sha1)) {
error("unpack should have generated %s, "
"but I can't find it!", sha1_to_hex(new_sha1));
- return "bad pack";
+ return xstrdup("bad pack");
}
if (!is_null_sha1(old_sha1) && is_null_sha1(new_sha1)) {
if (deny_deletes && starts_with(name, "refs/heads/")) {
rp_error("denying ref deletion for %s", name);
- return "deletion prohibited";
+ return xstrdup("deletion prohibited");
}
if (!strcmp(namespaced_name, head_name)) {
@@ -526,7 +525,7 @@ static const char *update(struct command *cmd, struct shallow_info *si)
if (deny_delete_current == DENY_UNCONFIGURED)
refuse_unconfigured_deny_delete_current();
rp_error("refusing to delete the current branch: %s", name);
- return "deletion of the current branch prohibited";
+ return xstrdup("deletion of the current branch prohibited");
}
}
}
@@ -544,19 +543,19 @@ static const char *update(struct command *cmd, struct shallow_info *si)
old_object->type != OBJ_COMMIT ||
new_object->type != OBJ_COMMIT) {
error("bad sha1 objects for %s", name);
- return "bad ref";
+ return xstrdup("bad ref");
}
old_commit = (struct commit *)old_object;
new_commit = (struct commit *)new_object;
if (!in_merge_bases(old_commit, new_commit)) {
rp_error("denying non-fast-forward %s"
" (you should pull first)", name);
- return "non-fast-forward";
+ return xstrdup("non-fast-forward");
}
}
if (run_update_hook(cmd)) {
rp_error("hook declined to update %s", name);
- return "hook declined";
+ return xstrdup("hook declined");
}
if (is_null_sha1(new_sha1)) {
@@ -571,24 +570,32 @@ static const char *update(struct command *cmd, struct shallow_info *si)
}
if (delete_ref(namespaced_name, old_sha1, 0)) {
rp_error("failed to delete %s", name);
- return "failed to delete";
+ return xstrdup("failed to delete");
}
return NULL; /* good */
}
else {
+ struct strbuf err = STRBUF_INIT;
+ struct ref_transaction *transaction;
+
if (shallow_update && si->shallow_ref[cmd->index] &&
update_shallow_ref(cmd, si))
- return "shallow error";
-
- lock = lock_any_ref_for_update(namespaced_name, old_sha1,
- 0, NULL);
- if (!lock) {
- rp_error("failed to lock %s", name);
- return "failed to lock";
- }
- if (write_ref_sha1(lock, new_sha1, "push")) {
- return "failed to write"; /* error() already called */
+ return xstrdup("shallow error");
+
+ transaction = ref_transaction_begin(&err);
+ if (!transaction ||
+ ref_transaction_update(transaction, namespaced_name,
+ new_sha1, old_sha1, 0, 1, &err) ||
+ ref_transaction_commit(transaction, "push", &err)) {
+ char *str = strbuf_detach(&err, NULL);
+ ref_transaction_free(transaction);
+
+ rp_error("%s", str);
+ return str;
}
+
+ ref_transaction_free(transaction);
+ strbuf_release(&err);
return NULL; /* good */
}
}
@@ -647,6 +654,9 @@ static void check_aliased_update(struct command *cmd, struct string_list *list)
char cmd_oldh[41], cmd_newh[41], dst_oldh[41], dst_newh[41];
int flag;
+ if (cmd->error_string)
+ die("BUG: check_alised_update called with failed cmd");
+
strbuf_addf(&buf, "%s%s", get_git_namespace(), cmd->ref_name);
dst_name = resolve_ref_unsafe(buf.buf, sha1, 0, &flag);
strbuf_release(&buf);
@@ -658,7 +668,7 @@ static void check_aliased_update(struct command *cmd, struct string_list *list)
if (!dst_name) {
rp_error("refusing update to broken symref '%s'", cmd->ref_name);
cmd->skip_update = 1;
- cmd->error_string = "broken symref";
+ cmd->error_string = xstrdup("broken symref");
return;
}
@@ -684,8 +694,9 @@ static void check_aliased_update(struct command *cmd, struct string_list *list)
cmd->ref_name, cmd_oldh, cmd_newh,
dst_cmd->ref_name, dst_oldh, dst_newh);
- cmd->error_string = dst_cmd->error_string =
- "inconsistent aliased update";
+ cmd->error_string = xstrdup("inconsistent aliased update");
+ free(dst_cmd->error_string);
+ dst_cmd->error_string = xstrdup("inconsistent aliased update");
}
static void check_aliased_updates(struct command *commands)
@@ -733,7 +744,9 @@ static void set_connectivity_errors(struct command *commands,
if (!check_everything_connected(command_singleton_iterator,
0, &singleton))
continue;
- cmd->error_string = "missing necessary objects";
+ if (cmd->error_string) /* can't happen */
+ continue;
+ cmd->error_string = xstrdup("missing necessary objects");
}
}
@@ -770,9 +783,9 @@ static void reject_updates_to_hidden(struct command *commands)
if (cmd->error_string || !ref_is_hidden(cmd->ref_name))
continue;
if (is_null_sha1(cmd->new_sha1))
- cmd->error_string = "deny deleting a hidden ref";
+ cmd->error_string = xstrdup("deny deleting a hidden ref");
else
- cmd->error_string = "deny updating a hidden ref";
+ cmd->error_string = xstrdup("deny updating a hidden ref");
}
}
@@ -786,8 +799,11 @@ static void execute_commands(struct command *commands,
struct iterate_data data;
if (unpacker_error) {
- for (cmd = commands; cmd; cmd = cmd->next)
- cmd->error_string = "unpacker error";
+ for (cmd = commands; cmd; cmd = cmd->next) {
+ if (cmd->error_string) /* can't happen */
+ continue;
+ cmd->error_string = xstrdup("unpacker error");
+ }
return;
}
@@ -800,8 +816,9 @@ static void execute_commands(struct command *commands,
if (run_receive_hook(commands, "pre-receive", 0)) {
for (cmd = commands; cmd; cmd = cmd->next) {
- if (!cmd->error_string)
- cmd->error_string = "pre-receive hook declined";
+ if (cmd->error_string)
+ continue;
+ cmd->error_string = xstrdup("pre-receive hook declined");
}
return;
}
@@ -1079,7 +1096,8 @@ static void update_shallow_info(struct command *commands,
if (is_null_sha1(cmd->new_sha1))
continue;
if (ref_status[cmd->index]) {
- cmd->error_string = "shallow update not allowed";
+ free(cmd->error_string);
+ cmd->error_string = xstrdup("shallow update not allowed");
cmd->skip_update = 1;
}
}
@@ -1120,6 +1138,17 @@ static int delete_only(struct command *commands)
return 1;
}
+static void free_commands(struct command *commands)
+{
+ while (commands) {
+ struct command *next = commands->next;
+
+ free(commands->error_string);
+ free(commands);
+ commands = next;
+ }
+}
+
int cmd_receive_pack(int argc, const char **argv, const char *prefix)
{
int advertise_refs = 0;
@@ -1215,5 +1244,6 @@ int cmd_receive_pack(int argc, const char **argv, const char *prefix)
packet_flush(1);
sha1_array_clear(&shallow);
sha1_array_clear(&ref);
+ free_commands(commands);
return 0;
}
--
2.0.0.282.g3799eda.dirty
next prev parent reply other threads:[~2014-06-16 18:07 UTC|newest]
Thread overview: 53+ messages / expand[flat|nested] mbox.gz Atom feed top
2014-06-16 18:03 [PATCH v17 00/48] Use ref transactions Ronnie Sahlberg
2014-06-16 18:03 ` [PATCH v17 01/48] refs.c: remove ref_transaction_rollback Ronnie Sahlberg
2014-06-16 18:03 ` [PATCH v17 02/48] refs.c: ref_transaction_commit should not free the transaction Ronnie Sahlberg
2014-06-16 18:03 ` [PATCH v17 03/48] refs.c: constify the sha arguments for ref_transaction_create|delete|update Ronnie Sahlberg
2014-06-16 18:03 ` [PATCH v17 04/48] refs.c: allow passing NULL to ref_transaction_free Ronnie Sahlberg
2014-06-16 18:03 ` [PATCH v17 05/48] refs.c: add a strbuf argument to ref_transaction_commit for error logging Ronnie Sahlberg
2014-06-16 18:03 ` [PATCH v17 06/48] lockfile.c: add a new public function unable_to_lock_message Ronnie Sahlberg
2014-06-16 18:03 ` [PATCH v17 07/48] lockfile.c: make lock_file return a meaningful errno on failurei Ronnie Sahlberg
2014-06-16 18:03 ` [PATCH v17 08/48] refs.c: add an err argument to repack_without_refs Ronnie Sahlberg
2014-06-16 18:03 ` [PATCH v17 09/48] refs.c: make sure log_ref_setup returns a meaningful errno Ronnie Sahlberg
2014-06-16 18:03 ` [PATCH v17 10/48] refs.c: verify_lock should set errno to something meaningful Ronnie Sahlberg
2014-06-16 18:03 ` [PATCH v17 11/48] refs.c: make remove_empty_directories alwasy set errno to something sane Ronnie Sahlberg
2014-06-16 18:03 ` [PATCH v17 12/48] refs.c: commit_packed_refs to return a meaningful errno on failure Ronnie Sahlberg
2014-06-16 18:03 ` [PATCH v17 13/48] refs.c: make resolve_ref_unsafe set errno to something meaningful on error Ronnie Sahlberg
2014-06-16 18:03 ` [PATCH v17 14/48] refs.c: log_ref_write should try to return meaningful errno Ronnie Sahlberg
2014-06-16 18:03 ` [PATCH v17 15/48] refs.c: make ref_update_reject_duplicates take a strbuf argument for errors Ronnie Sahlberg
2014-06-16 18:03 ` [PATCH v17 16/48] refs.c: add an err argument to delete_ref_loose Ronnie Sahlberg
2014-06-17 0:22 ` Junio C Hamano
2014-06-17 14:34 ` Ronnie Sahlberg
2014-06-17 15:55 ` Ronnie Sahlberg
2014-06-17 17:23 ` Junio C Hamano
2014-06-16 18:03 ` [PATCH v17 17/48] refs.c: make update_ref_write update a strbuf on failure Ronnie Sahlberg
2014-06-16 18:03 ` [PATCH v17 18/48] update-ref: use err argument to get error from ref_transaction_commit Ronnie Sahlberg
2014-06-16 18:03 ` [PATCH v17 19/48] refs.c: remove the onerr argument to ref_transaction_commit Ronnie Sahlberg
2014-06-16 18:03 ` [PATCH v17 20/48] refs.c: change ref_transaction_update() to do error checking and return status Ronnie Sahlberg
2014-06-16 18:03 ` [PATCH v17 21/48] refs.c: change ref_transaction_create " Ronnie Sahlberg
2014-06-16 18:03 ` [PATCH v17 22/48] refs.c: update ref_transaction_delete to check for error " Ronnie Sahlberg
2014-06-16 18:03 ` [PATCH v17 23/48] refs.c: make ref_transaction_begin take an err argument Ronnie Sahlberg
2014-06-16 18:03 ` [PATCH v17 24/48] refs.c: add transaction.status and track OPEN/CLOSED/ERROR Ronnie Sahlberg
2014-06-16 18:03 ` [PATCH v17 25/48] tag.c: use ref transactions when doing updates Ronnie Sahlberg
2014-06-16 18:03 ` [PATCH v17 26/48] replace.c: use the ref transaction functions for updates Ronnie Sahlberg
2014-06-16 18:03 ` [PATCH v17 27/48] commit.c: use ref transactions " Ronnie Sahlberg
2014-06-16 18:03 ` [PATCH v17 28/48] sequencer.c: use ref transactions for all ref updates Ronnie Sahlberg
2014-06-16 18:04 ` [PATCH v17 29/48] fast-import.c: change update_branch to use ref transactions Ronnie Sahlberg
2014-06-16 18:04 ` [PATCH v17 30/48] branch.c: use ref transaction for all ref updates Ronnie Sahlberg
2014-06-16 18:04 ` [PATCH v17 31/48] refs.c: change update_ref to use a transaction Ronnie Sahlberg
2014-06-16 18:04 ` Ronnie Sahlberg [this message]
2014-06-16 18:04 ` [PATCH v17 33/48] fast-import.c: use a ref transaction when dumping tags Ronnie Sahlberg
2014-06-16 18:04 ` [PATCH v17 34/48] walker.c: use ref transaction for ref updates Ronnie Sahlberg
2014-06-16 18:04 ` [PATCH v17 35/48] refs.c: make lock_ref_sha1 static Ronnie Sahlberg
2014-06-16 18:04 ` [PATCH v17 36/48] refs.c: remove the update_ref_lock function Ronnie Sahlberg
2014-06-16 18:04 ` [PATCH v17 37/48] refs.c: remove the update_ref_write function Ronnie Sahlberg
2014-06-16 18:04 ` [PATCH v17 38/48] refs.c: remove lock_ref_sha1 Ronnie Sahlberg
2014-06-16 18:04 ` [PATCH v17 39/48] refs.c: make prune_ref use a transaction to delete the ref Ronnie Sahlberg
2014-06-16 18:04 ` [PATCH v17 40/48] refs.c: make delete_ref use a transaction Ronnie Sahlberg
2014-06-16 18:04 ` [PATCH v17 41/48] refs.c: pass the ref log message to _create/delete/update instead of _commit Ronnie Sahlberg
2014-06-16 18:04 ` [PATCH v17 42/48] refs.c: pass NULL as *flags to read_ref_full Ronnie Sahlberg
2014-06-16 18:04 ` [PATCH v17 43/48] refs.c: move the check for valid refname to lock_ref_sha1_basic Ronnie Sahlberg
2014-06-16 18:04 ` [PATCH v17 44/48] refs.c: call lock_ref_sha1_basic directly from commit Ronnie Sahlberg
2014-06-16 18:04 ` [PATCH v17 45/48] refs.c: pass a skip list to name_conflict_fn Ronnie Sahlberg
2014-06-16 18:04 ` [PATCH v17 46/48] refs.c: propagate any errno==ENOTDIR from _commit back to the callers Ronnie Sahlberg
2014-06-16 18:04 ` [PATCH v17 47/48] fetch.c: change s_update_ref to use a ref transaction Ronnie Sahlberg
2014-06-16 18:04 ` [PATCH v17 48/48] refs.c: make write_ref_sha1 static Ronnie Sahlberg
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
List information: http://vger.kernel.org/majordomo-info.html
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=1402941859-29354-33-git-send-email-sahlberg@google.com \
--to=sahlberg@google.com \
--cc=git@vger.kernel.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
Code repositories for project(s) associated with this public inbox
https://80x24.org/mirrors/git.git
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for read-only IMAP folder(s) and NNTP newsgroup(s).