[PATCH] submodule--helper: check clone.defaultRemoteName in print-default-remote

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



From: Sean Barag <barag@xxxxxxxxxxxxxxxxx>

`git submodule update --remote` previously assumed that submodules in
detached-HEAD state (or with no remote configured via
branch.$branchname.remote) should always fetch from "origin". While this
was often true, users could rename the "origin" remote within that
submodule.  This would result in a pair of fatal errors:

    fatal: Needed a single revision
    fatal: Unable to find current origin/HEAD revision in submodule path
    'path/to/submodule'

Since de9ed3ef37 (clone: allow configurable default for `-o`/`--origin`,
2020-10-01) (released in git 2.30) however, that edge-case was trivially
easy to get into. Simply set `clone.defaultRemoteName`, clone a repo
with submodules, run `git submodule update --init`, and the next `git
submodule update --remote` would fail. Support clone.defaultRemoteName
in submodules by checking for the configured default remote name before
falling back to "origin".

Signed-off-by: Sean Barag <barag@xxxxxxxxxxxxxxxxx>
---
    [RFC PATCH] submodule--helper: check clone.defaultRemoteName in
    print-default-remote
    
    What used to be an edge case with git submodule update --remote got
    significantly easier to reproduce in 2.30+ with clone.defaultRemoteName
    set in git config. git-submodule.sh relies on git submodule--helper
    print-default-remote to determine which remote to fetch from, which
    falls back to "origin" when an explicit remote can't be found (e.g.
    detached HEAD states). Where clone.defaultRemoteName is set, however,
    that fallback to "origin" is no longer appropriate, because the "origin"
    remote doesn't exist. Either way, this results in a fatal error:
    
    fatal: Needed a single revision
    fatal: Unable to find current origin/HEAD revision in submodule path
    'path/to/submodule'
    
    
    This RFC patch adds a check for clone.defaultRemoteName in
    submodule-helper's get_default_remote(), but would love some feedback
    before this becomes a proper [PATCH] email. This works in my initial
    testing and in the added test case, but it feels like I'm missing
    something.
    
    Should I add test cases that don't rely on detached HEAD state perhaps?
    I'm not thrilled with the duplication of git_config_get_string calls for
    clone.defaultRemoteName, but extracting that logic to a static char
    *get_clone_default_remote_name() felt worse. I'd be happy to have any
    feedback y'all can provide :)
    
    From a high-level, I suppose this leaves the pre-2.30 edge case
    unresolved: users can still rename a submodule's remote to something
    else (something != clone.defaultRemoteName if it's set, or something !=
    "origin" if it's not set) and trigger this failure. Perhaps a long-term
    solution would involve querying the list of remotes for each submodule,
    and if there's only one remote assuming it's the correct one to fetch
    from, with the proposed solution as a fallback for multi-remote
    situations.
    
    Anyway, please let me know what I can do to clean this up, make it more
    robust, etc. before it becomes a more formal patch submission. Thanks in
    advance! Sean Barag
    
    Related conversations:
    
     * The original bug report:
       https://lore.kernel.org/git/2d58fe40-9e8c-4653-8170-5411fd3cf6f4@xxxxxxxxxxxxxxxx
    
     * An adjacent conversation about submodule--helper and remote names:
       https://lore.kernel.org/git/ae3baa2a-2f30-8e8a-f3cf-d8210e7225b0@xxxxxxxxx
       
       Philippe Blain levraiphilippeblain@xxxxxxxxx, Atharva Raykar
       raykar.ath@xxxxxxxxx, Christian Couder christian.couder@xxxxxxxxx,
       Shourya Shukla periperidip@xxxxxxxxx, Emily Shaffer
       emilyshaffer@xxxxxxxxxx

Published-As: https://github.com/gitgitgadget/git/releases/tag/pr-1112%2Fsjbarag%2Fcheck-clone.defaultremotename-in-submodule-helper-print-default-remote-v1
Fetch-It-Via: git fetch https://github.com/gitgitgadget/git pr-1112/sjbarag/check-clone.defaultremotename-in-submodule-helper-print-default-remote-v1
Pull-Request: https://github.com/gitgitgadget/git/pull/1112

 builtin/submodule--helper.c | 35 +++++++++++++++++++++++++----------
 t/t7400-submodule-basic.sh  | 18 ++++++++++++++++++
 2 files changed, 43 insertions(+), 10 deletions(-)

diff --git a/builtin/submodule--helper.c b/builtin/submodule--helper.c
index 9b25a508e6a..2d1468508ff 100644
--- a/builtin/submodule--helper.c
+++ b/builtin/submodule--helper.c
@@ -38,18 +38,33 @@ static char *get_default_remote(void)
 	if (!refname)
 		die(_("No such ref: %s"), "HEAD");
 
-	/* detached HEAD */
-	if (!strcmp(refname, "HEAD"))
-		return xstrdup("origin");
+	/* detached HEAD should use the configured default remote name or fall back to "origin" */
+	if (!strcmp(refname, "HEAD")) {
+		strbuf_addstr(&sb, "clone.defaultRemoteName");
 
-	if (!skip_prefix(refname, "refs/heads/", &refname))
-		die(_("Expecting a full ref name, got %s"), refname);
+		if (!git_config_get_string(sb.buf, &dest))
+			ret = dest;
+		else
+			ret = xstrdup("origin");
+	} else {
+		if (!skip_prefix(refname, "refs/heads/", &refname))
+			die(_("Expecting a full ref name, got %s"), refname);
 
-	strbuf_addf(&sb, "branch.%s.remote", refname);
-	if (git_config_get_string(sb.buf, &dest))
-		ret = xstrdup("origin");
-	else
-		ret = dest;
+		strbuf_addf(&sb, "branch.%s.remote", refname);
+		if (!git_config_get_string(sb.buf, &dest))
+			/* use configured remote name if one is present */
+			ret = dest;
+		else {
+			/* otherwise use configured *default* remote name, falling back to "origin" */
+			strbuf_reset(&sb);
+			strbuf_addstr(&sb, "clone.defaultRemoteName");
+
+			if (!git_config_get_string(sb.buf, &dest))
+				ret = dest;
+			else
+				ret = xstrdup("origin");
+		}
+	}
 
 	strbuf_release(&sb);
 	return ret;
diff --git a/t/t7400-submodule-basic.sh b/t/t7400-submodule-basic.sh
index e7cec2e457a..ce0801321a9 100755
--- a/t/t7400-submodule-basic.sh
+++ b/t/t7400-submodule-basic.sh
@@ -1416,4 +1416,22 @@ test_expect_success 'recursive clone respects -q' '
 	test_must_be_empty actual
 '
 
+test_expect_success 'submodule update --remote respects clone.defaultRemoteName' '
+	test_when_finished "rm -rf multisuper_clone" &&
+
+	# expect is intentionally empty.
+	# no output expected, since submodule "sub0" will be up to date
+	> expect &&
+
+	git clone multisuper multisuper_clone &&
+	git -C multisuper_clone submodule update --init -- sub0 &&
+	git -C multisuper_clone/sub0 remote rename origin upstream &&
+	git \
+		-C multisuper_clone \
+		-c clone.defaultRemoteName=upstream \
+		submodule update --remote -- sub0 \
+		2>&1 >actual &&
+	test_cmp expect actual
+'
+
 test_done

base-commit: e83ba647f7c61cf945690d6a0bd8c172a6498dc8
-- 
gitgitgadget



[Index of Archives]     [Linux Kernel Development]     [Gcc Help]     [IETF Annouce]     [DCCP]     [Netdev]     [Networking]     [Security]     [V4L]     [Bugtraq]     [Yosemite]     [MIPS Linux]     [ARM Linux]     [Linux Security]     [Linux RAID]     [Linux SCSI]     [Fedora Users]

  Powered by Linux