]> git.ipfire.org Git - thirdparty/git.git/commitdiff
diff --submodule=diff: do not fail on ever-initialied deleted submodules
authorDavid Turner <dturner@twosigma.com>
Tue, 31 Aug 2021 13:12:56 +0000 (09:12 -0400)
committerJunio C Hamano <gitster@pobox.com>
Tue, 31 Aug 2021 17:11:48 +0000 (10:11 -0700)
If you have ever initialized a submodule, open_submodule will open it.
If you then delete the submodule's worktree directory (but don't
remove it from .gitmodules), git diff --submodule=diff would error out
as it attempted to chdir into the now-deleted working tree directory.

This only matters if the submodules git dir is absorbed.  If not, then
we no longer have anywhere to run the diff.  But that case does not
trigger this error, because in that case, open_submodule fails, so we
don't resolve a left commit, so we exit early, which is the only thing
we could do.

If absorbed, then we can run the diff from the submodule's absorbed
git dir (.git/modules/sm2).  In practice, that's a bit more
complicated, because `git diff` expects to be run from inside a
working directory, not a git dir.  So it looks in the config for
core.worktree, and does chdir("../../../sm2"), which is the very dir
that we're trying to avoid visiting because it's been deleted.  We
work around this by setting GIT_WORK_TREE (and GIT_DIR) to ".".  It is
little weird to set GIT_WORK_TREE to something that is not a working
tree just to avoid an unnecessary chdir, but it works.

Signed-off-by: David Turner <dturner@twosigma.com
Signed-off-by: Junio C Hamano <gitster@pobox.com>
submodule.c
t/t4060-diff-submodule-option-diff-format.sh

