Merge branch 'vk/p4-submit-shelve'
[git] / t / t5000-tar-tree.sh
1 #!/bin/sh
2 #
3 # Copyright (C) 2005 Rene Scharfe
4 #
5
6 test_description='git archive and git get-tar-commit-id test
7
8 This test covers the topics of file contents, commit date handling and
9 commit 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
29 SUBSTFORMAT=%H%n
30
31 test_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
40 test_lazy_prereq GZIP 'gzip --version'
41
42 get_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
59 check_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
98 nongit () {
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
111 test_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
131 test_expect_success \
132     'add ignored file' \
133     'echo ignore me >a/ignored &&
134      echo ignored export-ignore >.git/info/attributes'
135
136 test_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
141 test_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
147 test_expect_success \
148     'create bare clone' \
149     'git clone --bare . bare.git &&
150      cp .git/info/attributes bare.git/info/attributes'
151
152 test_expect_success \
153     'remove ignored file' \
154     'rm a/ignored'
155
156 test_expect_success \
157     'git archive' \
158     'git archive HEAD >b.tar'
159
160 check_tar b
161
162 test_expect_success 'git archive --prefix=prefix/' '
163         git archive --prefix=prefix/ HEAD >with_prefix.tar
164 '
165
166 check_tar with_prefix prefix/
167
168 test_expect_success 'git-archive --prefix=olde-' '
169         git archive --prefix=olde- HEAD >with_olde-prefix.tar
170 '
171
172 check_tar with_olde-prefix olde-
173
174 test_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
180 test_expect_success \
181     'git archive in a bare repo' \
182     '(cd bare.git && git archive HEAD) >b3.tar'
183
184 test_expect_success \
185     'git archive vs. the same in a bare repo' \
186     'test_cmp_bin b.tar b3.tar'
187
188 test_expect_success 'git archive with --output' \
189     'git archive --output=b4.tar HEAD &&
190     test_cmp_bin b.tar b4.tar'
191
192 test_expect_success 'git archive --remote' \
193     'git archive --remote=. HEAD >b5.tar &&
194     test_cmp_bin b.tar b5.tar'
195
196 test_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
205 test_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
213 test_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
218 test_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
223 test_expect_success 'git archive --list outside of a git repo' '
224         nongit git archive --list
225 '
226
227 test_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
233 test_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
241 test_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
250 test_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
257 test_expect_success 'archive --list mentions user filter' '
258         git archive --list >output &&
259         grep "^tar\.foo\$" output &&
260         grep "^bar\$" output
261 '
262
263 test_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
269 test_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
278 test_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
285 test_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
290 test_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
295 test_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
302 test_expect_success GZIP 'git archive --format=tgz' '
303         git archive --format=tgz HEAD >j.tgz
304 '
305
306 test_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
311 test_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
316 test_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
321 test_expect_success GZIP 'extract tgz file' '
322         gzip -d -c <j.tgz >j.tar &&
323         test_cmp_bin b.tar j.tar
324 '
325
326 test_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
331 test_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
337 test_expect_success 'archive and :(glob)' '
338         git archive -v HEAD -- ":(glob)**/sh" >/dev/null 2>actual &&
339         cat >expect <<EOF &&
340 a/
341 a/bin/
342 a/bin/sh
343 EOF
344         test_cmp expect actual
345 '
346
347 test_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).
360 tar_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.
373 test_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
379 test_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).
392 test_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
401 test_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
407 test_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
415 test_expect_success LONG_IS_64BIT 'generate tar with future mtime' '
416         git archive HEAD >future.tar
417 '
418
419 test_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
425 test_done