submodule: fixup nested submodules after moving the submodule

connect_work_tree_and_git_dir is used to connect a submodule worktree with
its git directory and vice versa after events that require a reconnection
such as moving around the working tree. As submodules can have nested
submodules themselves, we'd also want to fix the nested submodules when
asked to. Add an option to recurse into the nested submodules and connect
them as well.

As submodules are identified by their name (which determines their git
directory in relation to their superproject's git directory) internally
and by their path in the working tree of the superproject, we need to
make sure that the mapping of name <-> path is kept intact. We can do
that in the git-mv command by writing out the gitmodules file first
and then forcing a reload of the submodule config machinery.

Signed-off-by: Stefan Beller <sbeller@google.com>
Reviewed-by: Jonathan Tan <jonathantanmy@google.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
Stefan Beller 2018-03-28 15:35:31 -07:00 committed by Junio C Hamano
parent 0c89fdd739
commit da62f786d2
8 changed files with 83 additions and 15 deletions

View File

@ -275,10 +275,12 @@ int cmd_mv(int argc, const char **argv, const char *prefix)
die_errno(_("renaming '%s' failed"), src); die_errno(_("renaming '%s' failed"), src);
} }
if (submodule_gitfile[i]) { if (submodule_gitfile[i]) {
if (submodule_gitfile[i] != SUBMODULE_WITH_GITDIR)
connect_work_tree_and_git_dir(dst, submodule_gitfile[i]);
if (!update_path_in_gitmodules(src, dst)) if (!update_path_in_gitmodules(src, dst))
gitmodules_modified = 1; gitmodules_modified = 1;
if (submodule_gitfile[i] != SUBMODULE_WITH_GITDIR)
connect_work_tree_and_git_dir(dst,
submodule_gitfile[i],
1);
} }
if (mode == WORKING_DIRECTORY) if (mode == WORKING_DIRECTORY)

View File

@ -1260,8 +1260,7 @@ static int module_clone(int argc, const char **argv, const char *prefix)
strbuf_reset(&sb); strbuf_reset(&sb);
} }
/* Connect module worktree and git dir */ connect_work_tree_and_git_dir(path, sm_gitdir, 0);
connect_work_tree_and_git_dir(path, sm_gitdir);
p = git_pathdup_submodule(path, "config"); p = git_pathdup_submodule(path, "config");
if (!p) if (!p)

60
dir.c
View File

@ -19,6 +19,7 @@
#include "varint.h" #include "varint.h"
#include "ewah/ewok.h" #include "ewah/ewok.h"
#include "fsmonitor.h" #include "fsmonitor.h"
#include "submodule-config.h"
/* /*
* Tells read_directory_recursive how a file or directory should be treated. * Tells read_directory_recursive how a file or directory should be treated.
@ -2988,8 +2989,57 @@ void untracked_cache_add_to_index(struct index_state *istate,
untracked_cache_invalidate_path(istate, path); untracked_cache_invalidate_path(istate, path);
} }
/* Update gitfile and core.worktree setting to connect work tree and git dir */ static void connect_wt_gitdir_in_nested(const char *sub_worktree,
void connect_work_tree_and_git_dir(const char *work_tree_, const char *git_dir_) const char *sub_gitdir)
{
int i;
struct repository subrepo;
struct strbuf sub_wt = STRBUF_INIT;
struct strbuf sub_gd = STRBUF_INIT;
const struct submodule *sub;
/* If the submodule has no working tree, we can ignore it. */
if (repo_init(&subrepo, sub_gitdir, sub_worktree))
return;
if (repo_read_index(&subrepo) < 0)
die("index file corrupt in repo %s", subrepo.gitdir);
for (i = 0; i < subrepo.index->cache_nr; i++) {
const struct cache_entry *ce = subrepo.index->cache[i];
if (!S_ISGITLINK(ce->ce_mode))
continue;
while (i + 1 < subrepo.index->cache_nr &&
!strcmp(ce->name, subrepo.index->cache[i + 1]->name))
/*
* Skip entries with the same name in different stages
* to make sure an entry is returned only once.
*/
i++;
sub = submodule_from_path(&subrepo, &null_oid, ce->name);
if (!sub || !is_submodule_active(&subrepo, ce->name))
/* .gitmodules broken or inactive sub */
continue;
strbuf_reset(&sub_wt);
strbuf_reset(&sub_gd);
strbuf_addf(&sub_wt, "%s/%s", sub_worktree, sub->path);
strbuf_addf(&sub_gd, "%s/modules/%s", sub_gitdir, sub->name);
connect_work_tree_and_git_dir(sub_wt.buf, sub_gd.buf, 1);
}
strbuf_release(&sub_wt);
strbuf_release(&sub_gd);
repo_clear(&subrepo);
}
void connect_work_tree_and_git_dir(const char *work_tree_,
const char *git_dir_,
int recurse_into_nested)
{ {
struct strbuf gitfile_sb = STRBUF_INIT; struct strbuf gitfile_sb = STRBUF_INIT;
struct strbuf cfg_sb = STRBUF_INIT; struct strbuf cfg_sb = STRBUF_INIT;
@ -3019,6 +3069,10 @@ void connect_work_tree_and_git_dir(const char *work_tree_, const char *git_dir_)
strbuf_release(&gitfile_sb); strbuf_release(&gitfile_sb);
strbuf_release(&cfg_sb); strbuf_release(&cfg_sb);
strbuf_release(&rel_path); strbuf_release(&rel_path);
if (recurse_into_nested)
connect_wt_gitdir_in_nested(work_tree, git_dir);
free(work_tree); free(work_tree);
free(git_dir); free(git_dir);
} }
@ -3032,5 +3086,5 @@ void relocate_gitdir(const char *path, const char *old_git_dir, const char *new_
die_errno(_("could not migrate git directory from '%s' to '%s'"), die_errno(_("could not migrate git directory from '%s' to '%s'"),
old_git_dir, new_git_dir); old_git_dir, new_git_dir);
connect_work_tree_and_git_dir(path, new_git_dir); connect_work_tree_and_git_dir(path, new_git_dir, 0);
} }

