diff options
Diffstat (limited to 'builtin/mv.c')
-rw-r--r-- | builtin/mv.c | 653 |
1 files changed, 653 insertions, 0 deletions
diff --git a/builtin/mv.c b/builtin/mv.c new file mode 100644 index 0000000000..d43925097b --- /dev/null +++ b/builtin/mv.c @@ -0,0 +1,653 @@ +/* + * "git mv" builtin command + * + * Copyright (C) 2006 Johannes Schindelin + */ + +#define USE_THE_REPOSITORY_VARIABLE +#define DISABLE_SIGN_COMPARE_WARNINGS + +#include "builtin.h" +#include "abspath.h" +#include "advice.h" +#include "config.h" +#include "environment.h" +#include "gettext.h" +#include "name-hash.h" +#include "object-file.h" +#include "path.h" +#include "pathspec.h" +#include "lockfile.h" +#include "dir.h" +#include "string-list.h" +#include "parse-options.h" +#include "read-cache-ll.h" + +#include "setup.h" +#include "strvec.h" +#include "submodule.h" +#include "entry.h" + +static const char * const builtin_mv_usage[] = { + N_("git mv [-v] [-f] [-n] [-k] <source> <destination>"), + N_("git mv [-v] [-f] [-n] [-k] <source>... <destination-directory>"), + NULL +}; + +enum update_mode { + WORKING_DIRECTORY = (1 << 1), + INDEX = (1 << 2), + SPARSE = (1 << 3), + SKIP_WORKTREE_DIR = (1 << 4), + /* + * A file gets moved implicitly via a move of one of its parent + * directories. This flag causes us to skip the check that we don't try + * to move a file and any of its parent directories at the same point + * in time. + */ + MOVE_VIA_PARENT_DIR = (1 << 5), +}; + +#define DUP_BASENAME 1 +#define KEEP_TRAILING_SLASH 2 + +static void internal_prefix_pathspec(struct strvec *out, + const char *prefix, + const char **pathspec, + int count, unsigned flags) +{ + int prefixlen = prefix ? strlen(prefix) : 0; + + /* Create an intermediate copy of the pathspec based on the flags */ + for (int i = 0; i < count; i++) { + size_t length = strlen(pathspec[i]); + size_t to_copy = length; + const char *maybe_basename; + char *trimmed, *prefixed_path; + + while (!(flags & KEEP_TRAILING_SLASH) && + to_copy > 0 && is_dir_sep(pathspec[i][to_copy - 1])) + to_copy--; + + trimmed = xmemdupz(pathspec[i], to_copy); + maybe_basename = (flags & DUP_BASENAME) ? basename(trimmed) : trimmed; + prefixed_path = prefix_path(prefix, prefixlen, maybe_basename); + strvec_push(out, prefixed_path); + + free(prefixed_path); + free(trimmed); + } +} + +static char *add_slash(const char *path) +{ + size_t len = strlen(path); + if (len && path[len - 1] != '/') { + char *with_slash = xmalloc(st_add(len, 2)); + memcpy(with_slash, path, len); + with_slash[len++] = '/'; + with_slash[len] = 0; + return with_slash; + } + return xstrdup(path); +} + +#define SUBMODULE_WITH_GITDIR ((const char *)1) + +static const char *submodule_gitfile_path(const char *src, int first) +{ + struct strbuf submodule_dotgit = STRBUF_INIT; + const char *path; + + if (!S_ISGITLINK(the_repository->index->cache[first]->ce_mode)) + die(_("Directory %s is in index and no submodule?"), src); + if (!is_staging_gitmodules_ok(the_repository->index)) + die(_("Please stage your changes to .gitmodules or stash them to proceed")); + + strbuf_addf(&submodule_dotgit, "%s/.git", src); + + path = read_gitfile(submodule_dotgit.buf); + strbuf_release(&submodule_dotgit); + if (path) + return path; + return SUBMODULE_WITH_GITDIR; +} + +static int index_range_of_same_dir(const char *src, int length, + int *first_p, int *last_p) +{ + char *src_w_slash = add_slash(src); + int first, last, len_w_slash = length + 1; + + first = index_name_pos(the_repository->index, src_w_slash, len_w_slash); + if (first >= 0) + die(_("%.*s is in index"), len_w_slash, src_w_slash); + + first = -1 - first; + for (last = first; last < the_repository->index->cache_nr; last++) { + const char *path = the_repository->index->cache[last]->name; + if (strncmp(path, src_w_slash, len_w_slash)) + break; + } + + free(src_w_slash); + *first_p = first; + *last_p = last; + return last - first; +} + +/* + * Given the path of a directory that does not exist on-disk, check whether the + * directory contains any entries in the index with the SKIP_WORKTREE flag + * enabled. + * Return 1 if such index entries exist. + * Return 0 otherwise. + */ +static int empty_dir_has_sparse_contents(const char *name) +{ + int ret = 0; + char *with_slash = add_slash(name); + int length = strlen(with_slash); + + int pos = index_name_pos(the_repository->index, with_slash, length); + const struct cache_entry *ce; + + if (pos < 0) { + pos = -pos - 1; + if (pos >= the_repository->index->cache_nr) + goto free_return; + ce = the_repository->index->cache[pos]; + if (strncmp(with_slash, ce->name, length)) + goto free_return; + if (ce_skip_worktree(ce)) + ret = 1; + } + +free_return: + free(with_slash); + return ret; +} + +static void remove_empty_src_dirs(const char **src_dir, size_t src_dir_nr) +{ + size_t i; + struct strbuf a_src_dir = STRBUF_INIT; + + for (i = 0; i < src_dir_nr; i++) { + int dummy; + strbuf_addstr(&a_src_dir, src_dir[i]); + /* + * if entries under a_src_dir are all moved away, + * recursively remove a_src_dir to cleanup + */ + if (index_range_of_same_dir(a_src_dir.buf, a_src_dir.len, + &dummy, &dummy) < 1) { + remove_dir_recursively(&a_src_dir, 0); + } + strbuf_reset(&a_src_dir); + } + + strbuf_release(&a_src_dir); +} + +struct pathmap_entry { + struct hashmap_entry ent; + const char *path; +}; + +static int pathmap_cmp(const void *cmp_data UNUSED, + const struct hashmap_entry *a, + const struct hashmap_entry *b, + const void *key UNUSED) +{ + const struct pathmap_entry *e1 = container_of(a, struct pathmap_entry, ent); + const struct pathmap_entry *e2 = container_of(b, struct pathmap_entry, ent); + return fspathcmp(e1->path, e2->path); +} + +int cmd_mv(int argc, + const char **argv, + const char *prefix, + struct repository *repo UNUSED) +{ + int i, flags, gitmodules_modified = 0; + int verbose = 0, show_only = 0, force = 0, ignore_errors = 0, ignore_sparse = 0; + struct option builtin_mv_options[] = { + OPT__VERBOSE(&verbose, N_("be verbose")), + OPT__DRY_RUN(&show_only, N_("dry run")), + OPT__FORCE(&force, N_("force move/rename even if target exists"), + PARSE_OPT_NOCOMPLETE), + OPT_BOOL('k', NULL, &ignore_errors, N_("skip move/rename errors")), + OPT_BOOL(0, "sparse", &ignore_sparse, N_("allow updating entries outside of the sparse-checkout cone")), + OPT_END(), + }; + struct strvec sources = STRVEC_INIT; + struct strvec dest_paths = STRVEC_INIT; + struct strvec destinations = STRVEC_INIT; + struct strvec submodule_gitfiles_to_free = STRVEC_INIT; + const char **submodule_gitfiles; + char *dst_w_slash = NULL; + struct strvec src_dir = STRVEC_INIT; + enum update_mode *modes, dst_mode = 0; + struct stat st, dest_st; + struct string_list src_for_dst = STRING_LIST_INIT_DUP; + struct lock_file lock_file = LOCK_INIT; + struct cache_entry *ce; + struct string_list only_match_skip_worktree = STRING_LIST_INIT_DUP; + struct string_list dirty_paths = STRING_LIST_INIT_DUP; + struct hashmap moved_dirs = HASHMAP_INIT(pathmap_cmp, NULL); + struct strbuf pathbuf = STRBUF_INIT; + int ret; + + repo_config(the_repository, git_default_config, NULL); + + argc = parse_options(argc, argv, prefix, builtin_mv_options, + builtin_mv_usage, 0); + if (--argc < 1) + usage_with_options(builtin_mv_usage, builtin_mv_options); + + repo_hold_locked_index(the_repository, &lock_file, LOCK_DIE_ON_ERROR); + if (repo_read_index(the_repository) < 0) + die(_("index file corrupt")); + + internal_prefix_pathspec(&sources, prefix, argv, argc, 0); + CALLOC_ARRAY(modes, argc); + + /* + * Keep trailing slash, needed to let + * "git mv file no-such-dir/" error out, except in the case + * "git mv directory no-such-dir/". + */ + flags = KEEP_TRAILING_SLASH; + if (argc == 1 && is_directory(argv[0]) && !is_directory(argv[1])) + flags = 0; + internal_prefix_pathspec(&dest_paths, prefix, argv + argc, 1, flags); + dst_w_slash = add_slash(dest_paths.v[0]); + submodule_gitfiles = xcalloc(argc, sizeof(char *)); + + if (dest_paths.v[0][0] == '\0') + /* special case: "." was normalized to "" */ + internal_prefix_pathspec(&destinations, dest_paths.v[0], argv, argc, DUP_BASENAME); + else if (!lstat(dest_paths.v[0], &st) && S_ISDIR(st.st_mode)) { + internal_prefix_pathspec(&destinations, dst_w_slash, argv, argc, DUP_BASENAME); + } else if (!path_in_sparse_checkout(dst_w_slash, the_repository->index) && + empty_dir_has_sparse_contents(dst_w_slash)) { + internal_prefix_pathspec(&destinations, dst_w_slash, argv, argc, DUP_BASENAME); + dst_mode = SKIP_WORKTREE_DIR; + } else if (argc != 1) { + die(_("destination '%s' is not a directory"), dest_paths.v[0]); + } else { + strvec_pushv(&destinations, dest_paths.v); + + /* + * <destination> is a file outside of sparse-checkout + * cone. Insist on cone mode here for backward + * compatibility. We don't want dst_mode to be assigned + * for a file when the repo is using no-cone mode (which + * is deprecated at this point) sparse-checkout. As + * SPARSE here is only considering cone-mode situation. + */ + if (!path_in_cone_mode_sparse_checkout(destinations.v[0], the_repository->index)) + dst_mode = SPARSE; + } + + /* Checking */ + for (i = 0; i < argc; i++) { + const char *src = sources.v[i], *dst = destinations.v[i]; + int length; + const char *bad = NULL; + int skip_sparse = 0; + + if (show_only) + printf(_("Checking rename of '%s' to '%s'\n"), src, dst); + + length = strlen(src); + if (lstat(src, &st) < 0) { + int pos; + const struct cache_entry *ce; + + pos = index_name_pos(the_repository->index, src, length); + if (pos < 0) { + char *src_w_slash = add_slash(src); + if (!path_in_sparse_checkout(src_w_slash, the_repository->index) && + empty_dir_has_sparse_contents(src)) { + free(src_w_slash); + modes[i] |= SKIP_WORKTREE_DIR; + goto dir_check; + } + free(src_w_slash); + /* only error if existence is expected. */ + if (!(modes[i] & SPARSE)) + bad = _("bad source"); + goto act_on_entry; + } + ce = the_repository->index->cache[pos]; + if (!ce_skip_worktree(ce)) { + bad = _("bad source"); + goto act_on_entry; + } + if (!ignore_sparse) { + string_list_append(&only_match_skip_worktree, src); + goto act_on_entry; + } + /* Check if dst exists in index */ + if (index_name_pos(the_repository->index, dst, strlen(dst)) < 0) { + modes[i] |= SPARSE; + goto act_on_entry; + } + if (!force) { + bad = _("destination exists"); + goto act_on_entry; + } + modes[i] |= SPARSE; + goto act_on_entry; + } + if (!strncmp(src, dst, length) && + (dst[length] == 0 || dst[length] == '/')) { + bad = _("can not move directory into itself"); + goto act_on_entry; + } + if (S_ISDIR(st.st_mode) + && lstat(dst, &dest_st) == 0) { + bad = _("destination already exists"); + goto act_on_entry; + } + +dir_check: + if (S_ISDIR(st.st_mode)) { + struct pathmap_entry *entry; + char *dst_with_slash; + size_t dst_with_slash_len; + int j, n; + int first = index_name_pos(the_repository->index, src, length), last; + + if (first >= 0) { + const char *path = submodule_gitfile_path(src, first); + if (path != SUBMODULE_WITH_GITDIR) + path = strvec_push(&submodule_gitfiles_to_free, path); + submodule_gitfiles[i] = path; + goto act_on_entry; + } else if (index_range_of_same_dir(src, length, + &first, &last) < 1) { + bad = _("source directory is empty"); + goto act_on_entry; + } + + entry = xmalloc(sizeof(*entry)); + entry->path = src; + hashmap_entry_init(&entry->ent, fspathhash(src)); + hashmap_add(&moved_dirs, &entry->ent); + + /* last - first >= 1 */ + modes[i] |= WORKING_DIRECTORY; + + strvec_push(&src_dir, src); + + n = argc + last - first; + REALLOC_ARRAY(modes, n); + REALLOC_ARRAY(submodule_gitfiles, n); + + dst_with_slash = add_slash(dst); + dst_with_slash_len = strlen(dst_with_slash); + + for (j = 0; j < last - first; j++) { + const struct cache_entry *ce = the_repository->index->cache[first + j]; + const char *path = ce->name; + char *prefixed_path = prefix_path(dst_with_slash, dst_with_slash_len, path + length + 1); + + strvec_push(&sources, path); + strvec_push(&destinations, prefixed_path); + + modes[argc + j] = MOVE_VIA_PARENT_DIR | (ce_skip_worktree(ce) ? SPARSE : INDEX); + submodule_gitfiles[argc + j] = NULL; + + free(prefixed_path); + } + + free(dst_with_slash); + argc += last - first; + goto act_on_entry; + } + if (!(ce = index_file_exists(the_repository->index, src, length, 0))) { + bad = _("not under version control"); + goto act_on_entry; + } + if (ce_stage(ce)) { + bad = _("conflicted"); + goto act_on_entry; + } + if (lstat(dst, &st) == 0 && + (!ignore_case || strcasecmp(src, dst))) { + bad = _("destination exists"); + if (force) { + /* + * only files can overwrite each other: + * check both source and destination + */ + if (S_ISREG(st.st_mode) || S_ISLNK(st.st_mode)) { + if (verbose) + warning(_("overwriting '%s'"), dst); + bad = NULL; + } else + bad = _("Cannot overwrite"); + } + goto act_on_entry; + } + if (string_list_has_string(&src_for_dst, dst)) { + bad = _("multiple sources for the same target"); + goto act_on_entry; + } + if (is_dir_sep(dst[strlen(dst) - 1])) { + bad = _("destination directory does not exist"); + goto act_on_entry; + } + + if (ignore_sparse && + (dst_mode & (SKIP_WORKTREE_DIR | SPARSE)) && + index_entry_exists(the_repository->index, dst, strlen(dst))) { + bad = _("destination exists in the index"); + if (force) { + if (verbose) + warning(_("overwriting '%s'"), dst); + bad = NULL; + } else { + goto act_on_entry; + } + } + /* + * We check if the paths are in the sparse-checkout + * definition as a very final check, since that + * allows us to point the user to the --sparse + * option as a way to have a successful run. + */ + if (!ignore_sparse && + !path_in_sparse_checkout(src, the_repository->index)) { + string_list_append(&only_match_skip_worktree, src); + skip_sparse = 1; + } + if (!ignore_sparse && + !path_in_sparse_checkout(dst, the_repository->index)) { + string_list_append(&only_match_skip_worktree, dst); + skip_sparse = 1; + } + + if (skip_sparse) + goto remove_entry; + + string_list_insert(&src_for_dst, dst); + +act_on_entry: + if (!bad) + continue; + if (!ignore_errors) + die(_("%s, source=%s, destination=%s"), + bad, src, dst); +remove_entry: + if (--argc > 0) { + int n = argc - i; + strvec_remove(&sources, i); + strvec_remove(&destinations, i); + MOVE_ARRAY(modes + i, modes + i + 1, n); + MOVE_ARRAY(submodule_gitfiles + i, + submodule_gitfiles + i + 1, n); + i--; + } + } + + for (i = 0; i < argc; i++) { + const char *slash_pos; + + if (modes[i] & MOVE_VIA_PARENT_DIR) + continue; + + strbuf_reset(&pathbuf); + strbuf_addstr(&pathbuf, sources.v[i]); + + slash_pos = strrchr(pathbuf.buf, '/'); + while (slash_pos > pathbuf.buf) { + struct pathmap_entry needle; + + strbuf_setlen(&pathbuf, slash_pos - pathbuf.buf); + + needle.path = pathbuf.buf; + hashmap_entry_init(&needle.ent, fspathhash(pathbuf.buf)); + + if (hashmap_get_entry(&moved_dirs, &needle, ent, NULL)) + die(_("cannot move both '%s' and its parent directory '%s'"), + sources.v[i], pathbuf.buf); + + slash_pos = strrchr(pathbuf.buf, '/'); + } + } + + if (only_match_skip_worktree.nr) { + advise_on_updating_sparse_paths(&only_match_skip_worktree); + if (!ignore_errors) { + ret = 1; + goto out; + } + } + + for (i = 0; i < argc; i++) { + const char *src = sources.v[i], *dst = destinations.v[i]; + enum update_mode mode = modes[i]; + int pos; + int sparse_and_dirty = 0; + struct checkout state = CHECKOUT_INIT; + state.istate = the_repository->index; + + if (force) + state.force = 1; + if (show_only || verbose) + printf(_("Renaming %s to %s\n"), src, dst); + if (show_only) + continue; + if (!(mode & (INDEX | SPARSE | SKIP_WORKTREE_DIR)) && + !(dst_mode & (SKIP_WORKTREE_DIR | SPARSE)) && + rename(src, dst) < 0) { + if (ignore_errors) + continue; + die_errno(_("renaming '%s' failed"), src); + } + if (submodule_gitfiles[i]) { + if (!update_path_in_gitmodules(src, dst)) + gitmodules_modified = 1; + if (submodule_gitfiles[i] != SUBMODULE_WITH_GITDIR) + connect_work_tree_and_git_dir(dst, + submodule_gitfiles[i], + 1); + } + + if (mode & (WORKING_DIRECTORY | SKIP_WORKTREE_DIR)) + continue; + + pos = index_name_pos(the_repository->index, src, strlen(src)); + if (pos < 0) + BUG("could not find source in index: '%s'", src); + if (!(mode & SPARSE) && !lstat(src, &st)) + sparse_and_dirty = ie_modified(the_repository->index, + the_repository->index->cache[pos], + &st, + 0); + rename_index_entry_at(the_repository->index, pos, dst); + + if (ignore_sparse && + core_apply_sparse_checkout && + core_sparse_checkout_cone) { + /* + * NEEDSWORK: we are *not* paying attention to + * "out-to-out" move (<source> is out-of-cone and + * <destination> is out-of-cone) at this point. It + * should be added in a future patch. + */ + if ((mode & SPARSE) && + path_in_sparse_checkout(dst, the_repository->index)) { + /* from out-of-cone to in-cone */ + int dst_pos = index_name_pos(the_repository->index, dst, + strlen(dst)); + struct cache_entry *dst_ce = the_repository->index->cache[dst_pos]; + + dst_ce->ce_flags &= ~CE_SKIP_WORKTREE; + + if (checkout_entry(dst_ce, &state, NULL, NULL)) + die(_("cannot checkout %s"), dst_ce->name); + } else if ((dst_mode & (SKIP_WORKTREE_DIR | SPARSE)) && + !(mode & SPARSE) && + !path_in_sparse_checkout(dst, the_repository->index)) { + /* from in-cone to out-of-cone */ + int dst_pos = index_name_pos(the_repository->index, dst, + strlen(dst)); + struct cache_entry *dst_ce = the_repository->index->cache[dst_pos]; + + /* + * if src is clean, it will suffice to remove it + */ + if (!sparse_and_dirty) { + dst_ce->ce_flags |= CE_SKIP_WORKTREE; + unlink_or_warn(src); + } else { + /* + * if src is dirty, move it to the + * destination and create leading + * dirs if necessary + */ + char *dst_dup = xstrdup(dst); + string_list_append(&dirty_paths, dst); + safe_create_leading_directories(the_repository, dst_dup); + FREE_AND_NULL(dst_dup); + rename(src, dst); + } + } + } + } + + remove_empty_src_dirs(src_dir.v, src_dir.nr); + + if (dirty_paths.nr) + advise_on_moving_dirty_path(&dirty_paths); + + if (gitmodules_modified) + stage_updated_gitmodules(the_repository->index); + + if (write_locked_index(the_repository->index, &lock_file, + COMMIT_LOCK | SKIP_IF_UNCHANGED)) + die(_("Unable to write new index file")); + + ret = 0; + +out: + strvec_clear(&src_dir); + free(dst_w_slash); + string_list_clear(&src_for_dst, 0); + string_list_clear(&dirty_paths, 0); + string_list_clear(&only_match_skip_worktree, 0); + strvec_clear(&sources); + strvec_clear(&dest_paths); + strvec_clear(&destinations); + strvec_clear(&submodule_gitfiles_to_free); + hashmap_clear_and_free(&moved_dirs, struct pathmap_entry, ent); + strbuf_release(&pathbuf); + free(submodule_gitfiles); + free(modes); + return ret; +} |