create_symref: use existing ref-lock code
authorJeff King <peff@peff.net>
Tue, 29 Dec 2015 05:57:01 +0000 (00:57 -0500)
committerJunio C Hamano <gitster@pobox.com>
Tue, 29 Dec 2015 18:33:31 +0000 (10:33 -0800)
The create_symref() function predates the existence of
"struct lock_file", let alone the more recent "struct
ref_lock". Instead, it just does its own manual dot-locking.
Besides being more code, this has a few downsides:

- if git is interrupted while holding the lock, we don't
clean up the lockfile

- we don't do the usual directory/filename conflict check.
So you can sometimes create a symref "refs/heads/foo/bar",
even if "refs/heads/foo" exists (namely, if the refs are
packed and we do not hit the d/f conflict in the
filesystem).

This patch refactors create_symref() to use the "struct
ref_lock" interface, which handles both of these things.
There are a few bonus cleanups that come along with it:

- we leaked ref_path in some error cases

- the symref contents were stored in a fixed-size buffer,
putting an artificial (albeit large) limitation on the
length of the refname. We now write through fprintf, and
handle refnames of any size.

- we called adjust_shared_perm only after the file was
renamed into place, creating a potential race with
readers in a shared repository. The lockfile code now
handles this when creating the lockfile, making it
atomic.

- the legacy prefer_symlink_refs path did not do any
locking at all. Admittedly, it is not atomic from a
reader's perspective (as it unlinks and re-creates the
symlink to overwrite), but at least it cannot conflict
with other writers now.

- the result of this patch is hopefully more readable. It
eliminates three goto labels. Two were for error checking
that is now simplified, and the third was to reach shared
code that has been pulled into its own function.

Signed-off-by: Jeff King <peff@peff.net>
Reviewed-by: Michael Haggerty <mhagger@alum.mit.edu>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
refs/files-backend.c
t/t1401-symbolic-ref.sh
index d6f91236359a7f97b3c45f3ab62ae8e69d5ffb80..3d1994debd4214b8e3a257576d7f8b09dbdb06f2 100644 (file)
@@ -2811,74 +2811,73 @@ static int commit_ref_update(struct ref_lock *lock,
        return 0;
 }
 
-int create_symref(const char *refname, const char *target, const char *logmsg)
+static int create_ref_symlink(struct ref_lock *lock, const char *target)
 {
-       char *lockpath = NULL;
-       char buf[1000];
-       int fd, len, written;
-       char *ref_path = git_pathdup("%s", refname);
-       unsigned char old_sha1[20], new_sha1[20];
-       struct strbuf err = STRBUF_INIT;
-
-       if (logmsg && read_ref(refname, old_sha1))
-               hashclr(old_sha1);
-
-       if (safe_create_leading_directories(ref_path) < 0)
-               return error("unable to create directory for %s", ref_path);
-
+       int ret = -1;
 #ifndef NO_SYMLINK_HEAD
-       if (prefer_symlink_refs) {
-               unlink(ref_path);
-               if (!symlink(target, ref_path))
-                       goto done;
+       char *ref_path = get_locked_file_path(lock->lk);
+       unlink(ref_path);
+       ret = symlink(target, ref_path);
+       free(ref_path);
+
+       if (ret)
                fprintf(stderr, "no symlink - falling back to symbolic ref\n");
-       }
 #endif
+       return ret;
+}
 
-       len = snprintf(buf, sizeof(buf), "ref: %s\n", target);
-       if (sizeof(buf) <= len) {
-               error("refname too long: %s", target);
-               goto error_free_return;
-       }
-       lockpath = mkpathdup("%s.lock", ref_path);
-       fd = open(lockpath, O_CREAT | O_EXCL | O_WRONLY, 0666);
-       if (fd < 0) {
-               error("Unable to open %s for writing", lockpath);
-               goto error_free_return;
-       }
-       written = write_in_full(fd, buf, len);
-       if (close(fd) != 0 || written != len) {
-               error("Unable to write to %s", lockpath);
-               goto error_unlink_return;
-       }
-       if (rename(lockpath, ref_path) < 0) {
-               error("Unable to create %s", ref_path);
-               goto error_unlink_return;
-       }
-       if (adjust_shared_perm(ref_path)) {
-               error("Unable to fix permissions on %s", lockpath);
-       error_unlink_return:
-               unlink_or_warn(lockpath);
-       error_free_return:
-               free(lockpath);
-               free(ref_path);
-               return -1;
-       }
-       free(lockpath);
-
-#ifndef NO_SYMLINK_HEAD
-       done:
-#endif
+static void update_symref_reflog(struct ref_lock *lock, const char *refname,
+                                const char *target, const char *logmsg)
+{
+       struct strbuf err = STRBUF_INIT;
+       unsigned char new_sha1[20];
        if (logmsg && !read_ref(target, new_sha1) &&
-               log_ref_write(refname, old_sha1, new_sha1, logmsg, 0, &err)) {
+           log_ref_write(refname, lock->old_oid.hash, new_sha1, logmsg, 0, &err)) {
                error("%s", err.buf);
                strbuf_release(&err);
        }
+}
 
-       free(ref_path);
+static int create_symref_locked(struct ref_lock *lock, const char *refname,
+                               const char *target, const char *logmsg)
+{
+       if (prefer_symlink_refs && !create_ref_symlink(lock, target)) {
+               update_symref_reflog(lock, refname, target, logmsg);
+               return 0;
+       }
+
+       if (!fdopen_lock_file(lock->lk, "w"))
+               return error("unable to fdopen %s: %s",
+                            lock->lk->tempfile.filename.buf, strerror(errno));
+
+       /* no error check; commit_ref will check ferror */
+       fprintf(lock->lk->tempfile.fp, "ref: %s\n", target);
+       if (commit_ref(lock) < 0)
+               return error("unable to write symref for %s: %s", refname,
+                            strerror(errno));
+       update_symref_reflog(lock, refname, target, logmsg);
        return 0;
 }
 
+int create_symref(const char *refname, const char *target, const char *logmsg)
+{
+       struct strbuf err = STRBUF_INIT;
+       struct ref_lock *lock;
+       int ret;
+
+       lock = lock_ref_sha1_basic(refname, NULL, NULL, NULL, REF_NODEREF, NULL,
+                                  &err);
+       if (!lock) {
+               error("%s", err.buf);
+               strbuf_release(&err);
+               return -1;
+       }
+
+       ret = create_symref_locked(lock, refname, target, logmsg);
+       unlock_ref(lock);
+       return ret;
+}
+
 int reflog_exists(const char *refname)
 {
        struct stat st;
index 1f0dff3a0b1cedb95b900a01de972e8a02e29f90..5db876c6a1e01b4a3e9bdc11988b8589225662ec 100755 (executable)
@@ -114,4 +114,12 @@ test_expect_success 'symbolic-ref writes reflog entry' '
        test_cmp expect actual
 '
 
+test_expect_success 'symbolic-ref does not create ref d/f conflicts' '
+       git checkout -b df &&
+       test_commit df &&
+       test_must_fail git symbolic-ref refs/heads/df/conflict refs/heads/df &&
+       git pack-refs --all --prune &&
+       test_must_fail git symbolic-ref refs/heads/df/conflict refs/heads/df
+'
+
 test_done