The `git-for-each-ref(1)` command is used to iterate over references
present in a repository. In large repositories with millions of
references, it would be optimal to paginate this output such that we
can start iteration from a given reference. This would avoid having to
iterate over all references from the beginning each time when paginating
through results.
The previous commit added 'seek' functionality to the reference
backends. Utilize this and expose a '--start-after' option in
'git-for-each-ref(1)'. When used, the reference iteration seeks to the
lexicographically next reference and iterates from there onward.
This enables efficient pagination workflows, where the calling script
can remember the last provided reference and use that as the starting
point for the next set of references:
git for-each-ref --count=100
git for-each-ref --count=100 --start-after=refs/heads/branch-100
git for-each-ref --count=100 --start-after=refs/heads/branch-200
Since the reference iterators only allow seeking to a specified marker
via the `ref_iterator_seek()`, we introduce a helper function
`start_ref_iterator_after()`, which seeks to next reference by simply
adding (char) 1 to the marker.
We must note that pagination always continues from the provided marker,
as such any concurrent reference updates lexicographically behind the
marker will not be output. Document the same.
Signed-off-by: Karthik Nayak <karthik.188@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
739 lines
20 KiB
Bash
Executable File
739 lines
20 KiB
Bash
Executable File
#!/bin/sh
|
|
|
|
test_description='test for-each-refs usage of ref-filter APIs'
|
|
|
|
. ./test-lib.sh
|
|
. "$TEST_DIRECTORY"/lib-gpg.sh
|
|
|
|
test_expect_success 'setup some history and refs' '
|
|
test_commit one &&
|
|
git branch -M main &&
|
|
test_commit two &&
|
|
test_commit three &&
|
|
git checkout -b side &&
|
|
test_commit four &&
|
|
git tag -m "An annotated tag" annotated-tag &&
|
|
git tag -m "Annotated doubly" doubly-annotated-tag annotated-tag &&
|
|
|
|
# Note that these "signed" tags might not actually be signed.
|
|
# Tests which care about the distinction should be marked
|
|
# with the GPG prereq.
|
|
if test_have_prereq GPG
|
|
then
|
|
sign=-s
|
|
else
|
|
sign=
|
|
fi &&
|
|
git tag $sign -m "A signed tag" signed-tag &&
|
|
git tag $sign -m "Signed doubly" doubly-signed-tag signed-tag &&
|
|
|
|
git checkout main &&
|
|
git update-ref refs/odd/spot main
|
|
'
|
|
|
|
test_expect_success '--include-root-refs pattern prints pseudorefs' '
|
|
cat >expect <<-\EOF &&
|
|
HEAD
|
|
ORIG_HEAD
|
|
refs/heads/main
|
|
refs/heads/side
|
|
refs/odd/spot
|
|
refs/tags/annotated-tag
|
|
refs/tags/doubly-annotated-tag
|
|
refs/tags/doubly-signed-tag
|
|
refs/tags/four
|
|
refs/tags/one
|
|
refs/tags/signed-tag
|
|
refs/tags/three
|
|
refs/tags/two
|
|
EOF
|
|
git update-ref ORIG_HEAD main &&
|
|
git for-each-ref --format="%(refname)" --include-root-refs >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success '--include-root-refs pattern does not print special refs' '
|
|
test_when_finished "rm -rf repo" &&
|
|
git init repo &&
|
|
(
|
|
cd repo &&
|
|
test_commit initial &&
|
|
git rev-parse HEAD >.git/MERGE_HEAD &&
|
|
git for-each-ref --format="%(refname)" --include-root-refs >actual &&
|
|
cat >expect <<-EOF &&
|
|
HEAD
|
|
$(git symbolic-ref HEAD)
|
|
refs/tags/initial
|
|
EOF
|
|
test_cmp expect actual
|
|
)
|
|
'
|
|
|
|
test_expect_success '--include-root-refs with other patterns' '
|
|
cat >expect <<-\EOF &&
|
|
HEAD
|
|
ORIG_HEAD
|
|
EOF
|
|
git update-ref ORIG_HEAD main &&
|
|
git for-each-ref --format="%(refname)" --include-root-refs "*HEAD" >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success '--include-root-refs omits dangling symrefs' '
|
|
test_when_finished "rm -rf repo" &&
|
|
git init repo &&
|
|
(
|
|
cd repo &&
|
|
test_commit initial &&
|
|
git symbolic-ref DANGLING_HEAD refs/heads/missing &&
|
|
cat >expect <<-EOF &&
|
|
HEAD
|
|
$(git symbolic-ref HEAD)
|
|
refs/tags/initial
|
|
EOF
|
|
git for-each-ref --format="%(refname)" --include-root-refs >actual &&
|
|
test_cmp expect actual
|
|
)
|
|
'
|
|
|
|
test_expect_success 'filtering with --points-at' '
|
|
cat >expect <<-\EOF &&
|
|
refs/heads/main
|
|
refs/odd/spot
|
|
refs/tags/three
|
|
EOF
|
|
git for-each-ref --format="%(refname)" --points-at=main >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'check signed tags with --points-at' '
|
|
sed -e "s/Z$//" >expect <<-\EOF &&
|
|
refs/heads/side Z
|
|
refs/tags/annotated-tag four
|
|
refs/tags/doubly-annotated-tag four
|
|
refs/tags/doubly-signed-tag four
|
|
refs/tags/four Z
|
|
refs/tags/signed-tag four
|
|
EOF
|
|
git for-each-ref --format="%(refname) %(*subject)" --points-at=side >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'filtering with --merged' '
|
|
cat >expect <<-\EOF &&
|
|
refs/heads/main
|
|
refs/odd/spot
|
|
refs/tags/one
|
|
refs/tags/three
|
|
refs/tags/two
|
|
EOF
|
|
git for-each-ref --format="%(refname)" --merged=main >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'filtering with --no-merged' '
|
|
cat >expect <<-\EOF &&
|
|
refs/heads/side
|
|
refs/tags/annotated-tag
|
|
refs/tags/doubly-annotated-tag
|
|
refs/tags/doubly-signed-tag
|
|
refs/tags/four
|
|
refs/tags/signed-tag
|
|
EOF
|
|
git for-each-ref --format="%(refname)" --no-merged=main >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'filtering with --contains' '
|
|
cat >expect <<-\EOF &&
|
|
refs/heads/main
|
|
refs/heads/side
|
|
refs/odd/spot
|
|
refs/tags/annotated-tag
|
|
refs/tags/doubly-annotated-tag
|
|
refs/tags/doubly-signed-tag
|
|
refs/tags/four
|
|
refs/tags/signed-tag
|
|
refs/tags/three
|
|
refs/tags/two
|
|
EOF
|
|
git for-each-ref --format="%(refname)" --contains=two >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'filtering with --no-contains' '
|
|
cat >expect <<-\EOF &&
|
|
refs/tags/one
|
|
EOF
|
|
git for-each-ref --format="%(refname)" --no-contains=two >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'filtering with --contains and --no-contains' '
|
|
cat >expect <<-\EOF &&
|
|
refs/tags/two
|
|
EOF
|
|
git for-each-ref --format="%(refname)" --contains=two --no-contains=three >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success '%(color) must fail' '
|
|
test_must_fail git for-each-ref --format="%(color)%(refname)"
|
|
'
|
|
|
|
test_expect_success '%(color:#aa22ac) must succeed' '
|
|
test_when_finished rm -rf test &&
|
|
git init test &&
|
|
(
|
|
cd test &&
|
|
test_commit initial &&
|
|
git branch -M main &&
|
|
cat >expect <<-\EOF &&
|
|
refs/heads/main
|
|
refs/tags/initial
|
|
EOF
|
|
git remote add origin nowhere &&
|
|
git config branch.main.remote origin &&
|
|
git config branch.main.merge refs/heads/main &&
|
|
git for-each-ref --format="%(color:#aa22ac)%(refname)" >actual &&
|
|
test_cmp expect actual
|
|
)
|
|
'
|
|
|
|
test_expect_success 'left alignment is default' '
|
|
cat >expect <<-\EOF &&
|
|
refname is refs/heads/main |refs/heads/main
|
|
refname is refs/heads/side |refs/heads/side
|
|
refname is refs/odd/spot |refs/odd/spot
|
|
refname is refs/tags/annotated-tag|refs/tags/annotated-tag
|
|
refname is refs/tags/doubly-annotated-tag|refs/tags/doubly-annotated-tag
|
|
refname is refs/tags/doubly-signed-tag|refs/tags/doubly-signed-tag
|
|
refname is refs/tags/four |refs/tags/four
|
|
refname is refs/tags/one |refs/tags/one
|
|
refname is refs/tags/signed-tag|refs/tags/signed-tag
|
|
refname is refs/tags/three |refs/tags/three
|
|
refname is refs/tags/two |refs/tags/two
|
|
EOF
|
|
git for-each-ref --format="%(align:30)refname is %(refname)%(end)|%(refname)" >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'middle alignment' '
|
|
cat >expect <<-\EOF &&
|
|
| refname is refs/heads/main |refs/heads/main
|
|
| refname is refs/heads/side |refs/heads/side
|
|
| refname is refs/odd/spot |refs/odd/spot
|
|
|refname is refs/tags/annotated-tag|refs/tags/annotated-tag
|
|
|refname is refs/tags/doubly-annotated-tag|refs/tags/doubly-annotated-tag
|
|
|refname is refs/tags/doubly-signed-tag|refs/tags/doubly-signed-tag
|
|
| refname is refs/tags/four |refs/tags/four
|
|
| refname is refs/tags/one |refs/tags/one
|
|
|refname is refs/tags/signed-tag|refs/tags/signed-tag
|
|
| refname is refs/tags/three |refs/tags/three
|
|
| refname is refs/tags/two |refs/tags/two
|
|
EOF
|
|
git for-each-ref --format="|%(align:middle,30)refname is %(refname)%(end)|%(refname)" >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'right alignment' '
|
|
cat >expect <<-\EOF &&
|
|
| refname is refs/heads/main|refs/heads/main
|
|
| refname is refs/heads/side|refs/heads/side
|
|
| refname is refs/odd/spot|refs/odd/spot
|
|
|refname is refs/tags/annotated-tag|refs/tags/annotated-tag
|
|
|refname is refs/tags/doubly-annotated-tag|refs/tags/doubly-annotated-tag
|
|
|refname is refs/tags/doubly-signed-tag|refs/tags/doubly-signed-tag
|
|
| refname is refs/tags/four|refs/tags/four
|
|
| refname is refs/tags/one|refs/tags/one
|
|
|refname is refs/tags/signed-tag|refs/tags/signed-tag
|
|
| refname is refs/tags/three|refs/tags/three
|
|
| refname is refs/tags/two|refs/tags/two
|
|
EOF
|
|
git for-each-ref --format="|%(align:30,right)refname is %(refname)%(end)|%(refname)" >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
cat >expect <<-\EOF
|
|
| refname is refs/heads/main |refs/heads/main
|
|
| refname is refs/heads/side |refs/heads/side
|
|
| refname is refs/odd/spot |refs/odd/spot
|
|
| refname is refs/tags/annotated-tag |refs/tags/annotated-tag
|
|
|refname is refs/tags/doubly-annotated-tag |refs/tags/doubly-annotated-tag
|
|
| refname is refs/tags/doubly-signed-tag |refs/tags/doubly-signed-tag
|
|
| refname is refs/tags/four |refs/tags/four
|
|
| refname is refs/tags/one |refs/tags/one
|
|
| refname is refs/tags/signed-tag |refs/tags/signed-tag
|
|
| refname is refs/tags/three |refs/tags/three
|
|
| refname is refs/tags/two |refs/tags/two
|
|
EOF
|
|
|
|
test_align_permutations() {
|
|
while read -r option
|
|
do
|
|
test_expect_success "align:$option" '
|
|
git for-each-ref --format="|%(align:$option)refname is %(refname)%(end)|%(refname)" >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
done
|
|
}
|
|
|
|
test_align_permutations <<-\EOF
|
|
middle,42
|
|
42,middle
|
|
position=middle,42
|
|
42,position=middle
|
|
middle,width=42
|
|
width=42,middle
|
|
position=middle,width=42
|
|
width=42,position=middle
|
|
EOF
|
|
|
|
# Last one wins (silently) when multiple arguments of the same type are given
|
|
|
|
test_align_permutations <<-\EOF
|
|
32,width=42,middle
|
|
width=30,42,middle
|
|
width=42,position=right,middle
|
|
42,right,position=middle
|
|
EOF
|
|
|
|
# Individual atoms inside %(align:...) and %(end) must not be quoted.
|
|
|
|
test_expect_success 'alignment with format quote' "
|
|
cat >expect <<-\EOF &&
|
|
|' '\''main| A U Thor'\'' '|
|
|
|' '\''side| A U Thor'\'' '|
|
|
|' '\''odd/spot| A U Thor'\'' '|
|
|
|' '\''annotated-tag| '\'' '|
|
|
|' '\''doubly-annotated-tag| '\'' '|
|
|
|' '\''doubly-signed-tag| '\'' '|
|
|
|' '\''four| A U Thor'\'' '|
|
|
|' '\''one| A U Thor'\'' '|
|
|
|' '\''signed-tag| '\'' '|
|
|
|' '\''three| A U Thor'\'' '|
|
|
|' '\''two| A U Thor'\'' '|
|
|
EOF
|
|
git for-each-ref --shell --format=\"|%(align:30,middle)'%(refname:short)| %(authorname)'%(end)|\" >actual &&
|
|
test_cmp expect actual
|
|
"
|
|
|
|
test_expect_success 'nested alignment with quote formatting' "
|
|
cat >expect <<-\EOF &&
|
|
|' main '|
|
|
|' side '|
|
|
|' odd/spot '|
|
|
|' annotated-tag '|
|
|
|'doubly-annotated-tag '|
|
|
|'doubly-signed-tag '|
|
|
|' four '|
|
|
|' one '|
|
|
|' signed-tag '|
|
|
|' three '|
|
|
|' two '|
|
|
EOF
|
|
git for-each-ref --shell --format='|%(align:30,left)%(align:15,right)%(refname:short)%(end)%(end)|' >actual &&
|
|
test_cmp expect actual
|
|
"
|
|
|
|
test_expect_success 'check `%(contents:lines=1)`' '
|
|
cat >expect <<-\EOF &&
|
|
main |three
|
|
side |four
|
|
odd/spot |three
|
|
annotated-tag |An annotated tag
|
|
doubly-annotated-tag |Annotated doubly
|
|
doubly-signed-tag |Signed doubly
|
|
four |four
|
|
one |one
|
|
signed-tag |A signed tag
|
|
three |three
|
|
two |two
|
|
EOF
|
|
git for-each-ref --format="%(refname:short) |%(contents:lines=1)" >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'check `%(contents:lines=0)`' '
|
|
cat >expect <<-\EOF &&
|
|
main |
|
|
side |
|
|
odd/spot |
|
|
annotated-tag |
|
|
doubly-annotated-tag |
|
|
doubly-signed-tag |
|
|
four |
|
|
one |
|
|
signed-tag |
|
|
three |
|
|
two |
|
|
EOF
|
|
git for-each-ref --format="%(refname:short) |%(contents:lines=0)" >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'check `%(contents:lines=99999)`' '
|
|
cat >expect <<-\EOF &&
|
|
main |three
|
|
side |four
|
|
odd/spot |three
|
|
annotated-tag |An annotated tag
|
|
doubly-annotated-tag |Annotated doubly
|
|
doubly-signed-tag |Signed doubly
|
|
four |four
|
|
one |one
|
|
signed-tag |A signed tag
|
|
three |three
|
|
two |two
|
|
EOF
|
|
git for-each-ref --format="%(refname:short) |%(contents:lines=99999)" >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success '`%(contents:lines=-1)` should fail' '
|
|
test_must_fail git for-each-ref --format="%(refname:short) |%(contents:lines=-1)"
|
|
'
|
|
|
|
test_expect_success 'setup for version sort' '
|
|
test_commit foo1.3 &&
|
|
test_commit foo1.6 &&
|
|
test_commit foo1.10
|
|
'
|
|
|
|
test_expect_success 'version sort' '
|
|
git for-each-ref --sort=version:refname --format="%(refname:short)" refs/tags/ | grep "foo" >actual &&
|
|
cat >expect <<-\EOF &&
|
|
foo1.3
|
|
foo1.6
|
|
foo1.10
|
|
EOF
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'version sort (shortened)' '
|
|
git for-each-ref --sort=v:refname --format="%(refname:short)" refs/tags/ | grep "foo" >actual &&
|
|
cat >expect <<-\EOF &&
|
|
foo1.3
|
|
foo1.6
|
|
foo1.10
|
|
EOF
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'reverse version sort' '
|
|
git for-each-ref --sort=-version:refname --format="%(refname:short)" refs/tags/ | grep "foo" >actual &&
|
|
cat >expect <<-\EOF &&
|
|
foo1.10
|
|
foo1.6
|
|
foo1.3
|
|
EOF
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'improper usage of %(if), %(then), %(else) and %(end) atoms' '
|
|
test_must_fail git for-each-ref --format="%(if)" &&
|
|
test_must_fail git for-each-ref --format="%(then) %(end)" &&
|
|
test_must_fail git for-each-ref --format="%(else) %(end)" &&
|
|
test_must_fail git for-each-ref --format="%(if) %(else) %(end)" &&
|
|
test_must_fail git for-each-ref --format="%(if) %(then) %(then) %(end)" &&
|
|
test_must_fail git for-each-ref --format="%(then) %(else) %(end)" &&
|
|
test_must_fail git for-each-ref --format="%(if) %(else) %(end)" &&
|
|
test_must_fail git for-each-ref --format="%(if) %(then) %(else)" &&
|
|
test_must_fail git for-each-ref --format="%(if) %(else) %(then) %(end)" &&
|
|
test_must_fail git for-each-ref --format="%(if) %(then) %(else) %(else) %(end)" &&
|
|
test_must_fail git for-each-ref --format="%(if) %(end)"
|
|
'
|
|
|
|
test_expect_success 'check %(if)...%(then)...%(end) atoms' '
|
|
git for-each-ref --format="%(refname)%(if)%(authorname)%(then) Author: %(authorname)%(end)" >actual &&
|
|
cat >expect <<-\EOF &&
|
|
refs/heads/main Author: A U Thor
|
|
refs/heads/side Author: A U Thor
|
|
refs/odd/spot Author: A U Thor
|
|
refs/tags/annotated-tag
|
|
refs/tags/doubly-annotated-tag
|
|
refs/tags/doubly-signed-tag
|
|
refs/tags/foo1.10 Author: A U Thor
|
|
refs/tags/foo1.3 Author: A U Thor
|
|
refs/tags/foo1.6 Author: A U Thor
|
|
refs/tags/four Author: A U Thor
|
|
refs/tags/one Author: A U Thor
|
|
refs/tags/signed-tag
|
|
refs/tags/three Author: A U Thor
|
|
refs/tags/two Author: A U Thor
|
|
EOF
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'check %(if)...%(then)...%(else)...%(end) atoms' '
|
|
git for-each-ref --format="%(if)%(authorname)%(then)%(authorname)%(else)No author%(end): %(refname)" >actual &&
|
|
cat >expect <<-\EOF &&
|
|
A U Thor: refs/heads/main
|
|
A U Thor: refs/heads/side
|
|
A U Thor: refs/odd/spot
|
|
No author: refs/tags/annotated-tag
|
|
No author: refs/tags/doubly-annotated-tag
|
|
No author: refs/tags/doubly-signed-tag
|
|
A U Thor: refs/tags/foo1.10
|
|
A U Thor: refs/tags/foo1.3
|
|
A U Thor: refs/tags/foo1.6
|
|
A U Thor: refs/tags/four
|
|
A U Thor: refs/tags/one
|
|
No author: refs/tags/signed-tag
|
|
A U Thor: refs/tags/three
|
|
A U Thor: refs/tags/two
|
|
EOF
|
|
test_cmp expect actual
|
|
'
|
|
test_expect_success 'ignore spaces in %(if) atom usage' '
|
|
git for-each-ref --format="%(refname:short): %(if)%(HEAD)%(then)Head ref%(else)Not Head ref%(end)" >actual &&
|
|
cat >expect <<-\EOF &&
|
|
main: Head ref
|
|
side: Not Head ref
|
|
odd/spot: Not Head ref
|
|
annotated-tag: Not Head ref
|
|
doubly-annotated-tag: Not Head ref
|
|
doubly-signed-tag: Not Head ref
|
|
foo1.10: Not Head ref
|
|
foo1.3: Not Head ref
|
|
foo1.6: Not Head ref
|
|
four: Not Head ref
|
|
one: Not Head ref
|
|
signed-tag: Not Head ref
|
|
three: Not Head ref
|
|
two: Not Head ref
|
|
EOF
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'check %(if:equals=<string>)' '
|
|
git for-each-ref --format="%(if:equals=main)%(refname:short)%(then)Found main%(else)Not main%(end)" refs/heads/ >actual &&
|
|
cat >expect <<-\EOF &&
|
|
Found main
|
|
Not main
|
|
EOF
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'check %(if:notequals=<string>)' '
|
|
git for-each-ref --format="%(if:notequals=main)%(refname:short)%(then)Not main%(else)Found main%(end)" refs/heads/ >actual &&
|
|
cat >expect <<-\EOF &&
|
|
Found main
|
|
Not main
|
|
EOF
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success '--merged is compatible with --no-merged' '
|
|
git for-each-ref --merged HEAD --no-merged HEAD
|
|
'
|
|
|
|
test_expect_success 'validate worktree atom' '
|
|
cat >expect <<-EOF &&
|
|
main: $(pwd)
|
|
main_worktree: $(pwd)/worktree_dir
|
|
side: not checked out
|
|
EOF
|
|
git worktree add -b main_worktree worktree_dir main &&
|
|
git for-each-ref --format="%(refname:short): %(if)%(worktreepath)%(then)%(worktreepath)%(else)not checked out%(end)" refs/heads/ >actual &&
|
|
rm -r worktree_dir &&
|
|
git worktree prune &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'start after with empty value' '
|
|
cat >expect <<-\EOF &&
|
|
refs/heads/main
|
|
refs/heads/main_worktree
|
|
refs/heads/side
|
|
refs/odd/spot
|
|
refs/tags/annotated-tag
|
|
refs/tags/doubly-annotated-tag
|
|
refs/tags/doubly-signed-tag
|
|
refs/tags/foo1.10
|
|
refs/tags/foo1.3
|
|
refs/tags/foo1.6
|
|
refs/tags/four
|
|
refs/tags/one
|
|
refs/tags/signed-tag
|
|
refs/tags/three
|
|
refs/tags/two
|
|
EOF
|
|
git for-each-ref --format="%(refname)" --start-after="" >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'start after a specific reference' '
|
|
cat >expect <<-\EOF &&
|
|
refs/tags/annotated-tag
|
|
refs/tags/doubly-annotated-tag
|
|
refs/tags/doubly-signed-tag
|
|
refs/tags/foo1.10
|
|
refs/tags/foo1.3
|
|
refs/tags/foo1.6
|
|
refs/tags/four
|
|
refs/tags/one
|
|
refs/tags/signed-tag
|
|
refs/tags/three
|
|
refs/tags/two
|
|
EOF
|
|
git for-each-ref --format="%(refname)" --start-after=refs/odd/spot >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'start after a specific reference with partial match' '
|
|
cat >expect <<-\EOF &&
|
|
refs/odd/spot
|
|
refs/tags/annotated-tag
|
|
refs/tags/doubly-annotated-tag
|
|
refs/tags/doubly-signed-tag
|
|
refs/tags/foo1.10
|
|
refs/tags/foo1.3
|
|
refs/tags/foo1.6
|
|
refs/tags/four
|
|
refs/tags/one
|
|
refs/tags/signed-tag
|
|
refs/tags/three
|
|
refs/tags/two
|
|
EOF
|
|
git for-each-ref --format="%(refname)" --start-after=refs/odd/sp >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'start after, just behind a specific reference' '
|
|
cat >expect <<-\EOF &&
|
|
refs/odd/spot
|
|
refs/tags/annotated-tag
|
|
refs/tags/doubly-annotated-tag
|
|
refs/tags/doubly-signed-tag
|
|
refs/tags/foo1.10
|
|
refs/tags/foo1.3
|
|
refs/tags/foo1.6
|
|
refs/tags/four
|
|
refs/tags/one
|
|
refs/tags/signed-tag
|
|
refs/tags/three
|
|
refs/tags/two
|
|
EOF
|
|
git for-each-ref --format="%(refname)" --start-after=refs/odd/parrot >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'start after with specific directory match' '
|
|
cat >expect <<-\EOF &&
|
|
refs/odd/spot
|
|
refs/tags/annotated-tag
|
|
refs/tags/doubly-annotated-tag
|
|
refs/tags/doubly-signed-tag
|
|
refs/tags/foo1.10
|
|
refs/tags/foo1.3
|
|
refs/tags/foo1.6
|
|
refs/tags/four
|
|
refs/tags/one
|
|
refs/tags/signed-tag
|
|
refs/tags/three
|
|
refs/tags/two
|
|
EOF
|
|
git for-each-ref --format="%(refname)" --start-after=refs/odd >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'start after with specific directory and trailing slash' '
|
|
cat >expect <<-\EOF &&
|
|
refs/odd/spot
|
|
refs/tags/annotated-tag
|
|
refs/tags/doubly-annotated-tag
|
|
refs/tags/doubly-signed-tag
|
|
refs/tags/foo1.10
|
|
refs/tags/foo1.3
|
|
refs/tags/foo1.6
|
|
refs/tags/four
|
|
refs/tags/one
|
|
refs/tags/signed-tag
|
|
refs/tags/three
|
|
refs/tags/two
|
|
EOF
|
|
git for-each-ref --format="%(refname)" --start-after=refs/odd/ >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'start after, just behind a specific directory' '
|
|
cat >expect <<-\EOF &&
|
|
refs/odd/spot
|
|
refs/tags/annotated-tag
|
|
refs/tags/doubly-annotated-tag
|
|
refs/tags/doubly-signed-tag
|
|
refs/tags/foo1.10
|
|
refs/tags/foo1.3
|
|
refs/tags/foo1.6
|
|
refs/tags/four
|
|
refs/tags/one
|
|
refs/tags/signed-tag
|
|
refs/tags/three
|
|
refs/tags/two
|
|
EOF
|
|
git for-each-ref --format="%(refname)" --start-after=refs/lost >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'start after, overflow specific reference length' '
|
|
cat >expect <<-\EOF &&
|
|
refs/tags/annotated-tag
|
|
refs/tags/doubly-annotated-tag
|
|
refs/tags/doubly-signed-tag
|
|
refs/tags/foo1.10
|
|
refs/tags/foo1.3
|
|
refs/tags/foo1.6
|
|
refs/tags/four
|
|
refs/tags/one
|
|
refs/tags/signed-tag
|
|
refs/tags/three
|
|
refs/tags/two
|
|
EOF
|
|
git for-each-ref --format="%(refname)" --start-after=refs/odd/spotnew >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'start after, overflow specific reference path' '
|
|
cat >expect <<-\EOF &&
|
|
refs/tags/annotated-tag
|
|
refs/tags/doubly-annotated-tag
|
|
refs/tags/doubly-signed-tag
|
|
refs/tags/foo1.10
|
|
refs/tags/foo1.3
|
|
refs/tags/foo1.6
|
|
refs/tags/four
|
|
refs/tags/one
|
|
refs/tags/signed-tag
|
|
refs/tags/three
|
|
refs/tags/two
|
|
EOF
|
|
git for-each-ref --format="%(refname)" --start-after=refs/odd/spot/new >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'start after, last reference' '
|
|
cat >expect <<-\EOF &&
|
|
EOF
|
|
git for-each-ref --format="%(refname)" --start-after=refs/tags/two >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'start after used with a pattern' '
|
|
cat >expect <<-\EOF &&
|
|
fatal: cannot use --start-after with patterns
|
|
EOF
|
|
test_must_fail git for-each-ref --format="%(refname)" --start-after=refs/odd/spot refs/tags 2>actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'start after used with custom sort order' '
|
|
cat >expect <<-\EOF &&
|
|
fatal: cannot use --start-after with custom sort options
|
|
EOF
|
|
test_must_fail git for-each-ref --format="%(refname)" --start-after=refs/odd/spot --sort=author 2>actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_done
|