On Tue, Mar 22, 2016 at 3:40 PM, Junio C Hamano <gitster@xxxxxxxxx> wrote: > Stefan Beller <sbeller@xxxxxxxxxx> writes: > >> This change introduces the 'submodule.actionOnLabel' variable >> in a repository configuration. Generally speaking 'submodule.actionOnLabel' >> restricts the action of a command when no submodules are selected via the >> command line explicitely to those submodules, which are selected by >> 'submodule.actionOnLabel'. It can occur multiple times and can specify >> the path, the name or one of the labels of a submodule to select that >> submodule. >> >> The introduction of 'submodule.actionOnLabel' starts with >> 'git submodule update' in this patch and other commands will follow >> in later patches. >> >> 'submodule.actionOnLabel' implies '--init' in 'git submodule update'. >> >> Signed-off-by: Stefan Beller <sbeller@xxxxxxxxxx> >> >> TODO: generic documentation for submodule.actionOnLabel >> TODO: documentation for submodule update > > TODO: a name that matches the concept better. This is one of the hardest parts of the series so far. The last reviews were mostly bike shedding about the name of the concept and I thought we were settled to actionOnLabel as that fits best to what we want to do. So let's revisit that. My current understanding of the design: Generic properties in the data model: * Each submodule has a set of "things" attached to it. (A submodule can have none, one or many) * A "thing" can be attached to many submodules (That's why it was called groups in v1, and labels now) * The attachments of "things" to submodules can be viewed as a bipartite graph. * The attachment needs to work in a way, such that upstream can influence and redefine these attachments (e.g. .gitmodules file as part of the repo; another approach would be to have yet another file .gitlabels or such where you'd have a list of submodules belonging to a "thing") * If this feature is enabled, the user can select a set of submodules by selecting a set of "things" and all submodules connected to these things in the bipartite graph are selected. The expectation for a graph is to select a lot fewer "things" than submodules. By having this indirection via the graph, the selection of a subset of submodules is expected to be easier. Properties I derived from discussion and the data model: * The user does not need to have a way of overwriting the bipartite graph, because they can specify submodules by these "things", by path or by name. (It would be convenient to do be able to overwrite these, but it is not a strict requirement as the you can get any specification via a set of paths) * The user needs to make the explicit choice to use the new feature or not, as it has implications on the default behavior of submodule commands. * To make change of selection easy (which happens e.g. when switching branches or pulling in upstream changes), all submodules are initialized by default. * Once this feature is enabled a command doesn't apply to all initialized submodules by default any more, but the default set of submodules will be the selected set via the bipartite graph of "things". (Originally I typed out some implementation specific thoughts, but they are loaded with even more assumptions, so maybe we'd want to stay on this high level first) So any other naming proposals? Thanks, Stefan > > So in general > > $ git submodule $subcmd . > > may be the way to say "do $subcmd to all submodules", and > > $ git submodule $subcmd > > may have been "operate on nothing" (or may have been "operate on > everything"), but with this feature, > > $ git submodule $subcmd > > will by default operate on submodules that match the criteria the > new configuration variable specifies? > > I suspect that copying this from .gitmodules to .git/config will > have security implications and will not be done? What is the > expected way for projects to suggest which set of submodules are the > good ones to work on by default using this mechanism? > >> Signed-off-by: Stefan Beller <sbeller@xxxxxxxxxx> >> --- >> builtin/submodule--helper.c | 22 ++++++++- >> t/t7400-submodule-basic.sh | 115 ++++++++++++++++++++++++++++++++++++++++++++ >> 2 files changed, 136 insertions(+), 1 deletion(-) >> >> diff --git a/builtin/submodule--helper.c b/builtin/submodule--helper.c >> index a69b1f4..93760ec 100644 >> --- a/builtin/submodule--helper.c >> +++ b/builtin/submodule--helper.c >> @@ -573,6 +573,8 @@ struct submodule_update_clone { >> int current; >> struct module_list list; >> unsigned warn_if_uninitialized : 1; >> + /* patterns to initialize */ >> + struct string_list *initialize; >> >> /* update parameter passed via commandline */ >> struct submodule_update_strategy update; >> @@ -590,7 +592,7 @@ struct submodule_update_clone { >> /* If we want to stop as fast as possible and return an error */ >> unsigned quickstop : 1; >> }; >> -#define SUBMODULE_UPDATE_CLONE_INIT {0, MODULE_LIST_INIT, 0, \ >> +#define SUBMODULE_UPDATE_CLONE_INIT {0, MODULE_LIST_INIT, 0, NULL, \ >> SUBMODULE_UPDATE_STRATEGY_INIT, 0, NULL, NULL, NULL, NULL, \ >> STRING_LIST_INIT_DUP, 0} >> >> @@ -644,6 +646,15 @@ static int prepare_to_clone_next_submodule(const struct cache_entry *ce, >> strbuf_reset(&sb); >> strbuf_addf(&sb, "submodule.%s.url", sub->name); >> git_config_get_string(sb.buf, &url); >> + if (suc->initialize) { >> + if (!url) { >> + init_submodule(sub->path, suc->prefix, suc->quiet); >> + url = xstrdup(sub->url); >> + } >> + if (!submodule_applicable_by_labels(suc->initialize, sub) >> + && !suc->warn_if_uninitialized) >> + goto cleanup; >> + } >> if (!url) { >> /* >> * Only mention uninitialized submodules when their >> @@ -745,6 +756,7 @@ static int update_clone(int argc, const char **argv, const char *prefix) >> const char *update = NULL; >> int max_jobs = -1; >> struct string_list_item *item; >> + const struct string_list *list; >> struct pathspec pathspec; >> struct submodule_update_clone suc = SUBMODULE_UPDATE_CLONE_INIT; >> >> @@ -793,6 +805,14 @@ static int update_clone(int argc, const char **argv, const char *prefix) >> gitmodules_config(); >> git_config(submodule_config, NULL); >> >> + list = git_config_get_value_multi("submodule.actionOnLabel"); >> + if (list) { >> + suc.initialize = xmalloc(sizeof(*suc.initialize)); >> + string_list_init(suc.initialize, 1); >> + for_each_string_list_item(item, list) >> + string_list_insert(suc.initialize, item->string); >> + } >> + >> if (max_jobs < 0) >> max_jobs = parallel_submodules(); >> >> diff --git a/t/t7400-submodule-basic.sh b/t/t7400-submodule-basic.sh >> index fc948fd..dc45551 100755 >> --- a/t/t7400-submodule-basic.sh >> +++ b/t/t7400-submodule-basic.sh >> @@ -1032,4 +1032,119 @@ test_expect_success 'submodule add records multiple labels' ' >> test_cmp expected actual >> ' >> >> +cat <<EOF > expected >> +submodule >> +-submodule2 >> +EOF >> + >> +test_expect_success 'update initializes all modules when action-on-label configured' ' >> + test_when_finished "rm -rf super super_clone" && >> + mkdir super && >> + pwd=$(pwd) && >> + ( >> + cd super && >> + git init && >> + git submodule add --label labelA file://"$pwd"/example2 submodule && >> + git submodule add file://"$pwd"/example2 submodule2 && >> + git commit -a -m "add two modules, one is labled" >> + ) && >> + git clone super super_clone && >> + ( >> + cd super_clone && >> + git config submodule.actionOnLabel \*labelA && >> + git submodule update && >> + git submodule status |cut -c1,42-52 | tr -d " " >../actual >> + ) && >> + test_cmp actual expected >> +' >> + >> +test_expect_success 'submodule update applies to action-on-label selection' ' >> + test_when_finished "rm -rf super super_clone" && >> + mkdir super && >> + oldSubmoduleHead=$(cd example2 && git rev-parse HEAD) && >> + pwd=$(pwd) && >> + ( >> + cd super && >> + git init && >> + git submodule add --label labelA file://"$pwd"/example2 submodule1 && >> + git submodule add --label labelA file://"$pwd"/example2 submodule2 && >> + git submodule add --label labelA file://"$pwd"/example2 submodule3 && >> + git commit -a -m "add two modules, both are labled" >> + ) && >> + git clone super super_clone && >> + ( >> + cd super_clone && >> + git config submodule.actionOnLabel \*labelA && >> + git submodule update >> + ) && >> + ( >> + cd example2 && >> + touch anotherfile && >> + git add anotherfile && >> + git commit -m "advance example2" && >> + git checkout -b branchName >> + ) && >> + newSubmoduleHead=$(cd example2 && git rev-parse HEAD) && >> + ( >> + cd super && >> + git submodule add --label labelA file://"$pwd"/example2 submodule4 && >> + git commit -a -m "add another labeled module" && >> + git config -f .gitmodules submodule.submodule2.label labelB && >> + git config -f .gitmodules --unset submodule.submodule3.label && >> + git commit -a -m "unlabel 2 and 3 upstream" && >> + git submodule foreach git pull origin branchName && >> + git commit -a -m "update all submodules" && >> + git submodule status |cut -c1-52 >../actual >> + ) && >> + cat <<EOF >expected && >> + $newSubmoduleHead submodule1 >> + $newSubmoduleHead submodule2 >> + $newSubmoduleHead submodule3 >> + $newSubmoduleHead submodule4 >> +EOF >> + test_cmp actual expected && >> + ( >> + cd super_clone && >> + git pull && >> + git submodule update && >> + git submodule status |cut -c1-52 >../actual >> + ) && >> + cat <<EOF >expected && >> + $newSubmoduleHead submodule1 >> ++$oldSubmoduleHead submodule2 >> ++$oldSubmoduleHead submodule3 >> + $newSubmoduleHead submodule4 >> +EOF >> + test_cmp actual expected >> +' >> + >> +cat <<EOF > expected >> +submodule1 >> +submodule2 >> +-submodule3 >> +EOF >> + >> +test_expect_success 'Change labels in .git/config' ' >> + test_when_finished "rm -rf super super_clone" && >> + mkdir super && >> + pwd=$(pwd) && >> + ( >> + cd super && >> + git init && >> + git submodule add --label labelA file://"$pwd"/example2 submodule1 && >> + git submodule add file://"$pwd"/example2 submodule2 && >> + git submodule add file://"$pwd"/example2 submodule3 && >> + git commit -a -m "add two modules, one is labled" >> + ) && >> + git clone super super_clone && >> + ( >> + cd super_clone && >> + git config submodule.actionOnLabel \*labelA && >> + git config submodule.submodule2.label labelA >> + git submodule update && >> + git submodule status |cut -c1,42-52 | tr -d " " >../actual >> + ) && >> + test_cmp actual expected >> +' >> + >> test_done -- To unsubscribe from this list: send the line "unsubscribe git" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html