index 0b1d9c1dde5a8fc5cec04f13241f94b8a4378d5f..8aeff95cfd05d10bdf3eb29e9cfa1dafb79213cb 100644 (file)
@@ -710,6 +710,16 @@ void show_submodule_inline_diff(struct diff_options *o, const char *path,
                strvec_push(&cp.args, oid_to_hex(new_oid));
 
        prepare_submodule_repo_env(&cp.env_array);
+
+       if (!is_directory(path)) {
+               /* fall back to absorbed git dir, if any */
+               if (!sub)
+                       goto done;
+               cp.dir = sub->gitdir;
+               strvec_push(&cp.env_array, GIT_DIR_ENVIRONMENT "=.");
+               strvec_push(&cp.env_array, GIT_WORK_TREE_ENVIRONMENT "=.");
+       }
+
        if (start_command(&cp))
                diff_emit_submodule_error(o, "(diff failed)\n");
 
index 69b99469318b05d83428c10b488c74c259eebad5..d86e38abd882220ec60d45b2a351d2947f7dcfcb 100755 (executable)
@@ -703,10 +703,26 @@ test_expect_success 'path filter' '
        diff_cmp expected actual
 '
 
-commit_file sm2
+cat >.gitmodules <<-EOF
+[submodule "sm2"]
+       path = sm2
+       url = bogus_url
+EOF
+git add .gitmodules
+commit_file sm2 .gitmodules
+
 test_expect_success 'given commit' '
        git diff-index -p --submodule=diff HEAD^ >actual &&
        cat >expected <<-EOF &&
+       diff --git a/.gitmodules b/.gitmodules
+       new file mode 100644
+       index 1234567..89abcde
+       --- /dev/null
+       +++ b/.gitmodules
+       @@ -0,0 +1,3 @@
+       +[submodule "sm2"]
+       +path = sm2
+       +url = bogus_url
        Submodule sm1 $head7...0000000 (submodule deleted)
        Submodule sm2 0000000...$head9 (new submodule)
        diff --git a/sm2/foo8 b/sm2/foo8
@@ -728,15 +744,21 @@ test_expect_success 'given commit' '
 '
 
 test_expect_success 'setup .git file for sm2' '
-       (cd sm2 &&
-        REAL="$(pwd)/../.real" &&
-        mv .git "$REAL" &&
-        echo "gitdir: $REAL" >.git)
+       git submodule absorbgitdirs sm2
 '
 
 test_expect_success 'diff --submodule=diff with .git file' '
        git diff --submodule=diff HEAD^ >actual &&
        cat >expected <<-EOF &&
+       diff --git a/.gitmodules b/.gitmodules
+       new file mode 100644
+       index 1234567..89abcde
+       --- /dev/null
+       +++ b/.gitmodules
+       @@ -0,0 +1,3 @@
+       +[submodule "sm2"]
+       +path = sm2
+       +url = bogus_url
        Submodule sm1 $head7...0000000 (submodule deleted)
        Submodule sm2 0000000...$head9 (new submodule)
        diff --git a/sm2/foo8 b/sm2/foo8
@@ -757,9 +779,67 @@ test_expect_success 'diff --submodule=diff with .git file' '
        diff_cmp expected actual
 '
 
+mv sm2 sm2-bak
+
+test_expect_success 'deleted submodule with .git file' '
+       git diff-index -p --submodule=diff HEAD >actual &&
+       cat >expected <<-EOF &&
+       Submodule sm1 $head7...0000000 (submodule deleted)
+       Submodule sm2 $head9...0000000 (submodule deleted)
+       diff --git a/sm2/foo8 b/sm2/foo8
+       deleted file mode 100644
+       index 1234567..89abcde
+       --- a/sm2/foo8
+       +++ /dev/null
+       @@ -1 +0,0 @@
+       -foo8
+       diff --git a/sm2/foo9 b/sm2/foo9
+       deleted file mode 100644
+       index 1234567..89abcde
+       --- a/sm2/foo9
+       +++ /dev/null
+       @@ -1 +0,0 @@
+       -foo9
+       EOF
+       diff_cmp expected actual
+'
+
+echo submodule-to-blob>sm2
+
+test_expect_success 'typechanged(submodule->blob) submodule with .git file' '
+       git diff-index -p --submodule=diff HEAD >actual &&
+       cat >expected <<-EOF &&
+       Submodule sm1 $head7...0000000 (submodule deleted)
+       Submodule sm2 $head9...0000000 (submodule deleted)
+       diff --git a/sm2/foo8 b/sm2/foo8
+       deleted file mode 100644
+       index 1234567..89abcde
+       --- a/sm2/foo8
+       +++ /dev/null
+       @@ -1 +0,0 @@
+       -foo8
+       diff --git a/sm2/foo9 b/sm2/foo9
+       deleted file mode 100644
+       index 1234567..89abcde
+       --- a/sm2/foo9
+       +++ /dev/null
+       @@ -1 +0,0 @@
+       -foo9
+       diff --git a/sm2 b/sm2
+       new file mode 100644
+       index 1234567..89abcde
+       --- /dev/null
+       +++ b/sm2
+       @@ -0,0 +1 @@
+       +submodule-to-blob
+       EOF
+       diff_cmp expected actual
+'
+
+rm sm2
+mv sm2-bak sm2
+
 test_expect_success 'setup nested submodule' '
-       git submodule add -f ./sm2 &&
-       git commit -a -m "add sm2" &&
        git -C sm2 submodule add ../sm2 nested &&
        git -C sm2 commit -a -m "nested sub" &&
        head10=$(git -C sm2 rev-parse --short --verify HEAD)
@@ -790,6 +870,7 @@ test_expect_success 'diff --submodule=diff with moved nested submodule HEAD' '
 
 test_expect_success 'diff --submodule=diff recurses into nested submodules' '
        cat >expected <<-EOF &&
+       Submodule sm1 $head7...0000000 (submodule deleted)
        Submodule sm2 contains modified content
        Submodule sm2 $head9..$head10:
        diff --git a/sm2/.gitmodules b/sm2/.gitmodules
@@ -829,4 +910,67 @@ test_expect_success 'diff --submodule=diff recurses into nested submodules' '
        diff_cmp expected actual
 '
 
+(cd sm2; commit_file nested)
+commit_file sm2
+head12=$(cd sm2; git rev-parse --short --verify HEAD)
+
+mv sm2 sm2-bak
+
+test_expect_success 'diff --submodule=diff recurses into deleted nested submodules' '
+       cat >expected <<-EOF &&
+       Submodule sm1 $head7...0000000 (submodule deleted)
+       Submodule sm2 $head12...0000000 (submodule deleted)
+       diff --git a/sm2/.gitmodules b/sm2/.gitmodules
+       deleted file mode 100644
+       index 3a816b8..0000000
+       --- a/sm2/.gitmodules
+       +++ /dev/null
+       @@ -1,3 +0,0 @@
+       -[submodule "nested"]
+       -       path = nested
+       -       url = ../sm2
+       diff --git a/sm2/foo8 b/sm2/foo8
+       deleted file mode 100644
+       index db9916b..0000000
+       --- a/sm2/foo8
+       +++ /dev/null
+       @@ -1 +0,0 @@
+       -foo8
+       diff --git a/sm2/foo9 b/sm2/foo9
+       deleted file mode 100644
+       index 9c3b4f6..0000000
+       --- a/sm2/foo9
+       +++ /dev/null
+       @@ -1 +0,0 @@
+       -foo9
+       Submodule nested $head11...0000000 (submodule deleted)
+       diff --git a/sm2/nested/file b/sm2/nested/file
+       deleted file mode 100644
+       index ca281f5..0000000
+       --- a/sm2/nested/file
+       +++ /dev/null
+       @@ -1 +0,0 @@
+       -nested content
+       diff --git a/sm2/nested/foo8 b/sm2/nested/foo8
+       deleted file mode 100644
+       index db9916b..0000000
+       --- a/sm2/nested/foo8
+       +++ /dev/null
+       @@ -1 +0,0 @@
+       -foo8
+       diff --git a/sm2/nested/foo9 b/sm2/nested/foo9
+       deleted file mode 100644
+       index 9c3b4f6..0000000
+       --- a/sm2/nested/foo9
+       +++ /dev/null
+       @@ -1 +0,0 @@
+       -foo9
+       EOF
+       git diff --submodule=diff >actual 2>err &&
+       test_must_be_empty err &&
+       diff_cmp expected actual
+'
+
+mv sm2-bak sm2
+
 test_done