| /* |
| * "git mv" builtin command |
| * |
| * Copyright (C) 2006 Johannes Schindelin |
| */ |
| |
| #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 "pathspec.h" |
| #include "lockfile.h" |
| #include "dir.h" |
| #include "string-list.h" |
| #include "parse-options.h" |
| #include "read-cache-ll.h" |
| #include "repository.h" |
| #include "setup.h" |
| #include "strvec.h" |
| #include "submodule.h" |
| #include "entry.h" |
| |
| static const char * const builtin_mv_usage[] = { |
| N_("git mv [<options>] <source>... <destination>"), |
| NULL |
| }; |
| |
| enum update_mode { |
| WORKING_DIRECTORY = (1 << 1), |
| INDEX = (1 << 2), |
| SPARSE = (1 << 3), |
| SKIP_WORKTREE_DIR = (1 << 4), |
| }; |
| |
| #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; |
| } |
| |
| int cmd_mv(int argc, const char **argv, const char *prefix) |
| { |
| 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; |
| const char **src_dir = NULL; |
| int src_dir_nr = 0, src_dir_alloc = 0; |
| struct strbuf a_src_dir = STRBUF_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; |
| int ret; |
| |
| git_config(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)) { |
| 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; |
| } |
| |
| /* last - first >= 1 */ |
| modes[i] |= WORKING_DIRECTORY; |
| |
| ALLOC_GROW(src_dir, src_dir_nr + 1, src_dir_alloc); |
| src_dir[src_dir_nr++] = 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); |
| |
| memset(modes + argc + j, 0, sizeof(enum update_mode)); |
| modes[argc + j] |= 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--; |
| } |
| } |
| |
| 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)); |
| assert(pos >= 0); |
| 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(dst_dup); |
| FREE_AND_NULL(dst_dup); |
| rename(src, dst); |
| } |
| } |
| } |
| } |
| |
| /* |
| * cleanup the empty src_dirs |
| */ |
| 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); |
| free(src_dir); |
| |
| 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: |
| 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); |
| free(submodule_gitfiles); |
| free(modes); |
| return ret; |
| } |