1#!/bin/sh
   2test_description='git status for submodule'
   4. ./test-lib.sh
   6test_create_repo_with_commit () {
   8        test_create_repo "$1" &&
   9        (
  10                cd "$1" &&
  11                : >bar &&
  12                git add bar &&
  13                git commit -m " Add bar" &&
  14                : >foo &&
  15                git add foo &&
  16                git commit -m " Add foo"
  17        )
  18}
  19sanitize_output () {
  21        sed -e "s/$OID_REGEX/HASH/" -e "s/$OID_REGEX/HASH/" output >output2 &&
  22        mv output2 output
  23}
  24test_expect_success 'setup' '
  27        test_create_repo_with_commit sub &&
  28        echo output > .gitignore &&
  29        git add sub .gitignore &&
  30        git commit -m "Add submodule sub"
  31'
  32test_expect_success 'status clean' '
  34        git status >output &&
  35        test_i18ngrep "nothing to commit" output
  36'
  37test_expect_success 'commit --dry-run -a clean' '
  39        test_must_fail git commit --dry-run -a >output &&
  40        test_i18ngrep "nothing to commit" output
  41'
  42test_expect_success 'status with modified file in submodule' '
  44        (cd sub && git reset --hard) &&
  45        echo "changed" >sub/foo &&
  46        git status >output &&
  47        test_i18ngrep "modified:   sub (modified content)" output
  48'
  49test_expect_success 'status with modified file in submodule (porcelain)' '
  51        (cd sub && git reset --hard) &&
  52        echo "changed" >sub/foo &&
  53        git status --porcelain >output &&
  54        diff output - <<-\EOF
  55         M sub
  56        EOF
  57'
  58test_expect_success 'status with modified file in submodule (short)' '
  60        (cd sub && git reset --hard) &&
  61        echo "changed" >sub/foo &&
  62        git status --short >output &&
  63        diff output - <<-\EOF
  64         m sub
  65        EOF
  66'
  67test_expect_success 'status with added file in submodule' '
  69        (cd sub && git reset --hard && echo >foo && git add foo) &&
  70        git status >output &&
  71        test_i18ngrep "modified:   sub (modified content)" output
  72'
  73test_expect_success 'status with added file in submodule (porcelain)' '
  75        (cd sub && git reset --hard && echo >foo && git add foo) &&
  76        git status --porcelain >output &&
  77        diff output - <<-\EOF
  78         M sub
  79        EOF
  80'
  81test_expect_success 'status with added file in submodule (short)' '
  83        (cd sub && git reset --hard && echo >foo && git add foo) &&
  84        git status --short >output &&
  85        diff output - <<-\EOF
  86         m sub
  87        EOF
  88'
  89test_expect_success 'status with untracked file in submodule' '
  91        (cd sub && git reset --hard) &&
  92        echo "content" >sub/new-file &&
  93        git status >output &&
  94        test_i18ngrep "modified:   sub (untracked content)" output
  95'
  96test_expect_success 'status -uno with untracked file in submodule' '
  98        git status -uno >output &&
  99        test_i18ngrep "^nothing to commit" output
 100'
 101test_expect_success 'status with untracked file in submodule (porcelain)' '
 103        git status --porcelain >output &&
 104        diff output - <<-\EOF
 105         M sub
 106        EOF
 107'
 108test_expect_success 'status with untracked file in submodule (short)' '
 110        git status --short >output &&
 111        diff output - <<-\EOF
 112         ? sub
 113        EOF
 114'
 115test_expect_success 'status with added and untracked file in submodule' '
 117        (cd sub && git reset --hard && echo >foo && git add foo) &&
 118        echo "content" >sub/new-file &&
 119        git status >output &&
 120        test_i18ngrep "modified:   sub (modified content, untracked content)" output
 121'
 122test_expect_success 'status with added and untracked file in submodule (porcelain)' '
 124        (cd sub && git reset --hard && echo >foo && git add foo) &&
 125        echo "content" >sub/new-file &&
 126        git status --porcelain >output &&
 127        diff output - <<-\EOF
 128         M sub
 129        EOF
 130'
 131test_expect_success 'status with modified file in modified submodule' '
 133        (cd sub && git reset --hard) &&
 134        rm sub/new-file &&
 135        (cd sub && echo "next change" >foo && git commit -m "next change" foo) &&
 136        echo "changed" >sub/foo &&
 137        git status >output &&
 138        test_i18ngrep "modified:   sub (new commits, modified content)" output
 139'
 140test_expect_success 'status with modified file in modified submodule (porcelain)' '
 142        (cd sub && git reset --hard) &&
 143        echo "changed" >sub/foo &&
 144        git status --porcelain >output &&
 145        diff output - <<-\EOF
 146         M sub
 147        EOF
 148'
 149test_expect_success 'status with added file in modified submodule' '
 151        (cd sub && git reset --hard && echo >foo && git add foo) &&
 152        git status >output &&
 153        test_i18ngrep "modified:   sub (new commits, modified content)" output
 154'
 155test_expect_success 'status with added file in modified submodule (porcelain)' '
 157        (cd sub && git reset --hard && echo >foo && git add foo) &&
 158        git status --porcelain >output &&
 159        diff output - <<-\EOF
 160         M sub
 161        EOF
 162'
 163test_expect_success 'status with untracked file in modified submodule' '
 165        (cd sub && git reset --hard) &&
 166        echo "content" >sub/new-file &&
 167        git status >output &&
 168        test_i18ngrep "modified:   sub (new commits, untracked content)" output
 169'
 170test_expect_success 'status with untracked file in modified submodule (porcelain)' '
 172        git status --porcelain >output &&
 173        diff output - <<-\EOF
 174         M sub
 175        EOF
 176'
 177test_expect_success 'status with added and untracked file in modified submodule' '
 179        (cd sub && git reset --hard && echo >foo && git add foo) &&
 180        echo "content" >sub/new-file &&
 181        git status >output &&
 182        test_i18ngrep "modified:   sub (new commits, modified content, untracked content)" output
 183'
 184test_expect_success 'status with added and untracked file in modified submodule (porcelain)' '
 186        (cd sub && git reset --hard && echo >foo && git add foo) &&
 187        echo "content" >sub/new-file &&
 188        git status --porcelain >output &&
 189        diff output - <<-\EOF
 190         M sub
 191        EOF
 192'
 193test_expect_success 'setup .git file for sub' '
 195        (cd sub &&
 196         rm -f new-file &&
 197         REAL="$(pwd)/../.real" &&
 198         mv .git "$REAL" &&
 199         echo "gitdir: $REAL" >.git) &&
 200         echo .real >>.gitignore &&
 201         git commit -m "added .real to .gitignore" .gitignore
 202'
 203test_expect_success 'status with added file in modified submodule with .git file' '
 205        (cd sub && git reset --hard && echo >foo && git add foo) &&
 206        git status >output &&
 207        test_i18ngrep "modified:   sub (new commits, modified content)" output
 208'
 209test_expect_success 'status with a lot of untracked files in the submodule' '
 211        (
 212                cd sub &&
 213                i=0 &&
 214                while test $i -lt 1024
 215                do
 216                        >some-file-$i &&
 217                        i=$(( $i + 1 )) || exit 1
 218                done
 219        ) &&
 220        git status --porcelain sub 2>err.actual &&
 221        test_must_be_empty err.actual &&
 222        rm err.actual
 223'
 224test_expect_success 'rm submodule contents' '
 226        rm -rf sub &&
 227        mkdir sub
 228'
 229test_expect_success 'status clean (empty submodule dir)' '
 231        git status >output &&
 232        test_i18ngrep "nothing to commit" output
 233'
 234test_expect_success 'status -a clean (empty submodule dir)' '
 236        test_must_fail git commit --dry-run -a >output &&
 237        test_i18ngrep "nothing to commit" output
 238'
 239cat >status_expect <<\EOF
 241AA .gitmodules
 242A  sub1
 243EOF
 244test_expect_success 'status with merge conflict in .gitmodules' '
 246        git clone . super &&
 247        test_create_repo_with_commit sub1 &&
 248        test_tick &&
 249        test_create_repo_with_commit sub2 &&
 250        (
 251                cd super &&
 252                prev=$(git rev-parse HEAD) &&
 253                git checkout -b add_sub1 &&
 254                git submodule add ../sub1 &&
 255                git commit -m "add sub1" &&
 256                git checkout -b add_sub2 $prev &&
 257                git submodule add ../sub2 &&
 258                git commit -m "add sub2" &&
 259                git checkout -b merge_conflict_gitmodules &&
 260                test_must_fail git merge add_sub1 &&
 261                git status -s >../status_actual 2>&1
 262        ) &&
 263        test_cmp status_actual status_expect
 264'
 265sha1_merge_sub1=$(cd sub1 && git rev-parse HEAD)
 267sha1_merge_sub2=$(cd sub2 && git rev-parse HEAD)
 268short_sha1_merge_sub1=$(cd sub1 && git rev-parse --short HEAD)
 269short_sha1_merge_sub2=$(cd sub2 && git rev-parse --short HEAD)
 270cat >diff_expect <<\EOF
 271diff --cc .gitmodules
 272index badaa4c,44f999a..0000000
 273--- a/.gitmodules
 274+++ b/.gitmodules
 275@@@ -1,3 -1,3 +1,9 @@@
 276++<<<<<<< HEAD
 277 +[submodule "sub2"]
 278 +      path = sub2
 279 +      url = ../sub2
 280++=======
 281+ [submodule "sub1"]
 282+       path = sub1
 283+       url = ../sub1
 284++>>>>>>> add_sub1
 285EOF
 286cat >diff_submodule_expect <<\EOF
 288diff --cc .gitmodules
 289index badaa4c,44f999a..0000000
 290--- a/.gitmodules
 291+++ b/.gitmodules
 292@@@ -1,3 -1,3 +1,9 @@@
 293++<<<<<<< HEAD
 294 +[submodule "sub2"]
 295 +      path = sub2
 296 +      url = ../sub2
 297++=======
 298+ [submodule "sub1"]
 299+       path = sub1
 300+       url = ../sub1
 301++>>>>>>> add_sub1
 302EOF
 303test_expect_success 'diff with merge conflict in .gitmodules' '
 305        (
 306                cd super &&
 307                git diff >../diff_actual 2>&1
 308        ) &&
 309        test_cmp diff_expect diff_actual
 310'
 311test_expect_success 'diff --submodule with merge conflict in .gitmodules' '
 313        (
 314                cd super &&
 315                git diff --submodule >../diff_submodule_actual 2>&1
 316        ) &&
 317        test_cmp diff_submodule_expect diff_submodule_actual
 318'
 319# We'll setup different cases for further testing:
 321# sub1 will contain a nested submodule,
 322# sub2 will have an untracked file
 323# sub3 will have an untracked repository
 324test_expect_success 'setup superproject with untracked file in nested submodule' '
 325        (
 326                cd super &&
 327                git clean -dfx &&
 328                git rm .gitmodules &&
 329                git commit -m "remove .gitmodules" &&
 330                git submodule add -f ./sub1 &&
 331                git submodule add -f ./sub2 &&
 332                git submodule add -f ./sub1 sub3 &&
 333                git commit -a -m "messy merge in superproject" &&
 334                (
 335                        cd sub1 &&
 336                        git submodule add ../sub2 &&
 337                        git commit -a -m "add sub2 to sub1"
 338                ) &&
 339                git add sub1 &&
 340                git commit -a -m "update sub1 to contain nested sub"
 341        ) &&
 342        echo content >super/sub1/sub2/file &&
 343        echo content >super/sub2/file &&
 344        git -C super/sub3 clone ../../sub2 untracked_repository
 345'
 346test_expect_success 'status with untracked file in nested submodule (porcelain)' '
 348        git -C super status --porcelain >output &&
 349        diff output - <<-\EOF
 350         M sub1
 351         M sub2
 352         M sub3
 353        EOF
 354'
 355test_expect_success 'status with untracked file in nested submodule (porcelain=2)' '
 357        git -C super status --porcelain=2 >output &&
 358        sanitize_output output &&
 359        diff output - <<-\EOF
 360        1 .M S..U 160000 160000 160000 HASH HASH sub1
 361        1 .M S..U 160000 160000 160000 HASH HASH sub2
 362        1 .M S..U 160000 160000 160000 HASH HASH sub3
 363        EOF
 364'
 365test_expect_success 'status with untracked file in nested submodule (short)' '
 367        git -C super status --short >output &&
 368        diff output - <<-\EOF
 369         ? sub1
 370         ? sub2
 371         ? sub3
 372        EOF
 373'
 374test_expect_success 'setup superproject with modified file in nested submodule' '
 376        git -C super/sub1/sub2 add file &&
 377        git -C super/sub2 add file
 378'
 379test_expect_success 'status with added file in nested submodule (porcelain)' '
 381        git -C super status --porcelain >output &&
 382        diff output - <<-\EOF
 383         M sub1
 384         M sub2
 385         M sub3
 386        EOF
 387'
 388test_expect_success 'status with added file in nested submodule (porcelain=2)' '
 390        git -C super status --porcelain=2 >output &&
 391        sanitize_output output &&
 392        diff output - <<-\EOF
 393        1 .M S.M. 160000 160000 160000 HASH HASH sub1
 394        1 .M S.M. 160000 160000 160000 HASH HASH sub2
 395        1 .M S..U 160000 160000 160000 HASH HASH sub3
 396        EOF
 397'
 398test_expect_success 'status with added file in nested submodule (short)' '
 400        git -C super status --short >output &&
 401        diff output - <<-\EOF
 402         m sub1
 403         m sub2
 404         ? sub3
 405        EOF
 406'
 407test_done