12
dir.h
View File

@ -359,7 +359,17 @@ struct untracked_cache *read_untracked_extension(const void *data, unsigned long
void write_untracked_extension(struct strbuf *out, struct untracked_cache *untracked); void write_untracked_extension(struct strbuf *out, struct untracked_cache *untracked);
void add_untracked_cache(struct index_state *istate); void add_untracked_cache(struct index_state *istate);
void remove_untracked_cache(struct index_state *istate); void remove_untracked_cache(struct index_state *istate);
extern void connect_work_tree_and_git_dir(const char *work_tree, const char *git_dir);
/*
* Connect a worktree to a git directory by creating (or overwriting) a
* '.git' file containing the location of the git directory. In the git
* directory set the core.worktree setting to indicate where the worktree is.
* When `recurse_into_nested` is set, recurse into any nested submodules,
* connecting them as well.
*/
extern void connect_work_tree_and_git_dir(const char *work_tree,
const char *git_dir,
int recurse_into_nested);
extern void relocate_gitdir(const char *path, extern void relocate_gitdir(const char *path,
const char *old_git_dir, const char *old_git_dir,
const char *new_git_dir); const char *new_git_dir);

View File

@ -135,9 +135,9 @@ static int read_and_verify_repository_format(struct repository_format *format,
* Initialize 'repo' based on the provided 'gitdir'. * Initialize 'repo' based on the provided 'gitdir'.
* Return 0 upon success and a non-zero value upon failure. * Return 0 upon success and a non-zero value upon failure.
*/ */
static int repo_init(struct repository *repo, int repo_init(struct repository *repo,
const char *gitdir, const char *gitdir,
const char *worktree) const char *worktree)
{ {
struct repository_format format; struct repository_format format;
memset(repo, 0, sizeof(*repo)); memset(repo, 0, sizeof(*repo));

View File

@ -97,6 +97,9 @@ extern void repo_set_gitdir(struct repository *repo,
extern void repo_set_worktree(struct repository *repo, const char *path); extern void repo_set_worktree(struct repository *repo, const char *path);
extern void repo_set_hash_algo(struct repository *repo, int algo); extern void repo_set_hash_algo(struct repository *repo, int algo);
extern void initialize_the_repository(void); extern void initialize_the_repository(void);
extern int repo_init(struct repository *r,
const char *gitdir,
const char *worktree);
extern int repo_submodule_init(struct repository *submodule, extern int repo_submodule_init(struct repository *submodule,
struct repository *superproject, struct repository *superproject,
const char *path); const char *path);

View File

@ -1625,7 +1625,7 @@ int submodule_move_head(const char *path,
} else { } else {
char *gitdir = xstrfmt("%s/modules/%s", char *gitdir = xstrfmt("%s/modules/%s",
get_git_common_dir(), sub->name); get_git_common_dir(), sub->name);
connect_work_tree_and_git_dir(path, gitdir); connect_work_tree_and_git_dir(path, gitdir, 0);
free(gitdir); free(gitdir);
/* make sure the index is clean as well */ /* make sure the index is clean as well */
@ -1635,7 +1635,7 @@ int submodule_move_head(const char *path,
if (old && (flags & SUBMODULE_MOVE_HEAD_FORCE)) { if (old && (flags & SUBMODULE_MOVE_HEAD_FORCE)) {
char *gitdir = xstrfmt("%s/modules/%s", char *gitdir = xstrfmt("%s/modules/%s",
get_git_common_dir(), sub->name); get_git_common_dir(), sub->name);
connect_work_tree_and_git_dir(path, gitdir); connect_work_tree_and_git_dir(path, gitdir, 1);
free(gitdir); free(gitdir);
} }
} }
@ -1948,7 +1948,7 @@ void absorb_git_dir_into_superproject(const char *prefix,
if (!sub) if (!sub)
die(_("could not lookup name for submodule '%s'"), path); die(_("could not lookup name for submodule '%s'"), path);
connect_work_tree_and_git_dir(path, connect_work_tree_and_git_dir(path,
git_path("modules/%s", sub->name)); git_path("modules/%s", sub->name), 0);
} else { } else {
/* Is it already absorbed into the superprojects git dir? */ /* Is it already absorbed into the superprojects git dir? */
char *real_sub_git_dir = real_pathdup(sub_git_dir, 1); char *real_sub_git_dir = real_pathdup(sub_git_dir, 1);

View File

@ -491,7 +491,7 @@ test_expect_success 'moving a submodule in nested directories' '
test_cmp expect actual test_cmp expect actual
' '
test_expect_failure 'moving nested submodules' ' test_expect_success 'moving nested submodules' '
git commit -am "cleanup commit" && git commit -am "cleanup commit" &&
mkdir sub_nested_nested && mkdir sub_nested_nested &&
(cd sub_nested_nested && (cd sub_nested_nested &&