1#!/bin/sh
2#
3# Copyright (c) 2007 Andy Parkins
4#
5
6test_description='for-each-ref test'
7
8. ./test-lib.sh
9. "$TEST_DIRECTORY"/lib-gpg.sh
10
11# Mon Jul 3 15:18:43 2006 +0000
12datestamp=1151939923
13setdate_and_increment () {
14 GIT_COMMITTER_DATE="$datestamp +0200"
15 datestamp=$(expr "$datestamp" + 1)
16 GIT_AUTHOR_DATE="$datestamp +0200"
17 datestamp=$(expr "$datestamp" + 1)
18 export GIT_COMMITTER_DATE GIT_AUTHOR_DATE
19}
20
21test_expect_success setup '
22 setdate_and_increment &&
23 echo "Using $datestamp" > one &&
24 git add one &&
25 git commit -m "Initial" &&
26 setdate_and_increment &&
27 git tag -a -m "Tagging at $datestamp" testtag &&
28 git update-ref refs/remotes/origin/master master &&
29 git remote add origin nowhere &&
30 git config branch.master.remote origin &&
31 git config branch.master.merge refs/heads/master
32'
33
34test_atom() {
35 case "$1" in
36 head) ref=refs/heads/master ;;
37 tag) ref=refs/tags/testtag ;;
38 *) ref=$1 ;;
39 esac
40 printf '%s\n' "$3" >expected
41 test_expect_${4:-success} $PREREQ "basic atom: $1 $2" "
42 git for-each-ref --format='%($2)' $ref >actual &&
43 sanitize_pgp <actual >actual.clean &&
44 test_cmp expected actual.clean
45 "
46}
47
48test_atom head refname refs/heads/master
49test_atom head upstream refs/remotes/origin/master
50test_atom head objecttype commit
51test_atom head objectsize 171
52test_atom head objectname $(git rev-parse refs/heads/master)
53test_atom head tree $(git rev-parse refs/heads/master^{tree})
54test_atom head parent ''
55test_atom head numparent 0
56test_atom head object ''
57test_atom head type ''
58test_atom head '*objectname' ''
59test_atom head '*objecttype' ''
60test_atom head author 'A U Thor <author@example.com> 1151939924 +0200'
61test_atom head authorname 'A U Thor'
62test_atom head authoremail '<author@example.com>'
63test_atom head authordate 'Mon Jul 3 17:18:44 2006 +0200'
64test_atom head committer 'C O Mitter <committer@example.com> 1151939923 +0200'
65test_atom head committername 'C O Mitter'
66test_atom head committeremail '<committer@example.com>'
67test_atom head committerdate 'Mon Jul 3 17:18:43 2006 +0200'
68test_atom head tag ''
69test_atom head tagger ''
70test_atom head taggername ''
71test_atom head taggeremail ''
72test_atom head taggerdate ''
73test_atom head creator 'C O Mitter <committer@example.com> 1151939923 +0200'
74test_atom head creatordate 'Mon Jul 3 17:18:43 2006 +0200'
75test_atom head subject 'Initial'
76test_atom head contents:subject 'Initial'
77test_atom head body ''
78test_atom head contents:body ''
79test_atom head contents:signature ''
80test_atom head contents 'Initial
81'
82test_atom head HEAD '*'
83
84test_atom tag refname refs/tags/testtag
85test_atom tag upstream ''
86test_atom tag objecttype tag
87test_atom tag objectsize 154
88test_atom tag objectname $(git rev-parse refs/tags/testtag)
89test_atom tag tree ''
90test_atom tag parent ''
91test_atom tag numparent ''
92test_atom tag object $(git rev-parse refs/tags/testtag^0)
93test_atom tag type 'commit'
94test_atom tag '*objectname' '67a36f10722846e891fbada1ba48ed035de75581'
95test_atom tag '*objecttype' 'commit'
96test_atom tag author ''
97test_atom tag authorname ''
98test_atom tag authoremail ''
99test_atom tag authordate ''
100test_atom tag committer ''
101test_atom tag committername ''
102test_atom tag committeremail ''
103test_atom tag committerdate ''
104test_atom tag tag 'testtag'
105test_atom tag tagger 'C O Mitter <committer@example.com> 1151939925 +0200'
106test_atom tag taggername 'C O Mitter'
107test_atom tag taggeremail '<committer@example.com>'
108test_atom tag taggerdate 'Mon Jul 3 17:18:45 2006 +0200'
109test_atom tag creator 'C O Mitter <committer@example.com> 1151939925 +0200'
110test_atom tag creatordate 'Mon Jul 3 17:18:45 2006 +0200'
111test_atom tag subject 'Tagging at 1151939927'
112test_atom tag contents:subject 'Tagging at 1151939927'
113test_atom tag body ''
114test_atom tag contents:body ''
115test_atom tag contents:signature ''
116test_atom tag contents 'Tagging at 1151939927
117'
118test_atom tag HEAD ' '
119
120test_expect_success 'Check invalid atoms names are errors' '
121 test_must_fail git for-each-ref --format="%(INVALID)" refs/heads
122'
123
124test_expect_success 'Check format specifiers are ignored in naming date atoms' '
125 git for-each-ref --format="%(authordate)" refs/heads &&
126 git for-each-ref --format="%(authordate:default) %(authordate)" refs/heads &&
127 git for-each-ref --format="%(authordate) %(authordate:default)" refs/heads &&
128 git for-each-ref --format="%(authordate:default) %(authordate:default)" refs/heads
129'
130
131test_expect_success 'Check valid format specifiers for date fields' '
132 git for-each-ref --format="%(authordate:default)" refs/heads &&
133 git for-each-ref --format="%(authordate:relative)" refs/heads &&
134 git for-each-ref --format="%(authordate:short)" refs/heads &&
135 git for-each-ref --format="%(authordate:local)" refs/heads &&
136 git for-each-ref --format="%(authordate:iso8601)" refs/heads &&
137 git for-each-ref --format="%(authordate:rfc2822)" refs/heads
138'
139
140test_expect_success 'Check invalid format specifiers are errors' '
141 test_must_fail git for-each-ref --format="%(authordate:INVALID)" refs/heads
142'
143
144cat >expected <<\EOF
145'refs/heads/master' 'Mon Jul 3 17:18:43 2006 +0200' 'Mon Jul 3 17:18:44 2006 +0200'
146'refs/tags/testtag' 'Mon Jul 3 17:18:45 2006 +0200'
147EOF
148
149test_expect_success 'Check unformatted date fields output' '
150 (git for-each-ref --shell --format="%(refname) %(committerdate) %(authordate)" refs/heads &&
151 git for-each-ref --shell --format="%(refname) %(taggerdate)" refs/tags) >actual &&
152 test_cmp expected actual
153'
154
155test_expect_success 'Check format "default" formatted date fields output' '
156 f=default &&
157 (git for-each-ref --shell --format="%(refname) %(committerdate:$f) %(authordate:$f)" refs/heads &&
158 git for-each-ref --shell --format="%(refname) %(taggerdate:$f)" refs/tags) >actual &&
159 test_cmp expected actual
160'
161
162# Don't know how to do relative check because I can't know when this script
163# is going to be run and can't fake the current time to git, and hence can't
164# provide expected output. Instead, I'll just make sure that "relative"
165# doesn't exit in error
166#
167#cat >expected <<\EOF
168#
169#EOF
170#
171test_expect_success 'Check format "relative" date fields output' '
172 f=relative &&
173 (git for-each-ref --shell --format="%(refname) %(committerdate:$f) %(authordate:$f)" refs/heads &&
174 git for-each-ref --shell --format="%(refname) %(taggerdate:$f)" refs/tags) >actual
175'
176
177cat >expected <<\EOF
178'refs/heads/master' '2006-07-03' '2006-07-03'
179'refs/tags/testtag' '2006-07-03'
180EOF
181
182test_expect_success 'Check format "short" date fields output' '
183 f=short &&
184 (git for-each-ref --shell --format="%(refname) %(committerdate:$f) %(authordate:$f)" refs/heads &&
185 git for-each-ref --shell --format="%(refname) %(taggerdate:$f)" refs/tags) >actual &&
186 test_cmp expected actual
187'
188
189cat >expected <<\EOF
190'refs/heads/master' 'Mon Jul 3 15:18:43 2006' 'Mon Jul 3 15:18:44 2006'
191'refs/tags/testtag' 'Mon Jul 3 15:18:45 2006'
192EOF
193
194test_expect_success 'Check format "local" date fields output' '
195 f=local &&
196 (git for-each-ref --shell --format="%(refname) %(committerdate:$f) %(authordate:$f)" refs/heads &&
197 git for-each-ref --shell --format="%(refname) %(taggerdate:$f)" refs/tags) >actual &&
198 test_cmp expected actual
199'
200
201cat >expected <<\EOF
202'refs/heads/master' '2006-07-03 17:18:43 +0200' '2006-07-03 17:18:44 +0200'
203'refs/tags/testtag' '2006-07-03 17:18:45 +0200'
204EOF
205
206test_expect_success 'Check format "iso8601" date fields output' '
207 f=iso8601 &&
208 (git for-each-ref --shell --format="%(refname) %(committerdate:$f) %(authordate:$f)" refs/heads &&
209 git for-each-ref --shell --format="%(refname) %(taggerdate:$f)" refs/tags) >actual &&
210 test_cmp expected actual
211'
212
213cat >expected <<\EOF
214'refs/heads/master' 'Mon, 3 Jul 2006 17:18:43 +0200' 'Mon, 3 Jul 2006 17:18:44 +0200'
215'refs/tags/testtag' 'Mon, 3 Jul 2006 17:18:45 +0200'
216EOF
217
218test_expect_success 'Check format "rfc2822" date fields output' '
219 f=rfc2822 &&
220 (git for-each-ref --shell --format="%(refname) %(committerdate:$f) %(authordate:$f)" refs/heads &&
221 git for-each-ref --shell --format="%(refname) %(taggerdate:$f)" refs/tags) >actual &&
222 test_cmp expected actual
223'
224
225cat >expected <<\EOF
226refs/heads/master
227refs/remotes/origin/master
228refs/tags/testtag
229EOF
230
231test_expect_success 'Verify ascending sort' '
232 git for-each-ref --format="%(refname)" --sort=refname >actual &&
233 test_cmp expected actual
234'
235
236
237cat >expected <<\EOF
238refs/tags/testtag
239refs/remotes/origin/master
240refs/heads/master
241EOF
242
243test_expect_success 'Verify descending sort' '
244 git for-each-ref --format="%(refname)" --sort=-refname >actual &&
245 test_cmp expected actual
246'
247
248cat >expected <<\EOF
249'refs/heads/master'
250'refs/remotes/origin/master'
251'refs/tags/testtag'
252EOF
253
254test_expect_success 'Quoting style: shell' '
255 git for-each-ref --shell --format="%(refname)" >actual &&
256 test_cmp expected actual
257'
258
259test_expect_success 'Quoting style: perl' '
260 git for-each-ref --perl --format="%(refname)" >actual &&
261 test_cmp expected actual
262'
263
264test_expect_success 'Quoting style: python' '
265 git for-each-ref --python --format="%(refname)" >actual &&
266 test_cmp expected actual
267'
268
269cat >expected <<\EOF
270"refs/heads/master"
271"refs/remotes/origin/master"
272"refs/tags/testtag"
273EOF
274
275test_expect_success 'Quoting style: tcl' '
276 git for-each-ref --tcl --format="%(refname)" >actual &&
277 test_cmp expected actual
278'
279
280for i in "--perl --shell" "-s --python" "--python --tcl" "--tcl --perl"; do
281 test_expect_success "more than one quoting style: $i" "
282 git for-each-ref $i 2>&1 | (read line &&
283 case \$line in
284 \"error: more than one quoting style\"*) : happy;;
285 *) false
286 esac)
287 "
288done
289
290cat >expected <<\EOF
291master
292testtag
293EOF
294
295test_expect_success 'Check short refname format' '
296 (git for-each-ref --format="%(refname:short)" refs/heads &&
297 git for-each-ref --format="%(refname:short)" refs/tags) >actual &&
298 test_cmp expected actual
299'
300
301cat >expected <<EOF
302origin/master
303EOF
304
305test_expect_success 'Check short upstream format' '
306 git for-each-ref --format="%(upstream:short)" refs/heads >actual &&
307 test_cmp expected actual
308'
309
310test_expect_success 'setup for upstream:track[short]' '
311 test_commit two
312'
313
314cat >expected <<EOF
315[ahead 1]
316EOF
317
318test_expect_success 'Check upstream:track format' '
319 git for-each-ref --format="%(upstream:track)" refs/heads >actual &&
320 test_cmp expected actual
321'
322
323cat >expected <<EOF
324>
325EOF
326
327test_expect_success 'Check upstream:trackshort format' '
328 git for-each-ref --format="%(upstream:trackshort)" refs/heads >actual &&
329 test_cmp expected actual
330'
331
332test_expect_success 'Check that :track[short] cannot be used with other atoms' '
333 test_must_fail git for-each-ref --format="%(refname:track)" 2>/dev/null &&
334 test_must_fail git for-each-ref --format="%(refname:trackshort)" 2>/dev/null
335'
336
337test_expect_success 'Check that :track[short] works when upstream is invalid' '
338 cat >expected <<-\EOF &&
339
340
341 EOF
342 test_when_finished "git config branch.master.merge refs/heads/master" &&
343 git config branch.master.merge refs/heads/does-not-exist &&
344 git for-each-ref \
345 --format="%(upstream:track)$LF%(upstream:trackshort)" \
346 refs/heads >actual &&
347 test_cmp expected actual
348'
349
350cat >expected <<EOF
351$(git rev-parse --short HEAD)
352EOF
353
354test_expect_success 'Check short objectname format' '
355 git for-each-ref --format="%(objectname:short)" refs/heads >actual &&
356 test_cmp expected actual
357'
358
359test_expect_success 'Check for invalid refname format' '
360 test_must_fail git for-each-ref --format="%(refname:INVALID)"
361'
362
363get_color ()
364{
365 git config --get-color no.such.slot "$1"
366}
367
368cat >expected <<EOF
369$(git rev-parse --short refs/heads/master) $(get_color green)master$(get_color reset)
370$(git rev-parse --short refs/remotes/origin/master) $(get_color green)origin/master$(get_color reset)
371$(git rev-parse --short refs/tags/testtag) $(get_color green)testtag$(get_color reset)
372$(git rev-parse --short refs/tags/two) $(get_color green)two$(get_color reset)
373EOF
374
375test_expect_success 'Check %(color:...) ' '
376 git for-each-ref --format="%(objectname:short) %(color:green)%(refname:short)" >actual &&
377 test_cmp expected actual
378'
379
380cat >expected <<\EOF
381heads/master
382tags/master
383EOF
384
385test_expect_success 'Check ambiguous head and tag refs (strict)' '
386 git config --bool core.warnambiguousrefs true &&
387 git checkout -b newtag &&
388 echo "Using $datestamp" > one &&
389 git add one &&
390 git commit -m "Branch" &&
391 setdate_and_increment &&
392 git tag -m "Tagging at $datestamp" master &&
393 git for-each-ref --format "%(refname:short)" refs/heads/master refs/tags/master >actual &&
394 test_cmp expected actual
395'
396
397cat >expected <<\EOF
398heads/master
399master
400EOF
401
402test_expect_success 'Check ambiguous head and tag refs (loose)' '
403 git config --bool core.warnambiguousrefs false &&
404 git for-each-ref --format "%(refname:short)" refs/heads/master refs/tags/master >actual &&
405 test_cmp expected actual
406'
407
408cat >expected <<\EOF
409heads/ambiguous
410ambiguous
411EOF
412
413test_expect_success 'Check ambiguous head and tag refs II (loose)' '
414 git checkout master &&
415 git tag ambiguous testtag^0 &&
416 git branch ambiguous testtag^0 &&
417 git for-each-ref --format "%(refname:short)" refs/heads/ambiguous refs/tags/ambiguous >actual &&
418 test_cmp expected actual
419'
420
421test_expect_success 'an unusual tag with an incomplete line' '
422
423 git tag -m "bogo" bogo &&
424 bogo=$(git cat-file tag bogo) &&
425 bogo=$(printf "%s" "$bogo" | git mktag) &&
426 git tag -f bogo "$bogo" &&
427 git for-each-ref --format "%(body)" refs/tags/bogo
428
429'
430
431test_expect_success 'create tag with subject and body content' '
432 cat >>msg <<-\EOF &&
433 the subject line
434
435 first body line
436 second body line
437 EOF
438 git tag -F msg subject-body
439'
440test_atom refs/tags/subject-body subject 'the subject line'
441test_atom refs/tags/subject-body body 'first body line
442second body line
443'
444test_atom refs/tags/subject-body contents 'the subject line
445
446first body line
447second body line
448'
449
450test_expect_success 'create tag with multiline subject' '
451 cat >msg <<-\EOF &&
452 first subject line
453 second subject line
454
455 first body line
456 second body line
457 EOF
458 git tag -F msg multiline
459'
460test_atom refs/tags/multiline subject 'first subject line second subject line'
461test_atom refs/tags/multiline contents:subject 'first subject line second subject line'
462test_atom refs/tags/multiline body 'first body line
463second body line
464'
465test_atom refs/tags/multiline contents:body 'first body line
466second body line
467'
468test_atom refs/tags/multiline contents:signature ''
469test_atom refs/tags/multiline contents 'first subject line
470second subject line
471
472first body line
473second body line
474'
475
476test_expect_success GPG 'create signed tags' '
477 git tag -s -m "" signed-empty &&
478 git tag -s -m "subject line" signed-short &&
479 cat >msg <<-\EOF &&
480 subject line
481
482 body contents
483 EOF
484 git tag -s -F msg signed-long
485'
486
487sig='-----BEGIN PGP SIGNATURE-----
488-----END PGP SIGNATURE-----
489'
490
491PREREQ=GPG
492test_atom refs/tags/signed-empty subject ''
493test_atom refs/tags/signed-empty contents:subject ''
494test_atom refs/tags/signed-empty body "$sig"
495test_atom refs/tags/signed-empty contents:body ''
496test_atom refs/tags/signed-empty contents:signature "$sig"
497test_atom refs/tags/signed-empty contents "$sig"
498
499test_atom refs/tags/signed-short subject 'subject line'
500test_atom refs/tags/signed-short contents:subject 'subject line'
501test_atom refs/tags/signed-short body "$sig"
502test_atom refs/tags/signed-short contents:body ''
503test_atom refs/tags/signed-short contents:signature "$sig"
504test_atom refs/tags/signed-short contents "subject line
505$sig"
506
507test_atom refs/tags/signed-long subject 'subject line'
508test_atom refs/tags/signed-long contents:subject 'subject line'
509test_atom refs/tags/signed-long body "body contents
510$sig"
511test_atom refs/tags/signed-long contents:body 'body contents
512'
513test_atom refs/tags/signed-long contents:signature "$sig"
514test_atom refs/tags/signed-long contents "subject line
515
516body contents
517$sig"
518
519cat >expected <<EOF
520$(git rev-parse refs/tags/master) <committer@example.com> refs/tags/master
521$(git rev-parse refs/tags/bogo) <committer@example.com> refs/tags/bogo
522EOF
523
524test_expect_success 'Verify sort with multiple keys' '
525 git for-each-ref --format="%(objectname) %(taggeremail) %(refname)" --sort=objectname --sort=taggeremail \
526 refs/tags/bogo refs/tags/master > actual &&
527 test_cmp expected actual
528'
529test_done