mirror of
https://github.com/git/git.git
synced 2026-01-11 13:23:12 +09:00
submodule--helper: add gitdir migration command
Manually running "git config submodule.<name>.gitdir .git/modules/<name>" for each submodule can be impractical, so add a migration command to submodule--helper to automatically create configs for all submodules as required by extensions.submodulePathConfig. The command calls create_default_gitdir_config() which validates the gitdir paths before adding the configs. Suggested-by: Junio C Hamano <gitster@pobox.com> Suggested-by: Patrick Steinhardt <ps@pks.im> Signed-off-by: Adrian Ratiu <adrian.ratiu@collabora.com> Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
parent
4cf8c114e3
commit
d5a4b9b73a
@ -93,8 +93,10 @@ Git will error out if a module does not have a corresponding
|
||||
`submodule.<name>.gitdir` set.
|
||||
+
|
||||
Existing (pre-extension) submodules need to be migrated by adding the missing
|
||||
config entries. This is done manually for now, e.g. for each submodule:
|
||||
`git config submodule.<name>.gitdir .git/modules/<name>`.
|
||||
config entries. This can be done manually, e.g. for each submodule:
|
||||
`git config submodule.<name>.gitdir .git/modules/<name>`, or via the
|
||||
`git submodule--helper migrate-gitdir-configs` command which iterates over all
|
||||
submodules and attempts to migrate them.
|
||||
+
|
||||
The extension can be enabled automatically for new repositories by setting
|
||||
`init.defaultSubmodulePathConfig` to `true`, for example by running
|
||||
|
||||
@ -1270,6 +1270,66 @@ static int module_gitdir(int argc, const char **argv, const char *prefix UNUSED,
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int module_migrate(int argc UNUSED, const char **argv UNUSED,
|
||||
const char *prefix UNUSED, struct repository *repo)
|
||||
{
|
||||
struct strbuf module_dir = STRBUF_INIT;
|
||||
DIR *dir;
|
||||
struct dirent *de;
|
||||
int repo_version = 0;
|
||||
|
||||
repo_git_path_append(repo, &module_dir, "modules/");
|
||||
|
||||
dir = opendir(module_dir.buf);
|
||||
if (!dir)
|
||||
die(_("could not open '%s'"), module_dir.buf);
|
||||
|
||||
while ((de = readdir(dir))) {
|
||||
struct strbuf gitdir_path = STRBUF_INIT;
|
||||
char *key;
|
||||
const char *value;
|
||||
|
||||
if (is_dot_or_dotdot(de->d_name))
|
||||
continue;
|
||||
|
||||
strbuf_addf(&gitdir_path, "%s/%s", module_dir.buf, de->d_name);
|
||||
if (!is_git_directory(gitdir_path.buf)) {
|
||||
strbuf_release(&gitdir_path);
|
||||
continue;
|
||||
}
|
||||
strbuf_release(&gitdir_path);
|
||||
|
||||
key = xstrfmt("submodule.%s.gitdir", de->d_name);
|
||||
if (!repo_config_get_string_tmp(repo, key, &value)) {
|
||||
/* Already has a gitdir config, nothing to do. */
|
||||
free(key);
|
||||
continue;
|
||||
}
|
||||
free(key);
|
||||
|
||||
create_default_gitdir_config(de->d_name);
|
||||
}
|
||||
|
||||
closedir(dir);
|
||||
strbuf_release(&module_dir);
|
||||
|
||||
repo_config_get_int(the_repository, "core.repositoryformatversion", &repo_version);
|
||||
if (repo_version == 0 &&
|
||||
repo_config_set_gently(repo, "core.repositoryformatversion", "1"))
|
||||
die(_("could not set core.repositoryformatversion to 1. "
|
||||
"Please set it for migration to work, for example: "
|
||||
"git config core.repositoryformatversion 1"));
|
||||
|
||||
if (repo_config_set_gently(repo, "extensions.submodulePathConfig", "true"))
|
||||
die(_("could not enable submodulePathConfig extension. It is required "
|
||||
"for migration to work. Please enable it in the root repo: "
|
||||
"git config extensions.submodulePathConfig true"));
|
||||
|
||||
repo->repository_format_submodule_path_cfg = 1;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
struct sync_cb {
|
||||
const char *prefix;
|
||||
const char *super_prefix;
|
||||
@ -3653,6 +3713,7 @@ int cmd_submodule__helper(int argc,
|
||||
NULL
|
||||
};
|
||||
struct option options[] = {
|
||||
OPT_SUBCOMMAND("migrate-gitdir-configs", &fn, module_migrate),
|
||||
OPT_SUBCOMMAND("gitdir", &fn, module_gitdir),
|
||||
OPT_SUBCOMMAND("clone", &fn, module_clone),
|
||||
OPT_SUBCOMMAND("add", &fn, module_add),
|
||||
|
||||
@ -160,8 +160,8 @@ test_expect_success 'fetch mixed submodule changes and verify updates' '
|
||||
test_expect_success '`git init` respects init.defaultSubmodulePathConfig' '
|
||||
git config --global init.defaultSubmodulePathConfig true &&
|
||||
git init repo-init &&
|
||||
git -C repo-init config extensions.submodulePathConfig > actual &&
|
||||
echo true > expect &&
|
||||
git -C repo-init config extensions.submodulePathConfig >actual &&
|
||||
echo true >expect &&
|
||||
test_cmp expect actual &&
|
||||
# create a submodule and check gitdir
|
||||
(
|
||||
@ -169,8 +169,8 @@ test_expect_success '`git init` respects init.defaultSubmodulePathConfig' '
|
||||
git init -b main sub &&
|
||||
test_commit -C sub sub-initial &&
|
||||
git submodule add ./sub sub &&
|
||||
git config submodule.sub.gitdir > actual &&
|
||||
echo ".git/modules/sub" > expect &&
|
||||
git config submodule.sub.gitdir >actual &&
|
||||
echo ".git/modules/sub" >expect &&
|
||||
test_cmp expect actual
|
||||
) &&
|
||||
git config --global --unset init.defaultSubmodulePathConfig
|
||||
@ -240,15 +240,15 @@ test_expect_success '`git clone` respects init.defaultSubmodulePathConfig' '
|
||||
cd repo-clone &&
|
||||
|
||||
# verify new repo extension is inherited from global config
|
||||
git config extensions.submodulePathConfig > actual &&
|
||||
echo true > expect &&
|
||||
git config extensions.submodulePathConfig >actual &&
|
||||
echo true >expect &&
|
||||
test_cmp expect actual &&
|
||||
|
||||
# new submodule has a gitdir config
|
||||
git submodule add ../sub sub &&
|
||||
test_path_is_dir .git/modules/sub &&
|
||||
git config submodule.sub.gitdir > actual &&
|
||||
echo ".git/modules/sub" > expect &&
|
||||
git config submodule.sub.gitdir >actual &&
|
||||
echo ".git/modules/sub" >expect &&
|
||||
test_cmp expect actual
|
||||
) &&
|
||||
git config --global --unset init.defaultSubmodulePathConfig
|
||||
@ -262,8 +262,8 @@ test_expect_success '`git clone --recurse-submodules` respects init.defaultSubmo
|
||||
cd repo-clone-recursive &&
|
||||
|
||||
# verify new repo extension is inherited from global config
|
||||
git config extensions.submodulePathConfig > actual &&
|
||||
echo true > expect &&
|
||||
git config extensions.submodulePathConfig >actual &&
|
||||
echo true >expect &&
|
||||
test_cmp expect actual &&
|
||||
|
||||
# previous submodules should exist
|
||||
@ -275,11 +275,79 @@ test_expect_success '`git clone --recurse-submodules` respects init.defaultSubmo
|
||||
# create another submodule and check that gitdir is created
|
||||
git submodule add ../sub new-sub &&
|
||||
test_path_is_dir .git/modules/new-sub &&
|
||||
git config submodule.new-sub.gitdir > actual &&
|
||||
echo ".git/modules/new-sub" > expect &&
|
||||
git config submodule.new-sub.gitdir >actual &&
|
||||
echo ".git/modules/new-sub" >expect &&
|
||||
test_cmp expect actual
|
||||
) &&
|
||||
git config --global --unset init.defaultSubmodulePathConfig
|
||||
'
|
||||
|
||||
test_expect_success 'submodule--helper migrates legacy modules' '
|
||||
(
|
||||
cd upstream &&
|
||||
|
||||
# previous submodules exist and were not migrated yet
|
||||
test_must_fail git config submodule.sub1.gitdir &&
|
||||
test_must_fail git config submodule.sub2.gitdir &&
|
||||
test_path_is_dir .git/modules/sub1 &&
|
||||
test_path_is_dir .git/modules/sub2 &&
|
||||
|
||||
# run migration
|
||||
git submodule--helper migrate-gitdir-configs &&
|
||||
|
||||
# test that migration worked
|
||||
git config submodule.sub1.gitdir >actual &&
|
||||
echo ".git/modules/sub1" >expect &&
|
||||
test_cmp expect actual &&
|
||||
git config submodule.sub2.gitdir >actual &&
|
||||
echo ".git/modules/sub2" >expect &&
|
||||
test_cmp expect actual &&
|
||||
|
||||
# repository extension is enabled after migration
|
||||
git config extensions.submodulePathConfig >actual &&
|
||||
echo "true" >expect &&
|
||||
test_cmp expect actual
|
||||
)
|
||||
'
|
||||
|
||||
test_expect_success '`git clone --recurse-submodules` works after migration' '
|
||||
test_when_finished "rm -rf repo-clone-recursive" &&
|
||||
|
||||
# test with extension disabled after the upstream repo was migrated
|
||||
git clone --recurse-submodules upstream repo-clone-recursive &&
|
||||
(
|
||||
cd repo-clone-recursive &&
|
||||
|
||||
# init.defaultSubmodulePathConfig was disabled before clone, so
|
||||
# the repo extension config should also be off, the migration ignored
|
||||
test_must_fail git config extensions.submodulePathConfig &&
|
||||
|
||||
# modules should look like there was no migration done
|
||||
test_must_fail git config submodule.sub1.gitdir &&
|
||||
test_must_fail git config submodule.sub2.gitdir &&
|
||||
test_path_is_dir .git/modules/sub1 &&
|
||||
test_path_is_dir .git/modules/sub2
|
||||
) &&
|
||||
rm -rf repo-clone-recursive &&
|
||||
|
||||
# enable the extension, then retry the clone
|
||||
git config --global init.defaultSubmodulePathConfig true &&
|
||||
git clone --recurse-submodules upstream repo-clone-recursive &&
|
||||
(
|
||||
cd repo-clone-recursive &&
|
||||
|
||||
# repository extension is enabled
|
||||
git config extensions.submodulePathConfig >actual &&
|
||||
echo "true" >expect &&
|
||||
test_cmp expect actual &&
|
||||
|
||||
# gitdir configs exist for submodules
|
||||
git config submodule.sub1.gitdir &&
|
||||
git config submodule.sub2.gitdir &&
|
||||
test_path_is_dir .git/modules/sub1 &&
|
||||
test_path_is_dir .git/modules/sub2
|
||||
) &&
|
||||
git config --global --unset init.defaultSubmodulePathConfig
|
||||
'
|
||||
|
||||
test_done
|
||||
|
||||
Loading…
x
Reference in New Issue
Block a user