summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/fileops.c39
-rw-r--r--src/fileops.h10
-rw-r--r--src/refs.c196
-rw-r--r--src/refs.h2
4 files changed, 149 insertions, 98 deletions
diff --git a/src/fileops.c b/src/fileops.c
index 52aeb41a3..9f3a65d27 100644
--- a/src/fileops.c
+++ b/src/fileops.c
@@ -231,13 +231,8 @@ int git_futils_direach(
size_t de_len;
int result;
- /* always skip '.' and '..' */
- if (de->d_name[0] == '.') {
- if (de->d_name[1] == '\0')
- continue;
- if (de->d_name[1] == '.' && de->d_name[2] == '\0')
- continue;
- }
+ if (is_dot_or_dotdot(de->d_name))
+ continue;
de_len = strlen(de->d_name);
if (path_sz < wd_len + de_len + 1) {
@@ -306,6 +301,36 @@ int git_futils_mkdir_r(const char *path, int mode)
return GIT_SUCCESS;
}
+static int _rmdir_recurs_foreach(void *force_removal_of_non_empty_directory, char *path)
+{
+ int error = GIT_SUCCESS;
+
+ GIT_UNUSED_ARG(nil)
+
+ error = git_futils_isdir(path);
+ if (error == GIT_SUCCESS) {
+ size_t root_size = strlen(path);
+
+ if ((error = git_futils_direach(path, GIT_PATH_MAX, _rmdir_recurs_foreach, force_removal_of_non_empty_directory)) < GIT_SUCCESS)
+ return git__rethrow(error, "Failed to remove directory `%s`", path);
+
+ path[root_size] = '\0';
+ return p_rmdir(path);
+ }
+
+ if (*(int *)(force_removal_of_non_empty_directory))
+ return p_unlink(path);
+ else
+ return git__rethrow(error, "Failed to remove directory. `%s` is not a directory", path);
+}
+
+int git_futils_rmdir_recurs(const char *path, int force_removal_of_non_empty_directory)
+{
+ char p[GIT_PATH_MAX];
+ strncpy(p, path, GIT_PATH_MAX);
+ return _rmdir_recurs_foreach(&force_removal_of_non_empty_directory, p);
+}
+
int git_futils_cmp_path(const char *name1, int len1, int isdir1,
const char *name2, int len2, int isdir2)
{
diff --git a/src/fileops.h b/src/fileops.h
index 4bfebe9d4..48071d6d1 100644
--- a/src/fileops.h
+++ b/src/fileops.h
@@ -80,6 +80,8 @@ extern int git_futils_mkdir_r(const char *path, int mode);
*/
extern int git_futils_mkpath2file(const char *path);
+extern int git_futils_rmdir_recurs(const char *path, int force_removal_of_non_empty_directory);
+
/**
* Create and open a temporary file with a `_git2_` suffix
*/
@@ -102,6 +104,14 @@ extern int git_futils_mv_withpath(const char *from, const char *to);
*/
extern git_off_t git_futils_filesize(git_file fd);
+/* Taken from git.git */
+GIT_INLINE(int) is_dot_or_dotdot(const char *name)
+{
+ return (name[0] == '.' &&
+ (name[1] == '\0' ||
+ (name[1] == '.' && name[2] == '\0')));
+}
+
/**
* Read-only map all or part of a file into memory.
* When possible this function should favor a virtual memory
diff --git a/src/refs.c b/src/refs.c
index eea96adba..94a7a15c1 100644
--- a/src/refs.c
+++ b/src/refs.c
@@ -1287,137 +1287,151 @@ int git_reference_set_target(git_reference *ref, const char *target)
* Other
*/
-/*
- * Rename a reference
- *
- * If the reference is packed, we need to rewrite the
- * packfile to remove the reference from it and create
- * the reference back as a loose one.
- *
- * If the reference is loose, we just rename it on
- * the filesystem.
- *
- * We also need to re-insert the reference on its corresponding
- * in-memory cache, since the caches are indexed by refname.
- */
int git_reference_rename(git_reference *ref, const char *new_name, int force)
{
int error;
- char *old_name;
- char old_path[GIT_PATH_MAX], new_path[GIT_PATH_MAX], normalized_name[GIT_REFNAME_MAX];
- git_reference *looked_up_ref, *old_ref = NULL;
+ char *old_name = git__strdup(ref->name);
+ char new_path[GIT_PATH_MAX];
+ char old_path[GIT_PATH_MAX];
+ char old_logs[GIT_PATH_MAX];
+ char normalized[GIT_REFNAME_MAX];
+ const char *target_ref = NULL;
+ const char *head_target = NULL;
+ const git_oid *target_oid = NULL;
+ git_reference *new_ref = NULL, *old_ref = NULL, *head = NULL;
assert(ref);
- /* Ensure the name is valid */
- error = normalize_name(normalized_name, sizeof(normalized_name), new_name, ref->type & GIT_REF_OID);
+ error = normalize_name(normalized, sizeof(normalized), new_name, ref->type & GIT_REF_OID);
if (error < GIT_SUCCESS)
- return git__rethrow(error, "Failed to rename reference");
+ return git__rethrow(error, "Failed to rename reference. Invalid name");
- new_name = normalized_name;
+ new_name = normalized;
- /* Ensure we're not going to overwrite an existing reference
- unless the user has allowed us */
- error = git_reference_lookup(&looked_up_ref, ref->owner, new_name);
+ error = git_reference_lookup(&new_ref, ref->owner, new_name);
if (error == GIT_SUCCESS && !force)
return git__throw(GIT_EEXISTS, "Failed to rename reference. Reference already exists");
- if (error < GIT_SUCCESS &&
- error != GIT_ENOTFOUND)
- return git__rethrow(error, "Failed to rename reference");
+ if (error < GIT_SUCCESS && error != GIT_ENOTFOUND)
+ goto cleanup;
if ((error = reference_available(ref->owner, new_name, ref->name)) < GIT_SUCCESS)
- return error == GIT_SUCCESS ? GIT_SUCCESS : git__rethrow(error, "Failed to rename reference. Reference already exists");
+ return git__rethrow(error, "Failed to rename reference. Reference already exists");
- old_name = ref->name;
- ref->name = git__strdup(new_name);
+ /*
+ * First, we backup the reference targets. Just keeping the old
+ * reference won't work, since we may have to remove it to create
+ * the new reference, e.g. when renaming foo/bar -> foo.
+ */
- if (ref->name == NULL) {
- ref->name = old_name;
- return GIT_ENOMEM;
+ if (ref->type & GIT_REF_SYMBOLIC) {
+ if ((target_ref = git_reference_target(ref)) == NULL)
+ goto cleanup;
+ } else {
+ if ((target_oid = git_reference_oid(ref)) == NULL)
+ goto cleanup;
}
- if (ref->type & GIT_REF_PACKED) {
- /* write the packfile to disk; note
- * that the state of the in-memory cache is not
- * consistent, because the reference is indexed
- * by its old name but it already has the new one.
- * This doesn't affect writing, though, and allows
- * us to rollback if writing fails
- */
-
- /* Create the loose ref under its new name */
- error = loose_write(ref);
- if (error < GIT_SUCCESS) {
- ref->type |= GIT_REF_PACKED;
- goto cleanup;
- }
+ /*
+ * Now delete the old ref and remove an possibly existing directory
+ * named `new_name`.
+ */
+ if (ref->type & GIT_REF_PACKED) {
ref->type &= ~GIT_REF_PACKED;
- /* Remove from the packfile cache in order to avoid packing it back
- * Note : we do not rely on git_reference_delete() because this would
- * invalidate the reference.
- */
git_hashtable_remove(ref->owner->references.packfile, old_name);
-
- /* Recreate the packed-refs file without the reference */
- error = packed_write(ref->owner);
- if (error < GIT_SUCCESS)
- goto rename_loose_to_old_name;
-
+ if ((error = packed_write(ref->owner)) < GIT_SUCCESS)
+ goto rollback;
} else {
git_path_join(old_path, ref->owner->path_repository, old_name);
- git_path_join(new_path, ref->owner->path_repository, ref->name);
-
- error = git_futils_mv_withpath(old_path, new_path);
- if (error < GIT_SUCCESS)
+ if ((error = p_unlink(old_path)) < GIT_SUCCESS)
goto cleanup;
- /* Once succesfully renamed, remove from the cache the reference known by its old name*/
git_hashtable_remove(ref->owner->references.loose_cache, old_name);
}
- /* Store the renamed reference into the loose ref cache */
- error = git_hashtable_insert2(ref->owner->references.loose_cache, ref->name, ref, (void **) &old_ref);
+ git_path_join(new_path, ref->owner->path_repository, new_name);
- /* If we force-replaced, we need to free the old reference */
- if(old_ref)
- reference_free(old_ref);
+ if (git_futils_exists(new_path) == GIT_SUCCESS) {
+ if (git_futils_isdir(new_path) == GIT_SUCCESS) {
+ if ((error = git_futils_rmdir_recurs(new_path, 0)) < GIT_SUCCESS)
+ goto rollback;
+ } else goto rollback;
+ }
- free(old_name);
- return error == GIT_SUCCESS ? GIT_SUCCESS : git__rethrow(error, "Failed to rename reference");
+ /*
+ * Crude hack: delete any logs till we support proper reflogs.
+ * Otherwise git.git will possibly fail and leave a mess. git.git
+ * writes reflogs by default in any repo with a working directory:
+ *
+ * "We only enable reflogs in repositories that have a working directory
+ * associated with them, as shared/bare repositories do not have
+ * an easy means to prune away old log entries, or may fail logging
+ * entirely if the user's gecos information is not valid during a push.
+ * This heuristic was suggested on the mailing list by Junio."
+ *
+ * Shawn O. Pearce - 0bee59186976b1d9e6b2dd77332480c9480131d5
+ *
+ * TODO
+ *
+ */
+
+ git_path_join_n(old_logs, 3, ref->owner->path_repository, "logs", old_name);
+ if (git_futils_exists(old_logs) == GIT_SUCCESS) {
+ if (git_futils_isfile(old_logs) == GIT_SUCCESS)
+ if ((error = p_unlink(old_logs)) < GIT_SUCCESS)
+ goto rollback;
+ }
+
+ /*
+ * Finally we can create the new reference.
+ */
+
+ if (ref->type & GIT_REF_SYMBOLIC) {
+ if ((error = git_reference_create_symbolic(&new_ref, ref->owner, new_name, target_ref, 0)) < GIT_SUCCESS)
+ goto rollback;
+ } else {
+ if ((error = git_reference_create_oid(&new_ref, ref->owner, new_name, target_oid, 0)) < GIT_SUCCESS)
+ goto rollback;
+ }
-cleanup:
- /* restore the old name if this failed */
free(ref->name);
- ref->name = old_name;
- return error == GIT_SUCCESS ? GIT_SUCCESS : git__rethrow(error, "Failed to rename reference");
+ ref->name = new_ref->name;
-rename_loose_to_old_name:
- /* If we hit this point. Something *bad* happened! Think "Ghostbusters
- * crossing the streams" definition of bad.
- * Either the packed-refs has been correctly generated and something else
- * has gone wrong, or the writing of the new packed-refs has failed, and
- * we're stuck with the old one. As a loose ref always takes priority over
- * a packed ref, we'll eventually try and rename the generated loose ref to
- * its former name. It even that fails, well... we might have lost the reference
- * for good. :-/
- */
+ if ((error = git_hashtable_insert2(ref->owner->references.loose_cache, new_ref->name, new_ref, (void **)&old_ref)) < GIT_SUCCESS)
+ goto rollback;
- git_path_join(old_path, ref->owner->path_repository, ref->name);
- git_path_join(new_path, ref->owner->path_repository, old_name);
+ /*
+ * Check if we have to update HEAD.
+ */
- /* No error checking. We'll return the initial error */
- git_futils_mv_withpath(old_path, new_path);
+ if ((error = git_reference_lookup(&head, ref->owner, GIT_HEAD_FILE)) < GIT_SUCCESS)
+ goto cleanup;
- /* restore the old name */
- free(ref->name);
- ref->name = old_name;
+ head_target = git_reference_target(head);
+ if (head_target && !strcmp(head_target, old_name))
+ if ((error = git_reference_create_symbolic(&head, new_ref->owner, "HEAD", new_ref->name, 1)) < GIT_SUCCESS)
+ goto rollback;
+
+cleanup:
+ free(old_name);
return error == GIT_SUCCESS ? GIT_SUCCESS : git__rethrow(error, "Failed to rename reference");
-}
+rollback:
+ /*
+ * Try to create the old reference again.
+ */
+ if (ref->type & GIT_REF_SYMBOLIC)
+ error = git_reference_create_symbolic(&new_ref, ref->owner, old_name, target_ref, 0);
+ else
+ error = git_reference_create_oid(&new_ref, ref->owner, old_name, target_oid, 0);
+
+ ref->name = old_name;
+
+ return error == GIT_SUCCESS ? GIT_SUCCESS : git__rethrow(error, "Failed to rename reference. Failed to rollback");
+}
/*
* Delete a reference.
diff --git a/src/refs.h b/src/refs.h
index a0159b091..dfac455e0 100644
--- a/src/refs.h
+++ b/src/refs.h
@@ -11,6 +11,8 @@
#define GIT_REFS_TAGS_DIR GIT_REFS_DIR "tags/"
#define GIT_REFS_REMOTES_DIR GIT_REFS_DIR "remotes/"
+#define GIT_RENAMED_REF_FILE GIT_REFS_DIR "RENAMED-REF"
+
#define GIT_SYMREF "ref: "
#define GIT_PACKEDREFS_FILE "packed-refs"
#define GIT_PACKEDREFS_HEADER "# pack-refs with: peeled "