t / t5000-tar-tree.shon commit fsck: report trees as dangling (b4584e4)
   1#!/bin/sh
   2#
   3# Copyright (C) 2005 Rene Scharfe
   4#
   5
   6test_description='git archive and git get-tar-commit-id test
   7
   8This test covers the topics of file contents, commit date handling and
   9commit id embedding:
  10
  11  The contents of the repository is compared to the extracted tar
  12  archive.  The repository contains simple text files, symlinks and a
  13  binary file (/bin/sh).  Only paths shorter than 99 characters are
  14  used.
  15
  16  git archive applies the commit date to every file in the archive it
  17  creates.  The test sets the commit date to a specific value and checks
  18  if the tar archive contains that value.
  19
  20  When giving git archive a commit id (in contrast to a tree id) it
  21  embeds this commit id into the tar archive as a comment.  The test
  22  checks the ability of git get-tar-commit-id to figure it out from the
  23  tar file.
  24
  25'
  26
  27. ./test-lib.sh
  28
  29SUBSTFORMAT=%H%n
  30
  31test_lazy_prereq TAR_NEEDS_PAX_FALLBACK '
  32        (
  33                mkdir pax &&
  34                cd pax &&
  35                "$TAR" xf "$TEST_DIRECTORY"/t5000/pax.tar &&
  36                test -f PaxHeaders.1791/file
  37        )
  38'
  39
  40test_lazy_prereq GZIP 'gzip --version'
  41
  42get_pax_header() {
  43        file=$1
  44        header=$2=
  45
  46        while read len rest
  47        do
  48                if test "$len" = $(echo "$len $rest" | wc -c)
  49                then
  50                        case "$rest" in
  51                        $header*)
  52                                echo "${rest#$header}"
  53                                ;;
  54                        esac
  55                fi
  56        done <"$file"
  57}
  58
  59check_tar() {
  60        tarfile=$1.tar
  61        listfile=$1.lst
  62        dir=$1
  63        dir_with_prefix=$dir/$2
  64
  65        test_expect_success ' extract tar archive' '
  66                (mkdir $dir && cd $dir && "$TAR" xf -) <$tarfile
  67        '
  68
  69        test_expect_success TAR_NEEDS_PAX_FALLBACK ' interpret pax headers' '
  70                (
  71                        cd $dir &&
  72                        for header in *.paxheader
  73                        do
  74                                data=${header%.paxheader}.data &&
  75                                if test -h $data || test -e $data
  76                                then
  77                                        path=$(get_pax_header $header path) &&
  78                                        if test -n "$path"
  79                                        then
  80                                                mv "$data" "$path"
  81                                        fi
  82                                fi
  83                        done
  84                )
  85        '
  86
  87        test_expect_success ' validate filenames' '
  88                (cd ${dir_with_prefix}a && find .) | sort >$listfile &&
  89                test_cmp a.lst $listfile
  90        '
  91
  92        test_expect_success ' validate file contents' '
  93                diff -r a ${dir_with_prefix}a
  94        '
  95}
  96
  97# run "$@" inside a non-git directory
  98nongit () {
  99        test -d non-repo ||
 100        mkdir non-repo ||
 101        return 1
 102
 103        (
 104                GIT_CEILING_DIRECTORIES=$(pwd) &&
 105                export GIT_CEILING_DIRECTORIES &&
 106                cd non-repo &&
 107                "$@"
 108        )
 109}
 110
 111test_expect_success \
 112    'populate workdir' \
 113    'mkdir a &&
 114     echo simple textfile >a/a &&
 115     ten=0123456789 && hundred=$ten$ten$ten$ten$ten$ten$ten$ten$ten$ten &&
 116     echo long filename >a/four$hundred &&
 117     mkdir a/bin &&
 118     test-genrandom "frotz" 500000 >a/bin/sh &&
 119     printf "A\$Format:%s\$O" "$SUBSTFORMAT" >a/substfile1 &&
 120     printf "A not substituted O" >a/substfile2 &&
 121     if test_have_prereq SYMLINKS; then
 122        ln -s a a/l1
 123     else
 124        printf %s a > a/l1
 125     fi &&
 126     (p=long_path_to_a_file && cd a &&
 127      for depth in 1 2 3 4 5; do mkdir $p && cd $p; done &&
 128      echo text >file_with_long_path) &&
 129     (cd a && find .) | sort >a.lst'
 130
 131test_expect_success \
 132    'add ignored file' \
 133    'echo ignore me >a/ignored &&
 134     echo ignored export-ignore >.git/info/attributes'
 135
 136test_expect_success 'add files to repository' '
 137        git add a &&
 138        GIT_COMMITTER_DATE="2005-05-27 22:00" git commit -m initial
 139'
 140
 141test_expect_success 'setup export-subst' '
 142        echo "substfile?" export-subst >>.git/info/attributes &&
 143        git log --max-count=1 "--pretty=format:A${SUBSTFORMAT}O" HEAD \
 144                >a/substfile1
 145'
 146
 147test_expect_success \
 148    'create bare clone' \
 149    'git clone --bare . bare.git &&
 150     cp .git/info/attributes bare.git/info/attributes'
 151
 152test_expect_success \
 153    'remove ignored file' \
 154    'rm a/ignored'
 155
 156test_expect_success \
 157    'git archive' \
 158    'git archive HEAD >b.tar'
 159
 160check_tar b
 161
 162test_expect_success 'git archive --prefix=prefix/' '
 163        git archive --prefix=prefix/ HEAD >with_prefix.tar
 164'
 165
 166check_tar with_prefix prefix/
 167
 168test_expect_success 'git-archive --prefix=olde-' '
 169        git archive --prefix=olde- HEAD >with_olde-prefix.tar
 170'
 171
 172check_tar with_olde-prefix olde-
 173
 174test_expect_success 'git archive on large files' '
 175    test_config core.bigfilethreshold 1 &&
 176    git archive HEAD >b3.tar &&
 177    test_cmp_bin b.tar b3.tar
 178'
 179
 180test_expect_success \
 181    'git archive in a bare repo' \
 182    '(cd bare.git && git archive HEAD) >b3.tar'
 183
 184test_expect_success \
 185    'git archive vs. the same in a bare repo' \
 186    'test_cmp_bin b.tar b3.tar'
 187
 188test_expect_success 'git archive with --output' \
 189    'git archive --output=b4.tar HEAD &&
 190    test_cmp_bin b.tar b4.tar'
 191
 192test_expect_success 'git archive --remote' \
 193    'git archive --remote=. HEAD >b5.tar &&
 194    test_cmp_bin b.tar b5.tar'
 195
 196test_expect_success 'git archive --remote with configured remote' '
 197        git config remote.foo.url . &&
 198        (
 199                cd a &&
 200                git archive --remote=foo --output=../b5-nick.tar HEAD
 201        ) &&
 202        test_cmp_bin b.tar b5-nick.tar
 203'
 204
 205test_expect_success \
 206    'validate file modification time' \
 207    'mkdir extract &&
 208     "$TAR" xf b.tar -C extract a/a &&
 209     test-chmtime -v +0 extract/a/a |cut -f 1 >b.mtime &&
 210     echo "1117231200" >expected.mtime &&
 211     test_cmp expected.mtime b.mtime'
 212
 213test_expect_success \
 214    'git get-tar-commit-id' \
 215    'git get-tar-commit-id <b.tar >b.commitid &&
 216     test_cmp .git/$(git symbolic-ref HEAD) b.commitid'
 217
 218test_expect_success 'git archive with --output, override inferred format' '
 219        git archive --format=tar --output=d4.zip HEAD &&
 220        test_cmp_bin b.tar d4.zip
 221'
 222
 223test_expect_success 'git archive --list outside of a git repo' '
 224        nongit git archive --list
 225'
 226
 227test_expect_success 'git archive --remote outside of a git repo' '
 228        git archive HEAD >expect.tar &&
 229        nongit git archive --remote="$PWD" HEAD >actual.tar &&
 230        test_cmp_bin expect.tar actual.tar
 231'
 232
 233test_expect_success 'clients cannot access unreachable commits' '
 234        test_commit unreachable &&
 235        sha1=$(git rev-parse HEAD) &&
 236        git reset --hard HEAD^ &&
 237        git archive $sha1 >remote.tar &&
 238        test_must_fail git archive --remote=. $sha1 >remote.tar
 239'
 240
 241test_expect_success 'upload-archive can allow unreachable commits' '
 242        test_commit unreachable1 &&
 243        sha1=$(git rev-parse HEAD) &&
 244        git reset --hard HEAD^ &&
 245        git archive $sha1 >remote.tar &&
 246        test_config uploadarchive.allowUnreachable true &&
 247        git archive --remote=. $sha1 >remote.tar
 248'
 249
 250test_expect_success 'setup tar filters' '
 251        git config tar.tar.foo.command "tr ab ba" &&
 252        git config tar.bar.command "tr ab ba" &&
 253        git config tar.bar.remote true &&
 254        git config tar.invalid baz
 255'
 256
 257test_expect_success 'archive --list mentions user filter' '
 258        git archive --list >output &&
 259        grep "^tar\.foo\$" output &&
 260        grep "^bar\$" output
 261'
 262
 263test_expect_success 'archive --list shows only enabled remote filters' '
 264        git archive --list --remote=. >output &&
 265        ! grep "^tar\.foo\$" output &&
 266        grep "^bar\$" output
 267'
 268
 269test_expect_success 'invoke tar filter by format' '
 270        git archive --format=tar.foo HEAD >config.tar.foo &&
 271        tr ab ba <config.tar.foo >config.tar &&
 272        test_cmp_bin b.tar config.tar &&
 273        git archive --format=bar HEAD >config.bar &&
 274        tr ab ba <config.bar >config.tar &&
 275        test_cmp_bin b.tar config.tar
 276'
 277
 278test_expect_success 'invoke tar filter by extension' '
 279        git archive -o config-implicit.tar.foo HEAD &&
 280        test_cmp_bin config.tar.foo config-implicit.tar.foo &&
 281        git archive -o config-implicit.bar HEAD &&
 282        test_cmp_bin config.tar.foo config-implicit.bar
 283'
 284
 285test_expect_success 'default output format remains tar' '
 286        git archive -o config-implicit.baz HEAD &&
 287        test_cmp_bin b.tar config-implicit.baz
 288'
 289
 290test_expect_success 'extension matching requires dot' '
 291        git archive -o config-implicittar.foo HEAD &&
 292        test_cmp_bin b.tar config-implicittar.foo
 293'
 294
 295test_expect_success 'only enabled filters are available remotely' '
 296        test_must_fail git archive --remote=. --format=tar.foo HEAD \
 297                >remote.tar.foo &&
 298        git archive --remote=. --format=bar >remote.bar HEAD &&
 299        test_cmp_bin remote.bar config.bar
 300'
 301
 302test_expect_success GZIP 'git archive --format=tgz' '
 303        git archive --format=tgz HEAD >j.tgz
 304'
 305
 306test_expect_success GZIP 'git archive --format=tar.gz' '
 307        git archive --format=tar.gz HEAD >j1.tar.gz &&
 308        test_cmp_bin j.tgz j1.tar.gz
 309'
 310
 311test_expect_success GZIP 'infer tgz from .tgz filename' '
 312        git archive --output=j2.tgz HEAD &&
 313        test_cmp_bin j.tgz j2.tgz
 314'
 315
 316test_expect_success GZIP 'infer tgz from .tar.gz filename' '
 317        git archive --output=j3.tar.gz HEAD &&
 318        test_cmp_bin j.tgz j3.tar.gz
 319'
 320
 321test_expect_success GZIP 'extract tgz file' '
 322        gzip -d -c <j.tgz >j.tar &&
 323        test_cmp_bin b.tar j.tar
 324'
 325
 326test_expect_success GZIP 'remote tar.gz is allowed by default' '
 327        git archive --remote=. --format=tar.gz HEAD >remote.tar.gz &&
 328        test_cmp_bin j.tgz remote.tar.gz
 329'
 330
 331test_expect_success GZIP 'remote tar.gz can be disabled' '
 332        git config tar.tar.gz.remote false &&
 333        test_must_fail git archive --remote=. --format=tar.gz HEAD \
 334                >remote.tar.gz
 335'
 336
 337test_expect_success 'archive and :(glob)' '
 338        git archive -v HEAD -- ":(glob)**/sh" >/dev/null 2>actual &&
 339        cat >expect <<EOF &&
 340a/
 341a/bin/
 342a/bin/sh
 343EOF
 344        test_cmp expect actual
 345'
 346
 347test_expect_success 'catch non-matching pathspec' '
 348        test_must_fail git archive -v HEAD -- "*.abc" >/dev/null
 349'
 350
 351# Pull the size and date of each entry in a tarfile using the system tar.
 352#
 353# We'll pull out only the year from the date; that avoids any question of
 354# timezones impacting the result (as long as we keep our test times away from a
 355# year boundary; our reference times are all in August).
 356#
 357# The output of tar_info is expected to be "<size> <year>", both in decimal. It
 358# ignores the return value of tar. We have to do this, because some of our test
 359# input is only partial (the real data is 64GB in some cases).
 360tar_info () {
 361        "$TAR" tvf "$1" |
 362        awk '{
 363                split($4, date, "-")
 364                print $3 " " date[1]
 365        }'
 366}
 367
 368# See if our system tar can handle a tar file with huge sizes and dates far in
 369# the future, and that we can actually parse its output.
 370#
 371# The reference file was generated by GNU tar, and the magic time and size are
 372# both octal 01000000000001, which overflows normal ustar fields.
 373test_lazy_prereq TAR_HUGE '
 374        echo "68719476737 4147" >expect &&
 375        tar_info "$TEST_DIRECTORY"/t5000/huge-and-future.tar >actual &&
 376        test_cmp expect actual
 377'
 378
 379test_expect_success LONG_IS_64BIT 'set up repository with huge blob' '
 380        obj_d=19 &&
 381        obj_f=f9c8273ec45a8938e6999cb59b3ff66739902a &&
 382        obj=${obj_d}${obj_f} &&
 383        mkdir -p .git/objects/$obj_d &&
 384        cp "$TEST_DIRECTORY"/t5000/$obj .git/objects/$obj_d/$obj_f &&
 385        rm -f .git/index &&
 386        git update-index --add --cacheinfo 100644,$obj,huge &&
 387        git commit -m huge
 388'
 389
 390# We expect git to die with SIGPIPE here (otherwise we
 391# would generate the whole 64GB).
 392test_expect_success LONG_IS_64BIT 'generate tar with huge size' '
 393        {
 394                git archive HEAD
 395                echo $? >exit-code
 396        } | test_copy_bytes 4096 >huge.tar &&
 397        echo 141 >expect &&
 398        test_cmp expect exit-code
 399'
 400
 401test_expect_success TAR_HUGE,LONG_IS_64BIT 'system tar can read our huge size' '
 402        echo 68719476737 >expect &&
 403        tar_info huge.tar | cut -d" " -f1 >actual &&
 404        test_cmp expect actual
 405'
 406
 407test_expect_success LONG_IS_64BIT 'set up repository with far-future commit' '
 408        rm -f .git/index &&
 409        echo content >file &&
 410        git add file &&
 411        GIT_COMMITTER_DATE="@68719476737 +0000" \
 412                git commit -m "tempori parendum"
 413'
 414
 415test_expect_success LONG_IS_64BIT 'generate tar with future mtime' '
 416        git archive HEAD >future.tar
 417'
 418
 419test_expect_success TAR_HUGE,LONG_IS_64BIT 'system tar can read our future mtime' '
 420        echo 4147 >expect &&
 421        tar_info future.tar | cut -d" " -f2 >actual &&
 422        test_cmp expect actual
 423'
 424
 425test_done