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