2007-01-14 03:37:32 +01:00
|
|
|
#!/bin/sh
|
|
|
|
|
2019-11-12 11:38:09 +01:00
|
|
|
test_description='test describe'
|
|
|
|
|
|
|
|
# o---o-----o----o----o-------o----x
|
|
|
|
# \ D,R e /
|
|
|
|
# \---o-------------o-'
|
|
|
|
# \ B /
|
|
|
|
# `-o----o----o-'
|
|
|
|
# A c
|
|
|
|
#
|
|
|
|
# First parent of a merge commit is on the same line, second parent below.
|
|
|
|
|
2007-01-14 03:37:32 +01:00
|
|
|
. ./test-lib.sh
|
|
|
|
|
|
|
|
check_describe () {
|
|
|
|
expect="$1"
|
|
|
|
shift
|
2019-11-12 11:38:10 +01:00
|
|
|
describe_opts="$@"
|
|
|
|
test_expect_success "describe $describe_opts" '
|
|
|
|
R=$(git describe $describe_opts 2>err.actual) &&
|
2007-01-14 03:37:32 +01:00
|
|
|
case "$R" in
|
|
|
|
$expect) echo happy ;;
|
2019-11-12 11:38:10 +01:00
|
|
|
*) echo "Oops - $R is not $expect" &&
|
2007-01-14 03:37:32 +01:00
|
|
|
false ;;
|
|
|
|
esac
|
|
|
|
'
|
|
|
|
}
|
|
|
|
|
|
|
|
test_expect_success setup '
|
|
|
|
|
|
|
|
test_tick &&
|
2008-09-03 10:59:29 +02:00
|
|
|
echo one >file && git add file && git commit -m initial &&
|
2007-07-03 07:52:14 +02:00
|
|
|
one=$(git rev-parse HEAD) &&
|
2007-01-14 03:37:32 +01:00
|
|
|
|
2009-10-23 20:42:39 +02:00
|
|
|
git describe --always HEAD &&
|
|
|
|
|
2007-01-14 03:37:32 +01:00
|
|
|
test_tick &&
|
2008-09-03 10:59:29 +02:00
|
|
|
echo two >file && git add file && git commit -m second &&
|
2007-07-03 07:52:14 +02:00
|
|
|
two=$(git rev-parse HEAD) &&
|
2007-01-14 03:37:32 +01:00
|
|
|
|
|
|
|
test_tick &&
|
2008-09-03 10:59:29 +02:00
|
|
|
echo three >file && git add file && git commit -m third &&
|
2007-01-14 03:37:32 +01:00
|
|
|
|
|
|
|
test_tick &&
|
2008-09-03 10:59:29 +02:00
|
|
|
echo A >file && git add file && git commit -m A &&
|
2007-01-14 03:37:32 +01:00
|
|
|
test_tick &&
|
2008-09-03 10:59:29 +02:00
|
|
|
git tag -a -m A A &&
|
2007-01-14 03:37:32 +01:00
|
|
|
|
|
|
|
test_tick &&
|
2008-09-03 10:59:29 +02:00
|
|
|
echo c >file && git add file && git commit -m c &&
|
2007-01-14 03:37:32 +01:00
|
|
|
test_tick &&
|
2008-09-03 10:59:29 +02:00
|
|
|
git tag c &&
|
2007-01-14 03:37:32 +01:00
|
|
|
|
|
|
|
git reset --hard $two &&
|
|
|
|
test_tick &&
|
2008-09-03 10:59:29 +02:00
|
|
|
echo B >side && git add side && git commit -m B &&
|
2007-01-14 03:37:32 +01:00
|
|
|
test_tick &&
|
2008-09-03 10:59:29 +02:00
|
|
|
git tag -a -m B B &&
|
2007-01-14 03:37:32 +01:00
|
|
|
|
|
|
|
test_tick &&
|
2008-09-03 10:59:29 +02:00
|
|
|
git merge -m Merged c &&
|
2007-07-03 07:52:14 +02:00
|
|
|
merged=$(git rev-parse HEAD) &&
|
2007-01-14 03:37:32 +01:00
|
|
|
|
|
|
|
git reset --hard $two &&
|
|
|
|
test_tick &&
|
2008-09-03 10:59:29 +02:00
|
|
|
echo D >another && git add another && git commit -m D &&
|
2007-01-14 03:37:32 +01:00
|
|
|
test_tick &&
|
2008-09-03 10:59:29 +02:00
|
|
|
git tag -a -m D D &&
|
2010-04-13 01:25:29 +02:00
|
|
|
test_tick &&
|
|
|
|
git tag -a -m R R &&
|
2007-01-14 03:37:32 +01:00
|
|
|
|
|
|
|
test_tick &&
|
|
|
|
echo DD >another && git commit -a -m another &&
|
|
|
|
|
|
|
|
test_tick &&
|
2008-09-03 10:59:29 +02:00
|
|
|
git tag e &&
|
2007-01-14 03:37:32 +01:00
|
|
|
|
|
|
|
test_tick &&
|
|
|
|
echo DDD >another && git commit -a -m "yet another" &&
|
|
|
|
|
|
|
|
test_tick &&
|
2008-09-03 10:59:29 +02:00
|
|
|
git merge -m Merged $merged &&
|
2007-01-14 03:37:32 +01:00
|
|
|
|
|
|
|
test_tick &&
|
2007-07-03 07:52:14 +02:00
|
|
|
echo X >file && echo X >side && git add file side &&
|
2008-09-03 10:59:29 +02:00
|
|
|
git commit -m x
|
2007-01-14 03:37:32 +01:00
|
|
|
|
|
|
|
'
|
|
|
|
|
|
|
|
check_describe A-* HEAD
|
|
|
|
check_describe A-* HEAD^
|
2010-04-13 01:25:29 +02:00
|
|
|
check_describe R-* HEAD^^
|
2007-01-14 03:37:32 +01:00
|
|
|
check_describe A-* HEAD^^2
|
|
|
|
check_describe B HEAD^^2^
|
2010-04-13 01:25:29 +02:00
|
|
|
check_describe R-* HEAD^^^
|
2007-01-14 03:37:32 +01:00
|
|
|
|
2008-10-13 16:39:46 +02:00
|
|
|
check_describe c-* --tags HEAD
|
|
|
|
check_describe c-* --tags HEAD^
|
|
|
|
check_describe e-* --tags HEAD^^
|
|
|
|
check_describe c-* --tags HEAD^^2
|
2007-01-14 03:37:32 +01:00
|
|
|
check_describe B --tags HEAD^^2^
|
2009-11-18 14:32:26 +01:00
|
|
|
check_describe e --tags HEAD^^^
|
|
|
|
|
|
|
|
check_describe heads/master --all HEAD
|
|
|
|
check_describe tags/c-* --all HEAD^
|
|
|
|
check_describe tags/e --all HEAD^^^
|
2007-01-14 03:37:32 +01:00
|
|
|
|
2008-02-25 10:43:33 +01:00
|
|
|
check_describe B-0-* --long HEAD^^2^
|
2008-03-04 03:29:51 +01:00
|
|
|
check_describe A-3-* --long HEAD^^2
|
2008-02-25 10:43:33 +01:00
|
|
|
|
2013-05-17 22:56:18 +02:00
|
|
|
check_describe c-7-* --tags
|
|
|
|
check_describe e-3-* --first-parent --tags
|
|
|
|
|
describe --contains: default to HEAD when no commit-ish is given
'git describe --contains' doesn't default to HEAD when no commit is
given, and it doesn't produce any output, not even an error:
~/src/git ((v2.5.0))$ ./git describe --contains
~/src/git ((v2.5.0))$ ./git describe --contains HEAD
v2.5.0^0
Unlike other 'git describe' options, the '--contains' code path is
implemented by calling 'name-rev' with a bunch of options plus all the
commit-ishes that were passed to 'git describe'. If no commit-ish was
present, then 'name-rev' got invoked with none, which then leads to the
behavior illustrated above.
Porcelain commands usually default to HEAD when no commit-ish is given,
and 'git describe' already does so in all other cases, so it should do
so with '--contains' as well.
Pass HEAD to 'name-rev' when no commit-ish is given on the command line
to make '--contains' behave consistently with other 'git describe'
options. While at it, use argv_array_pushv() instead of the loop to
pass commit-ishes to 'git name-rev'.
'git describe's short help already indicates that the commit-ish is
optional, but the synopsis in the man page doesn't, so update it
accordingly as well.
Signed-off-by: SZEDER Gábor <szeder@ira.uka.de>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2015-08-24 18:15:18 +02:00
|
|
|
test_expect_success 'describe --contains defaults to HEAD without commit-ish' '
|
|
|
|
echo "A^0" >expect &&
|
|
|
|
git checkout A &&
|
|
|
|
test_when_finished "git checkout -" &&
|
|
|
|
git describe --contains >actual &&
|
|
|
|
test_cmp expect actual
|
|
|
|
'
|
|
|
|
|
2017-12-11 18:24:54 +01:00
|
|
|
check_describe tags/A --all A^0
|
2008-12-26 23:02:01 +01:00
|
|
|
test_expect_success 'no warning was displayed for A' '
|
tests: use 'test_must_be_empty' instead of 'test_cmp <empty> <out>'
Using 'test_must_be_empty' is shorter and more idiomatic than
>empty &&
test_cmp empty out
as it saves the creation of an empty file. Furthermore, sometimes the
expected empty file doesn't have such a descriptive name like 'empty',
and its creation is far away from the place where it's finally used
for comparison (e.g. in 't7600-merge.sh', where two expected empty
files are created in the 'setup' test, but are used only about 500
lines later).
These cases were found by instrumenting 'test_cmp' to error out the
test script when it's used to compare empty files, and then converted
manually.
Note that even after this patch there still remain a lot of cases
where we use 'test_cmp' to check empty files:
- Sometimes the expected output is not hard-coded in the test, but
'test_cmp' is used to ensure that two similar git commands produce
the same output, and that output happens to be empty, e.g. the
test 'submodule update --merge - ignores --merge for new
submodules' in 't7406-submodule-update.sh'.
- Repetitive common tasks, including preparing the expected results
and running 'test_cmp', are often extracted into a helper
function, and some of this helper's callsites expect no output.
- For the same reason as above, the whole 'test_expect_success'
block is within a helper function, e.g. in 't3070-wildmatch.sh'.
- Or 'test_cmp' is invoked in a loop, e.g. the test 'cvs update
(-p)' in 't9400-git-cvsserver-server.sh'.
Signed-off-by: SZEDER Gábor <szeder.dev@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2018-08-19 23:57:25 +02:00
|
|
|
test_must_be_empty err.actual
|
2008-12-26 23:02:01 +01:00
|
|
|
'
|
|
|
|
|
2008-03-04 02:09:38 +01:00
|
|
|
test_expect_success 'rename tag A to Q locally' '
|
|
|
|
mv .git/refs/tags/A .git/refs/tags/Q
|
|
|
|
'
|
|
|
|
cat - >err.expect <<EOF
|
|
|
|
warning: tag 'A' is really 'Q' here
|
|
|
|
EOF
|
|
|
|
check_describe A-* HEAD
|
2011-04-13 01:33:39 +02:00
|
|
|
test_expect_success 'warning was displayed for Q' '
|
|
|
|
test_i18ncmp err.expect err.actual
|
2008-03-04 02:09:38 +01:00
|
|
|
'
|
|
|
|
test_expect_success 'rename tag Q back to A' '
|
|
|
|
mv .git/refs/tags/Q .git/refs/tags/A
|
|
|
|
'
|
|
|
|
|
2008-03-04 02:09:35 +01:00
|
|
|
test_expect_success 'pack tag refs' 'git pack-refs'
|
|
|
|
check_describe A-* HEAD
|
|
|
|
|
2019-02-03 07:00:25 +01:00
|
|
|
test_expect_success 'describe works from outside repo using --git-dir' '
|
|
|
|
git clone --bare "$TRASH_DIRECTORY" "$TRASH_DIRECTORY/bare" &&
|
|
|
|
git --git-dir "$TRASH_DIRECTORY/bare" describe >out &&
|
2019-02-21 20:28:48 +01:00
|
|
|
grep -E "^A-[1-9][0-9]?-g[0-9a-f]+$" out
|
2019-02-03 07:00:25 +01:00
|
|
|
'
|
|
|
|
|
2009-10-21 15:35:22 +02:00
|
|
|
check_describe "A-*[0-9a-f]" --dirty
|
|
|
|
|
2019-02-03 07:00:24 +01:00
|
|
|
test_expect_success 'describe --dirty with --work-tree' '
|
|
|
|
(
|
|
|
|
cd "$TEST_DIRECTORY" &&
|
|
|
|
git --git-dir "$TRASH_DIRECTORY/.git" --work-tree "$TRASH_DIRECTORY" describe --dirty >"$TRASH_DIRECTORY/out"
|
|
|
|
) &&
|
2019-02-21 20:28:48 +01:00
|
|
|
grep -E "^A-[1-9][0-9]?-g[0-9a-f]+$" out
|
2019-02-03 07:00:24 +01:00
|
|
|
'
|
|
|
|
|
2009-10-21 15:35:22 +02:00
|
|
|
test_expect_success 'set-up dirty work tree' '
|
|
|
|
echo >>file
|
|
|
|
'
|
|
|
|
|
|
|
|
check_describe "A-*[0-9a-f]-dirty" --dirty
|
|
|
|
|
2019-02-03 07:00:24 +01:00
|
|
|
test_expect_success 'describe --dirty with --work-tree (dirty)' '
|
|
|
|
(
|
|
|
|
cd "$TEST_DIRECTORY" &&
|
|
|
|
git --git-dir "$TRASH_DIRECTORY/.git" --work-tree "$TRASH_DIRECTORY" describe --dirty >"$TRASH_DIRECTORY/out"
|
|
|
|
) &&
|
2019-02-21 20:28:48 +01:00
|
|
|
grep -E "^A-[1-9][0-9]?-g[0-9a-f]+-dirty$" out
|
2019-02-03 07:00:24 +01:00
|
|
|
'
|
|
|
|
|
2009-10-21 15:35:22 +02:00
|
|
|
check_describe "A-*[0-9a-f].mod" --dirty=.mod
|
|
|
|
|
2019-02-03 07:00:24 +01:00
|
|
|
test_expect_success 'describe --dirty=.mod with --work-tree (dirty)' '
|
|
|
|
(
|
|
|
|
cd "$TEST_DIRECTORY" &&
|
|
|
|
git --git-dir "$TRASH_DIRECTORY/.git" --work-tree "$TRASH_DIRECTORY" describe --dirty=.mod >"$TRASH_DIRECTORY/out"
|
|
|
|
) &&
|
2019-02-21 20:28:48 +01:00
|
|
|
grep -E "^A-[1-9][0-9]?-g[0-9a-f]+.mod$" out
|
2019-02-03 07:00:24 +01:00
|
|
|
'
|
|
|
|
|
2009-10-21 15:35:22 +02:00
|
|
|
test_expect_success 'describe --dirty HEAD' '
|
|
|
|
test_must_fail git describe --dirty HEAD
|
|
|
|
'
|
|
|
|
|
2008-06-04 21:06:31 +02:00
|
|
|
test_expect_success 'set-up matching pattern tests' '
|
|
|
|
git tag -a -m test-annotated test-annotated &&
|
|
|
|
echo >>file &&
|
|
|
|
test_tick &&
|
|
|
|
git commit -a -m "one more" &&
|
|
|
|
git tag test1-lightweight &&
|
|
|
|
echo >>file &&
|
|
|
|
test_tick &&
|
|
|
|
git commit -a -m "yet another" &&
|
|
|
|
git tag test2-lightweight &&
|
|
|
|
echo >>file &&
|
|
|
|
test_tick &&
|
|
|
|
git commit -a -m "even more"
|
|
|
|
|
|
|
|
'
|
|
|
|
|
|
|
|
check_describe "test-annotated-*" --match="test-*"
|
|
|
|
|
|
|
|
check_describe "test1-lightweight-*" --tags --match="test1-*"
|
|
|
|
|
|
|
|
check_describe "test2-lightweight-*" --tags --match="test2-*"
|
|
|
|
|
2008-07-03 04:32:45 +02:00
|
|
|
check_describe "test2-lightweight-*" --long --tags --match="test2-*" HEAD^
|
|
|
|
|
2017-09-16 07:53:44 +02:00
|
|
|
check_describe "test2-lightweight-*" --long --tags --match="test1-*" --match="test2-*" HEAD^
|
2017-01-19 00:06:07 +01:00
|
|
|
|
|
|
|
check_describe "test2-lightweight-*" --long --tags --match="test1-*" --no-match --match="test2-*" HEAD^
|
|
|
|
|
2017-09-16 07:53:44 +02:00
|
|
|
check_describe "test1-lightweight-*" --long --tags --match="test1-*" --match="test3-*" HEAD
|
|
|
|
|
|
|
|
check_describe "test1-lightweight-*" --long --tags --match="test3-*" --match="test1-*" HEAD
|
|
|
|
|
2017-09-20 03:10:10 +02:00
|
|
|
test_expect_success 'set-up branches' '
|
|
|
|
git branch branch_A A &&
|
|
|
|
git branch branch_C c &&
|
|
|
|
git update-ref refs/remotes/origin/remote_branch_A "A^{commit}" &&
|
|
|
|
git update-ref refs/remotes/origin/remote_branch_C "c^{commit}" &&
|
|
|
|
git update-ref refs/original/original_branch_A test-annotated~2
|
|
|
|
'
|
|
|
|
|
|
|
|
check_describe "heads/branch_A*" --all --match="branch_*" --exclude="branch_C" HEAD
|
|
|
|
|
|
|
|
check_describe "remotes/origin/remote_branch_A*" --all --match="origin/remote_branch_*" --exclude="origin/remote_branch_C" HEAD
|
|
|
|
|
|
|
|
check_describe "original/original_branch_A*" --all test-annotated~1
|
|
|
|
|
|
|
|
test_expect_success '--match does not work for other types' '
|
|
|
|
test_must_fail git describe --all --match="*original_branch_*" test-annotated~1
|
|
|
|
'
|
|
|
|
|
|
|
|
test_expect_success '--exclude does not work for other types' '
|
|
|
|
R=$(git describe --all --exclude="any_pattern_even_not_matching" test-annotated~1) &&
|
|
|
|
case "$R" in
|
|
|
|
*original_branch_A*) echo "fail: Found unknown reference $R with --exclude"
|
|
|
|
false;;
|
|
|
|
*) echo ok: Found some known type;;
|
|
|
|
esac
|
|
|
|
'
|
|
|
|
|
2013-07-18 23:11:35 +02:00
|
|
|
test_expect_success 'name-rev with exact tags' '
|
|
|
|
echo A >expect &&
|
|
|
|
tag_object=$(git rev-parse refs/tags/A) &&
|
|
|
|
git name-rev --tags --name-only $tag_object >actual &&
|
|
|
|
test_cmp expect actual &&
|
|
|
|
|
|
|
|
echo "A^0" >expect &&
|
|
|
|
tagged_commit=$(git rev-parse "refs/tags/A^0") &&
|
|
|
|
git name-rev --tags --name-only $tagged_commit >actual &&
|
|
|
|
test_cmp expect actual
|
|
|
|
'
|
|
|
|
|
2017-09-07 16:02:21 +02:00
|
|
|
test_expect_success 'name-rev --all' '
|
|
|
|
>expect.unsorted &&
|
|
|
|
for rev in $(git rev-list --all)
|
|
|
|
do
|
|
|
|
git name-rev $rev >>expect.unsorted
|
|
|
|
done &&
|
|
|
|
sort <expect.unsorted >expect &&
|
|
|
|
git name-rev --all >actual.unsorted &&
|
|
|
|
sort <actual.unsorted >actual &&
|
|
|
|
test_cmp expect actual
|
|
|
|
'
|
|
|
|
|
|
|
|
test_expect_success 'name-rev --stdin' '
|
|
|
|
>expect.unsorted &&
|
|
|
|
for rev in $(git rev-list --all)
|
|
|
|
do
|
|
|
|
name=$(git name-rev --name-only $rev) &&
|
|
|
|
echo "$rev ($name)" >>expect.unsorted
|
|
|
|
done &&
|
|
|
|
sort <expect.unsorted >expect &&
|
|
|
|
git rev-list --all | git name-rev --stdin >actual.unsorted &&
|
|
|
|
sort <actual.unsorted >actual &&
|
|
|
|
test_cmp expect actual
|
|
|
|
'
|
|
|
|
|
2013-07-18 23:46:51 +02:00
|
|
|
test_expect_success 'describe --contains with the exact tags' '
|
|
|
|
echo "A^0" >expect &&
|
|
|
|
tag_object=$(git rev-parse refs/tags/A) &&
|
|
|
|
git describe --contains $tag_object >actual &&
|
|
|
|
test_cmp expect actual &&
|
|
|
|
|
|
|
|
echo "A^0" >expect &&
|
|
|
|
tagged_commit=$(git rev-parse "refs/tags/A^0") &&
|
|
|
|
git describe --contains $tagged_commit >actual &&
|
|
|
|
test_cmp expect actual
|
|
|
|
'
|
|
|
|
|
2017-01-19 00:06:07 +01:00
|
|
|
test_expect_success 'describe --contains and --match' '
|
|
|
|
echo "A^0" >expect &&
|
|
|
|
tagged_commit=$(git rev-parse "refs/tags/A^0") &&
|
|
|
|
test_must_fail git describe --contains --match="B" $tagged_commit &&
|
|
|
|
git describe --contains --match="B" --match="A" $tagged_commit >actual &&
|
|
|
|
test_cmp expect actual
|
|
|
|
'
|
|
|
|
|
2017-01-19 00:06:08 +01:00
|
|
|
test_expect_success 'describe --exclude' '
|
|
|
|
echo "c~1" >expect &&
|
|
|
|
tagged_commit=$(git rev-parse "refs/tags/A^0") &&
|
|
|
|
test_must_fail git describe --contains --match="B" $tagged_commit &&
|
|
|
|
git describe --contains --match="?" --exclude="A" $tagged_commit >actual &&
|
|
|
|
test_cmp expect actual
|
|
|
|
'
|
|
|
|
|
2017-01-19 00:06:07 +01:00
|
|
|
test_expect_success 'describe --contains and --no-match' '
|
|
|
|
echo "A^0" >expect &&
|
|
|
|
tagged_commit=$(git rev-parse "refs/tags/A^0") &&
|
|
|
|
git describe --contains --match="B" --no-match $tagged_commit >actual &&
|
|
|
|
test_cmp expect actual
|
|
|
|
'
|
|
|
|
|
2017-03-21 23:57:18 +01:00
|
|
|
test_expect_success 'setup and absorb a submodule' '
|
|
|
|
test_create_repo sub1 &&
|
|
|
|
test_commit -C sub1 initial &&
|
|
|
|
git submodule add ./sub1 &&
|
|
|
|
git submodule absorbgitdirs &&
|
|
|
|
git commit -a -m "add submodule" &&
|
|
|
|
git describe --dirty >expect &&
|
|
|
|
git describe --broken >out &&
|
|
|
|
test_cmp expect out
|
|
|
|
'
|
|
|
|
|
2017-06-25 12:20:41 +02:00
|
|
|
test_expect_success 'describe chokes on severely broken submodules' '
|
2017-03-21 23:57:18 +01:00
|
|
|
mv .git/modules/sub1/ .git/modules/sub_moved &&
|
|
|
|
test_must_fail git describe --dirty
|
|
|
|
'
|
2019-02-03 07:00:24 +01:00
|
|
|
|
2017-11-02 20:41:42 +01:00
|
|
|
test_expect_success 'describe ignoring a broken submodule' '
|
2017-03-21 23:57:18 +01:00
|
|
|
git describe --broken >out &&
|
2019-02-03 07:00:24 +01:00
|
|
|
grep broken out
|
|
|
|
'
|
|
|
|
|
|
|
|
test_expect_success 'describe with --work-tree ignoring a broken submodule' '
|
|
|
|
(
|
|
|
|
cd "$TEST_DIRECTORY" &&
|
|
|
|
git --git-dir "$TRASH_DIRECTORY/.git" --work-tree "$TRASH_DIRECTORY" describe --broken >"$TRASH_DIRECTORY/out"
|
|
|
|
) &&
|
2017-09-07 16:02:22 +02:00
|
|
|
test_when_finished "mv .git/modules/sub_moved .git/modules/sub1" &&
|
2017-03-21 23:57:18 +01:00
|
|
|
grep broken out
|
|
|
|
'
|
|
|
|
|
builtin/describe.c: describe a blob
Sometimes users are given a hash of an object and they want to
identify it further (ex.: Use verify-pack to find the largest blobs,
but what are these? or [1])
When describing commits, we try to anchor them to tags or refs, as these
are conceptually on a higher level than the commit. And if there is no ref
or tag that matches exactly, we're out of luck. So we employ a heuristic
to make up a name for the commit. These names are ambiguous, there might
be different tags or refs to anchor to, and there might be different
path in the DAG to travel to arrive at the commit precisely.
When describing a blob, we want to describe the blob from a higher layer
as well, which is a tuple of (commit, deep/path) as the tree objects
involved are rather uninteresting. The same blob can be referenced by
multiple commits, so how we decide which commit to use? This patch
implements a rather naive approach on this: As there are no back pointers
from blobs to commits in which the blob occurs, we'll start walking from
any tips available, listing the blobs in-order of the commit and once we
found the blob, we'll take the first commit that listed the blob. For
example
git describe --tags v0.99:Makefile
conversion-901-g7672db20c2:Makefile
tells us the Makefile as it was in v0.99 was introduced in commit 7672db20.
The walking is performed in reverse order to show the introduction of a
blob rather than its last occurrence.
[1] https://stackoverflow.com/questions/223678/which-commit-has-this-blob
Signed-off-by: Stefan Beller <sbeller@google.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2017-11-16 03:00:39 +01:00
|
|
|
test_expect_success 'describe a blob at a directly tagged commit' '
|
|
|
|
echo "make it a unique blob" >file &&
|
|
|
|
git add file && git commit -m "content in file" &&
|
|
|
|
git tag -a -m "latest annotated tag" unique-file &&
|
|
|
|
git describe HEAD:file >actual &&
|
|
|
|
echo "unique-file:file" >expect &&
|
|
|
|
test_cmp expect actual
|
|
|
|
'
|
|
|
|
|
|
|
|
test_expect_success 'describe a blob with its first introduction' '
|
|
|
|
git commit --allow-empty -m "empty commit" &&
|
|
|
|
git rm file &&
|
|
|
|
git commit -m "delete blob" &&
|
|
|
|
git revert HEAD &&
|
|
|
|
git commit --allow-empty -m "empty commit" &&
|
|
|
|
git describe HEAD:file >actual &&
|
|
|
|
echo "unique-file:file" >expect &&
|
|
|
|
test_cmp expect actual
|
|
|
|
'
|
|
|
|
|
|
|
|
test_expect_success 'describe directly tagged blob' '
|
|
|
|
git tag test-blob unique-file:file &&
|
|
|
|
git describe test-blob >actual &&
|
|
|
|
echo "unique-file:file" >expect &&
|
|
|
|
# suboptimal: we rather want to see "test-blob"
|
|
|
|
test_cmp expect actual
|
|
|
|
'
|
|
|
|
|
|
|
|
test_expect_success 'describe tag object' '
|
|
|
|
git tag test-blob-1 -a -m msg unique-file:file &&
|
|
|
|
test_must_fail git describe test-blob-1 2>actual &&
|
|
|
|
test_i18ngrep "fatal: test-blob-1 is neither a commit nor blob" actual
|
|
|
|
'
|
|
|
|
|
name-rev: eliminate recursion in name_rev()
The name_rev() function calls itself recursively for each interesting
parent of the commit it got as parameter, and, consequently, it can
segfault when processing a deep history if it exhausts the available
stack space. E.g. running 'git name-rev --all' and 'git name-rev
HEAD~100000' in the gcc, gecko-dev, llvm, and WebKit repositories
results in segfaults on my machine ('ulimit -s' reports 8192kB of
stack size limit), and nowadays the former segfaults in the Linux repo
as well (it reached the necessasry depth sometime between v5.3-rc4 and
-rc5).
Eliminate the recursion by inserting the interesting parents into a
LIFO 'prio_queue' [1] and iterating until the queue becomes empty.
Note that the parent commits must be added in reverse order to the
LIFO 'prio_queue', so their relative order is preserved during
processing, i.e. the first parent should come out first from the
queue, because otherwise performance greatly suffers on mergy
histories [2].
The stacksize-limited test 'name-rev works in a deep repo' in
't6120-describe.sh' demonstrated this issue and expected failure. Now
the recursion is gone, so flip it to expect success. Also gone are
the dmesg entries logging the segfault of that segfaulting 'git
name-rev' process on every execution of the test suite.
Note that this slightly changes the order of lines in the output of
'git name-rev --all', usually swapping two lines every 35 lines in
git.git or every 150 lines in linux.git. This shouldn't matter in
practice, because the output has always been unordered anyway.
This patch is best viewed with '--ignore-all-space'.
[1] Early versions of this patch used a 'commit_list', resulting in
~15% performance penalty for 'git name-rev --all' in 'linux.git',
presumably because of the memory allocation and release for each
insertion and removal. Using a LIFO 'prio_queue' has basically no
effect on performance.
[2] We prefer shorter names, i.e. 'v0.1~234' is preferred over
'v0.1^2~5', meaning that usually following the first parent of a
merge results in the best name for its ancestors. So when later
we follow the remaining parent(s) of a merge, and reach an already
named commit, then we usually find that we can't give that commit
a better name, and thus we don't have to visit any of its
ancestors again.
OTOH, if we were to follow the Nth parent of the merge first, then
the name of all its ancestors would include a corresponding '^N'.
Those are not the best names for those commits, so when later we
reach an already named commit following the first parent of that
merge, then we would have to update the name of that commit and
the names of all of its ancestors as well. Consequently, we would
have to visit many commits several times, resulting in a
significant slowdown.
Signed-off-by: SZEDER Gábor <szeder.dev@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2019-12-09 12:52:57 +01:00
|
|
|
test_expect_success ULIMIT_STACK_SIZE 'name-rev works in a deep repo' '
|
2017-09-07 16:02:23 +02:00
|
|
|
i=1 &&
|
|
|
|
while test $i -lt 8000
|
|
|
|
do
|
|
|
|
echo "commit refs/heads/master
|
|
|
|
committer A U Thor <author@example.com> $((1000000000 + $i * 100)) +0200
|
|
|
|
data <<EOF
|
|
|
|
commit #$i
|
|
|
|
EOF"
|
|
|
|
test $i = 1 && echo "from refs/heads/master^0"
|
|
|
|
i=$(($i + 1))
|
|
|
|
done | git fast-import &&
|
|
|
|
git checkout master &&
|
|
|
|
git tag far-far-away HEAD^ &&
|
|
|
|
echo "HEAD~4000 tags/far-far-away~3999" >expect &&
|
|
|
|
git name-rev HEAD~4000 >actual &&
|
|
|
|
test_cmp expect actual &&
|
|
|
|
run_with_limited_stack git name-rev HEAD~4000 >actual &&
|
|
|
|
test_cmp expect actual
|
|
|
|
'
|
|
|
|
|
|
|
|
test_expect_success ULIMIT_STACK_SIZE 'describe works in a deep repo' '
|
|
|
|
git tag -f far-far-away HEAD~7999 &&
|
|
|
|
echo "far-far-away" >expect &&
|
|
|
|
git describe --tags --abbrev=0 HEAD~4000 >actual &&
|
|
|
|
test_cmp expect actual &&
|
|
|
|
run_with_limited_stack git describe --tags --abbrev=0 HEAD~4000 >actual &&
|
|
|
|
test_cmp expect actual
|
|
|
|
'
|
|
|
|
|
2017-12-11 18:24:54 +01:00
|
|
|
check_describe tags/A --all A
|
|
|
|
check_describe tags/c --all c
|
|
|
|
check_describe heads/branch_A --all --match='branch_*' branch_A
|
|
|
|
|
describe: confirm that blobs actually exist
Prior to 644eb60bd0 (builtin/describe.c: describe a blob,
2017-11-15), we noticed and complained about missing
objects, since they were not valid commits:
$ git describe 0000000000000000000000000000000000000000
fatal: 0000000000000000000000000000000000000000 is not a valid 'commit' object
After that commit, we feed any non-commit to lookup_blob(),
and complain only if it returns NULL. But the lookup_*
functions do not actually look at the on-disk object
database at all. They return an entry from the in-memory
object hash if present (and if it matches the requested
type), and otherwise auto-create a "struct object" of the
requested type.
A missing object would hit that latter case: we create a
bogus blob struct, walk all of history looking for it, and
then exit successfully having produced no output.
One reason nobody may have noticed this is that some related
cases do still work OK:
1. If we ask for a tree by sha1, then the call to
lookup_commit_referecne_gently() would have parsed it,
and we would have its true type in the in-memory object
hash.
2. If we ask for a name that doesn't exist but isn't a
40-hex sha1, then get_oid() would complain before we
even look at the objects at all.
We can fix this by replacing the lookup_blob() call with a
check of the true type via sha1_object_info(). This is not
quite as efficient as we could possibly make this check. We
know in most cases that the object was already parsed in the
earlier commit lookup, so we could call lookup_object(),
which does auto-create, and check the resulting struct's
type (or NULL). However it's not worth the fragility nor
code complexity to save a single object lookup.
The new tests cover this case, as well as that of a
tree-by-sha1 (which does work as described above, but was
not explicitly tested).
Noticed-by: Michael Haggerty <mhagger@alum.mit.edu>
Signed-off-by: Jeff King <peff@peff.net>
Acked-by: Stefan Beller <sbeller@google.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2018-02-12 18:23:06 +01:00
|
|
|
test_expect_success 'describe complains about tree object' '
|
|
|
|
test_must_fail git describe HEAD^{tree}
|
|
|
|
'
|
|
|
|
|
|
|
|
test_expect_success 'describe complains about missing object' '
|
2018-05-13 04:24:13 +02:00
|
|
|
test_must_fail git describe $ZERO_OID
|
describe: confirm that blobs actually exist
Prior to 644eb60bd0 (builtin/describe.c: describe a blob,
2017-11-15), we noticed and complained about missing
objects, since they were not valid commits:
$ git describe 0000000000000000000000000000000000000000
fatal: 0000000000000000000000000000000000000000 is not a valid 'commit' object
After that commit, we feed any non-commit to lookup_blob(),
and complain only if it returns NULL. But the lookup_*
functions do not actually look at the on-disk object
database at all. They return an entry from the in-memory
object hash if present (and if it matches the requested
type), and otherwise auto-create a "struct object" of the
requested type.
A missing object would hit that latter case: we create a
bogus blob struct, walk all of history looking for it, and
then exit successfully having produced no output.
One reason nobody may have noticed this is that some related
cases do still work OK:
1. If we ask for a tree by sha1, then the call to
lookup_commit_referecne_gently() would have parsed it,
and we would have its true type in the in-memory object
hash.
2. If we ask for a name that doesn't exist but isn't a
40-hex sha1, then get_oid() would complain before we
even look at the objects at all.
We can fix this by replacing the lookup_blob() call with a
check of the true type via sha1_object_info(). This is not
quite as efficient as we could possibly make this check. We
know in most cases that the object was already parsed in the
earlier commit lookup, so we could call lookup_object(),
which does auto-create, and check the resulting struct's
type (or NULL). However it's not worth the fragility nor
code complexity to save a single object lookup.
The new tests cover this case, as well as that of a
tree-by-sha1 (which does work as described above, but was
not explicitly tested).
Noticed-by: Michael Haggerty <mhagger@alum.mit.edu>
Signed-off-by: Jeff King <peff@peff.net>
Acked-by: Stefan Beller <sbeller@google.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2018-02-12 18:23:06 +01:00
|
|
|
'
|
|
|
|
|
name-rev: avoid cutoff timestamp underflow
When 'git name-rev' is invoked with commit-ish parameters, it tries to
save some work, and doesn't visit commits older than the committer
date of the oldest given commit minus a one day worth of slop. Since
our 'timestamp_t' is an unsigned type, this leads to a timestamp
underflow when the committer date of the oldest given commit is within
a day of the UNIX epoch. As a result the cutoff timestamp ends up
far-far in the future, and 'git name-rev' doesn't visit any commits,
and names each given commit as 'undefined'.
Check whether subtracting the slop from the oldest committer date
would lead to an underflow, and use no cutoff in that case. We don't
have a TIME_MIN constant, dddbad728c (timestamp_t: a new data type for
timestamps, 2017-04-26) didn't add one, so do it now.
Note that the type of the cutoff timestamp variable used to be signed
before 5589e87fd8 (name-rev: change a "long" variable to timestamp_t,
2017-05-20). The behavior was still the same even back then, but the
underflow didn't happen when substracting the slop from the oldest
committer date, but when comparing the signed cutoff timestamp with
unsigned committer dates in name_rev(). IOW, this underflow bug is as
old as 'git name-rev' itself.
Helped-by: Johannes Sixt <j6t@kdbg.org>
Signed-off-by: SZEDER Gábor <szeder.dev@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2019-09-24 09:32:13 +02:00
|
|
|
test_expect_success 'name-rev a rev shortly after epoch' '
|
|
|
|
test_when_finished "git checkout master" &&
|
|
|
|
|
|
|
|
git checkout --orphan no-timestamp-underflow &&
|
|
|
|
# Any date closer to epoch than the CUTOFF_DATE_SLOP constant
|
|
|
|
# in builtin/name-rev.c.
|
|
|
|
GIT_COMMITTER_DATE="@1234 +0000" \
|
|
|
|
git commit -m "committer date shortly after epoch" &&
|
|
|
|
old_commit_oid=$(git rev-parse HEAD) &&
|
|
|
|
|
|
|
|
echo "$old_commit_oid no-timestamp-underflow" >expect &&
|
|
|
|
git name-rev $old_commit_oid >actual &&
|
|
|
|
test_cmp expect actual
|
|
|
|
'
|
|
|
|
|
t6120: add a test to cover inner conditions in 'git name-rev's name_rev()
In 'builtin/name-rev.c' in the name_rev() function there is a loop
iterating over all parents of the given commit, and the loop body
looks like this:
if (parent_number > 1) {
if (generation > 0)
// branch #1
new_name = ...
else
// branch #2
new_name = ...
name_rev(parent, new_name, ...);
} else {
// branch #3
name_rev(...);
}
These conditions are not covered properly in the test suite. As far
as purely test coverage goes, they are all executed several times over
in 't6120-describe.sh'. However, they don't directly influence the
command's output, because the repository used in that test script
contains several branches and tags pointing somewhere into the middle
of the commit DAG, and thus result in a better name for the
to-be-named commit. This can hide bugs: e.g. by replacing the
'new_name' parameter of the first recursive name_rev() call with
'tip_name' (effectively making both branch #1 and #2 a noop) 'git
name-rev --all' shows thousands of bogus names in the Git repository,
but the whole test suite still passes successfully. In an early
version of a later patch in this series I managed to mess up all three
branches (at once!), but the test suite still passed.
So add a new test case that operates on the following history:
A--------------master
\ /
\----------M2
\ /
\---M1-C
\ /
B
and names the commit 'B' to make sure that all three branches are
crucial to determine 'B's name:
- There is only a single ref, so all names are based on 'master',
without any undesired interference from other refs.
- Each time name_rev() follows the second parent of a merge commit,
it appends "^2" to the name. Following 'master's second parent
right at the start ensures that all commits on the ancestry path
from 'master' to 'B' have a different base name from the original
'tip_name' of the very first name_rev() invocation. Currently,
while name_rev() is recursive, it doesn't matter, but it will be
necessary to properly cover all three branches after the recursion
is eliminated later in this series.
- Following 'M2's second parent makes sure that branch #2 (i.e. when
'generation = 0') affects 'B's name.
- Following the only parent of the non-merge commit 'C' ensures that
branch #3 affects 'B's name, and that it increments 'generation'.
- Coming from 'C' 'generation' is 1, thus following 'M1's second
parent makes sure that branch #1 affects 'B's name.
Signed-off-by: SZEDER Gábor <szeder.dev@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2019-11-12 11:38:14 +01:00
|
|
|
# A--------------master
|
|
|
|
# \ /
|
|
|
|
# \----------M2
|
|
|
|
# \ /
|
|
|
|
# \---M1-C
|
|
|
|
# \ /
|
|
|
|
# B
|
|
|
|
test_expect_success 'name-rev covers all conditions while looking at parents' '
|
|
|
|
git init repo &&
|
|
|
|
(
|
|
|
|
cd repo &&
|
|
|
|
|
|
|
|
echo A >file &&
|
|
|
|
git add file &&
|
|
|
|
git commit -m A &&
|
|
|
|
A=$(git rev-parse HEAD) &&
|
|
|
|
|
|
|
|
git checkout --detach &&
|
|
|
|
echo B >file &&
|
|
|
|
git commit -m B file &&
|
|
|
|
B=$(git rev-parse HEAD) &&
|
|
|
|
|
|
|
|
git checkout $A &&
|
|
|
|
git merge --no-ff $B && # M1
|
|
|
|
|
|
|
|
echo C >file &&
|
|
|
|
git commit -m C file &&
|
|
|
|
|
|
|
|
git checkout $A &&
|
|
|
|
git merge --no-ff HEAD@{1} && # M2
|
|
|
|
|
|
|
|
git checkout master &&
|
|
|
|
git merge --no-ff HEAD@{1} &&
|
|
|
|
|
|
|
|
echo "$B master^2^2~1^2" >expect &&
|
|
|
|
git name-rev $B >actual &&
|
|
|
|
|
|
|
|
test_cmp expect actual
|
|
|
|
)
|
|
|
|
'
|
|
|
|
|
2020-02-26 18:48:53 +01:00
|
|
|
# B
|
|
|
|
# o
|
|
|
|
# \
|
|
|
|
# o-----o---o----x
|
|
|
|
# A
|
|
|
|
#
|
|
|
|
test_expect_success 'describe commits with disjoint bases' '
|
|
|
|
git init disjoint1 &&
|
|
|
|
(
|
|
|
|
cd disjoint1 &&
|
|
|
|
|
|
|
|
echo o >> file && git add file && git commit -m o &&
|
|
|
|
echo A >> file && git add file && git commit -m A &&
|
|
|
|
git tag A -a -m A &&
|
|
|
|
echo o >> file && git add file && git commit -m o &&
|
|
|
|
|
|
|
|
git checkout --orphan branch && rm file &&
|
|
|
|
echo B > file2 && git add file2 && git commit -m B &&
|
|
|
|
git tag B -a -m B &&
|
|
|
|
git merge --no-ff --allow-unrelated-histories master -m x &&
|
|
|
|
|
|
|
|
check_describe "A-3-*" HEAD
|
|
|
|
)
|
|
|
|
'
|
|
|
|
|
|
|
|
# B
|
|
|
|
# o---o---o------------.
|
|
|
|
# \
|
|
|
|
# o---o---x
|
|
|
|
# A
|
|
|
|
#
|
|
|
|
test_expect_success 'describe commits with disjoint bases 2' '
|
|
|
|
git init disjoint2 &&
|
|
|
|
(
|
|
|
|
cd disjoint2 &&
|
|
|
|
|
|
|
|
echo A >> file && git add file && GIT_COMMITTER_DATE="2020-01-01 18:00" git commit -m A &&
|
|
|
|
git tag A -a -m A &&
|
|
|
|
echo o >> file && git add file && GIT_COMMITTER_DATE="2020-01-01 18:01" git commit -m o &&
|
|
|
|
|
|
|
|
git checkout --orphan branch &&
|
|
|
|
echo o >> file2 && git add file2 && GIT_COMMITTER_DATE="2020-01-01 15:00" git commit -m o &&
|
|
|
|
echo o >> file2 && git add file2 && GIT_COMMITTER_DATE="2020-01-01 15:01" git commit -m o &&
|
|
|
|
echo B >> file2 && git add file2 && GIT_COMMITTER_DATE="2020-01-01 15:02" git commit -m B &&
|
|
|
|
git tag B -a -m B &&
|
|
|
|
git merge --no-ff --allow-unrelated-histories master -m x &&
|
|
|
|
|
|
|
|
check_describe "B-3-*" HEAD
|
|
|
|
)
|
|
|
|
'
|
|
|
|
|
2007-01-14 03:37:32 +01:00
|
|
|
test_done
|