|
|
|
# Performance testing framework. Each perf script starts much like
|
|
|
|
# a normal test script, except it sources this library instead of
|
|
|
|
# test-lib.sh. See t/perf/README for documentation.
|
|
|
|
#
|
|
|
|
# Copyright (c) 2011 Thomas Rast
|
|
|
|
#
|
|
|
|
# This program is free software: you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation, either version 2 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# This program is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with this program. If not, see http://www.gnu.org/licenses/ .
|
|
|
|
|
perf-lib.sh: rely on test-lib.sh for --tee handling
Since its inception, the perf-lib.sh script has manually handled the
"--tee" option (and other options which imply it, like "--valgrind")
with a cut-and-pasted block from test-lib.sh. That block has grown stale
over the years, and has at least three problems:
1. It uses $SHELL to re-exec the script, whereas the version in
test-lib.sh learned to use $TEST_SHELL_PATH.
2. It does an ad-hoc search of the "$*" string, whereas test-lib.sh
learned to carefully parse the arguments left to right.
3. It never learned about --verbose-log (which also implies --tee),
so it would not trigger for that option.
This last one was especially annoying, because t/perf/run uses the
GIT_TEST_OPTS from your config.mak to run the perf scripts. So if you've
set, say, "-x --verbose-log" there, it will be passed as part of most
perf runs. And while this script doesn't recognize the option, the
test-lib.sh that we source _does_, and the behavior ends up being much
more annoying:
- as the comment at the top of the block says, we have to run this
tee code early, before we start munging variables (it says
GIT_BUILD_DIR, but the problematic variable is actually
GIT_TEST_INSTALLED).
- since we don't recognize --verbose-log, we don't trigger the block.
We go on to munge GIT_TEST_INSTALLED, converting it from a relative
to an absolute path.
- then we source test-lib.sh, which _does_ recognize --verbose-log. It
re-execs the script, which runs again. But this time with an
absolute version of GIT_TEST_INSTALLED.
- As a result, we copy the absolute version of GIT_TEST_INSTALLED into
perf_results_prefix. Instead of writing our results to the expected
"test-results/build_1234abcd.p1234-whatever.times", we instead write
them to "test-results/_full_path_to_repo_t_perf_build_1234...".
The aggregate.perl script doesn't expect this, and so it prints
"<missing>" for each result (even though it spent considerable time
running the tests!).
We can solve all of these in one blow by just deleting our custom
handling, and relying on the inclusion of test-lib.sh to handle --tee,
--verbose-log, etc.
There's one catch, though. We want to handle GIT_TEST_INSTALLED after
we've included test-lib.sh, since we want it un-munged in the re-exec'd
version of the script. But if we want to convert it from a relative
to an absolute path, we must do so before we load test-lib.sh, since it
will change our working directory. So we compute the absolute directory
first, store it away, then include test-lib.sh, and finally assign to
GIT_TEST_INSTALLED as appropriate.
Signed-off-by: Jeff King <peff@peff.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
6 years ago
|
|
|
# These variables must be set before the inclusion of test-lib.sh below,
|
|
|
|
# because it will change our working directory.
|
|
|
|
TEST_DIRECTORY=$(pwd)/..
|
|
|
|
TEST_OUTPUT_DIRECTORY=$(pwd)
|
|
|
|
|
|
|
|
TEST_NO_CREATE_REPO=t
|
|
|
|
TEST_NO_MALLOC_CHECK=t
|
|
|
|
|
|
|
|
. ../test-lib.sh
|
|
|
|
|
|
|
|
if test -n "$GIT_TEST_INSTALLED" -a -z "$PERF_SET_GIT_TEST_INSTALLED"
|
|
|
|
then
|
|
|
|
error "Do not use GIT_TEST_INSTALLED with the perf tests.
|
|
|
|
|
|
|
|
Instead use:
|
|
|
|
|
|
|
|
./run <path-to-git> -- <tests>
|
|
|
|
|
|
|
|
See t/perf/README for details."
|
|
|
|
fi
|
|
|
|
|
|
|
|
# Variables from test-lib that are normally internal to the tests; we
|
|
|
|
# need to export them for test_perf subshells
|
|
|
|
export TEST_DIRECTORY TRASH_DIRECTORY GIT_BUILD_DIR GIT_TEST_CMP
|
|
|
|
|
|
|
|
MODERN_GIT=$GIT_BUILD_DIR/bin-wrappers/git
|
|
|
|
export MODERN_GIT
|
|
|
|
|
|
|
|
perf_results_dir=$TEST_OUTPUT_DIRECTORY/test-results
|
|
|
|
test -n "$GIT_PERF_SUBSECTION" && perf_results_dir="$perf_results_dir/$GIT_PERF_SUBSECTION"
|
|
|
|
mkdir -p "$perf_results_dir"
|
|
|
|
rm -f "$perf_results_dir"/$(basename "$0" .sh).subtests
|
|
|
|
|
|
|
|
die_if_build_dir_not_repo () {
|
|
|
|
if ! ( cd "$TEST_DIRECTORY/.." &&
|
|
|
|
git rev-parse --build-dir >/dev/null 2>&1 ); then
|
|
|
|
error "No $1 defined, and your build directory is not a repo"
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
|
|
|
if test -z "$GIT_PERF_REPO"; then
|
|
|
|
die_if_build_dir_not_repo '$GIT_PERF_REPO'
|
|
|
|
GIT_PERF_REPO=$TEST_DIRECTORY/..
|
|
|
|
fi
|
|
|
|
if test -z "$GIT_PERF_LARGE_REPO"; then
|
|
|
|
die_if_build_dir_not_repo '$GIT_PERF_LARGE_REPO'
|
|
|
|
GIT_PERF_LARGE_REPO=$TEST_DIRECTORY/..
|
|
|
|
fi
|
|
|
|
|
|
|
|
test_perf_do_repo_symlink_config_ () {
|
|
|
|
test_have_prereq SYMLINKS || git config core.symlinks false
|
|
|
|
}
|
|
|
|
|
|
|
|
test_perf_create_repo_from () {
|
|
|
|
test "$#" = 2 ||
|
tests: send "bug in the test script" errors to the script's stderr
Some of the functions in our test library check that they were invoked
properly with conditions like this:
test "$#" = 2 ||
error "bug in the test script: not 2 parameters to test-expect-success"
If this particular condition is triggered, then 'error' will abort the
whole test script with a bold red error message [1] right away.
However, under certain circumstances the test script will be aborted
completely silently, namely if:
- a similar condition in a test helper function like
'test_line_count' is triggered,
- which is invoked from the test script's "main" shell [2],
- and the test script is run manually (i.e. './t1234-foo.sh' as
opposed to 'make t1234-foo.sh' or 'make test') [3]
- and without the '--verbose' option,
because the error message is printed from within 'test_eval_', where
standard output is redirected either to /dev/null or to a log file.
The only indication that something is wrong is that not all tests in
the script are executed and at the end of the test script's output
there is no "# passed all N tests" message, which are subtle and can
easily go unnoticed, as I had to experience myself.
Send these "bug in the test script" error messages directly to the
test scripts standard error and thus to the terminal, so those bugs
will be much harder to overlook. Instead of updating all ~20 such
'error' calls with a redirection, let's add a BUG() function to
'test-lib.sh', wrapping an 'error' call with the proper redirection
and also including the common prefix of those error messages, and
convert all those call sites [4] to use this new BUG() function
instead.
[1] That particular error message from 'test_expect_success' is
printed in color only when running with or without '--verbose';
with '--tee' or '--verbose-log' the error is printed without
color, but it is printed to the terminal nonetheless.
[2] If such a condition is triggered in a subshell of a test, then
'error' won't be able to abort the whole test script, but only the
subshell, which in turn causes the test to fail in the usual way,
indicating loudly and clearly that something is wrong.
[3] Well, 'error' aborts the test script the same way when run
manually or by 'make' or 'prove', but both 'make' and 'prove' pay
attention to the test script's exit status, and even a silently
aborted test script would then trigger those tools' usual
noticable error messages.
[4] Strictly speaking, not all those 'error' calls need that
redirection to send their output to the terminal, see e.g.
'test_expect_success' in the opening example, but I think it's
better to be consistent.
Signed-off-by: SZEDER Gábor <szeder.dev@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
6 years ago
|
|
|
BUG "not 2 parameters to test-create-repo"
|
|
|
|
repo="$1"
|
|
|
|
source="$2"
|
t/perf: use $MODERN_GIT for all repo-copying steps
Since 1a0962dee (t/perf: fix regression in testing older
versions of git, 2016-06-22), we point "$MODERN_GIT" to a
copy of git that matches the t/perf script itself, and which
can be used for tasks outside of the actual timings. This is
needed because the setup done by perf scripts keeps moving
forward in time, and may use features that the older
versions of git we are testing do not have.
That commit used $MODERN_GIT to fix a case where we relied
on the relatively recent --git-path option. But if you go
back further still, there are more problems.
Since 7501b5921 (perf: make the tests work in worktrees,
2016-05-13), we use "git -C", but versions of git older than
44e1e4d67 (git: run in a directory given with -C option,
2013-09-09) don't know about "-C". So testing an old version
of git with a new version of t/perf will fail the setup
step.
We can fix this by using $MODERN_GIT during the setup;
there's no need to use the antique version, since it doesn't
affect the timings. Likewise, we'll adjust the "init"
invocation; antique versions of git called this "init-db".
Signed-off-by: Jeff King <peff@peff.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
8 years ago
|
|
|
source_git="$("$MODERN_GIT" -C "$source" rev-parse --git-dir)"
|
|
|
|
objects_dir="$("$MODERN_GIT" -C "$source" rev-parse --git-path objects)"
|
|
|
|
mkdir -p "$repo/.git"
|
|
|
|
(
|
|
|
|
cd "$source" &&
|
|
|
|
{ cp -Rl "$objects_dir" "$repo/.git/" 2>/dev/null ||
|
|
|
|
cp -R "$objects_dir" "$repo/.git/"; } &&
|
|
|
|
for stuff in "$source_git"/*; do
|
|
|
|
case "$stuff" in
|
|
|
|
*/objects|*/hooks|*/config|*/commondir)
|
|
|
|
;;
|
|
|
|
*)
|
|
|
|
cp -R "$stuff" "$repo/.git/" || exit 1
|
|
|
|
;;
|
|
|
|
esac
|
|
|
|
done
|
|
|
|
) &&
|
|
|
|
(
|
|
|
|
cd "$repo" &&
|
|
|
|
"$MODERN_GIT" init -q &&
|
|
|
|
test_perf_do_repo_symlink_config_ &&
|
|
|
|
mv .git/hooks .git/hooks-disabled 2>/dev/null &&
|
|
|
|
if test -f .git/index.lock
|
|
|
|
then
|
|
|
|
# We may be copying a repo that can't run "git
|
|
|
|
# status" due to a locked index. Since we have
|
|
|
|
# a copy it's fine to remove the lock.
|
|
|
|
rm .git/index.lock
|
|
|
|
fi
|
|
|
|
) || error "failed to copy repository '$source' to '$repo'"
|
|
|
|
}
|
|
|
|
|
|
|
|
# call at least one of these to establish an appropriately-sized repository
|
|
|
|
test_perf_fresh_repo () {
|
|
|
|
repo="${1:-$TRASH_DIRECTORY}"
|
|
|
|
"$MODERN_GIT" init -q "$repo" &&
|
|
|
|
(
|
|
|
|
cd "$repo" &&
|
|
|
|
test_perf_do_repo_symlink_config_
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
test_perf_default_repo () {
|
|
|
|
test_perf_create_repo_from "${1:-$TRASH_DIRECTORY}" "$GIT_PERF_REPO"
|
|
|
|
}
|
|
|
|
test_perf_large_repo () {
|
|
|
|
if test "$GIT_PERF_LARGE_REPO" = "$GIT_BUILD_DIR"; then
|
|
|
|
echo "warning: \$GIT_PERF_LARGE_REPO is \$GIT_BUILD_DIR." >&2
|
|
|
|
echo "warning: This will work, but may not be a sufficiently large repo" >&2
|
|
|
|
echo "warning: for representative measurements." >&2
|
|
|
|
fi
|
|
|
|
test_perf_create_repo_from "${1:-$TRASH_DIRECTORY}" "$GIT_PERF_LARGE_REPO"
|
|
|
|
}
|
|
|
|
test_checkout_worktree () {
|
|
|
|
git checkout-index -u -a ||
|
|
|
|
error "git checkout-index failed"
|
|
|
|
}
|
|
|
|
|
|
|
|
# Performance tests should never fail. If they do, stop immediately
|
|
|
|
immediate=t
|
|
|
|
|
|
|
|
# Perf tests require GNU time
|
|
|
|
case "$(uname -s)" in Darwin) GTIME="${GTIME:-gtime}";; esac
|
|
|
|
GTIME="${GTIME:-/usr/bin/time}"
|
|
|
|
|
|
|
|
test_run_perf_ () {
|
|
|
|
test_cleanup=:
|
|
|
|
test_export_="test_cleanup"
|
|
|
|
export test_cleanup test_export_
|
|
|
|
"$GTIME" -f "%E %U %S" -o test_time.$i "$SHELL" -c '
|
|
|
|
. '"$TEST_DIRECTORY"/test-lib-functions.sh'
|
|
|
|
test_export () {
|
|
|
|
[ $# != 0 ] || return 0
|
|
|
|
test_export_="$test_export_\\|$1"
|
|
|
|
shift
|
|
|
|
test_export "$@"
|
|
|
|
}
|
|
|
|
'"$1"'
|
|
|
|
ret=$?
|
|
|
|
set | sed -n "s'"/'/'\\\\''/g"';s/^\\($test_export_\\)/export '"'&'"'/p" >test_vars
|
|
|
|
exit $ret' >&3 2>&4
|
|
|
|
eval_ret=$?
|
|
|
|
|
|
|
|
if test $eval_ret = 0 || test -n "$expecting_failure"
|
|
|
|
then
|
|
|
|
test_eval_ "$test_cleanup"
|
|
|
|
. ./test_vars || error "failed to load updated environment"
|
|
|
|
fi
|
|
|
|
if test "$verbose" = "t" && test -n "$HARNESS_ACTIVE"; then
|
|
|
|
echo ""
|
|
|
|
fi
|
|
|
|
return "$eval_ret"
|
|
|
|
}
|
|
|
|
|
|
|
|
test_wrapper_ () {
|
|
|
|
test_wrapper_func_=$1; shift
|
|
|
|
test_start_
|
|
|
|
test "$#" = 3 && { test_prereq=$1; shift; } || test_prereq=
|
|
|
|
test "$#" = 2 ||
|
tests: send "bug in the test script" errors to the script's stderr
Some of the functions in our test library check that they were invoked
properly with conditions like this:
test "$#" = 2 ||
error "bug in the test script: not 2 parameters to test-expect-success"
If this particular condition is triggered, then 'error' will abort the
whole test script with a bold red error message [1] right away.
However, under certain circumstances the test script will be aborted
completely silently, namely if:
- a similar condition in a test helper function like
'test_line_count' is triggered,
- which is invoked from the test script's "main" shell [2],
- and the test script is run manually (i.e. './t1234-foo.sh' as
opposed to 'make t1234-foo.sh' or 'make test') [3]
- and without the '--verbose' option,
because the error message is printed from within 'test_eval_', where
standard output is redirected either to /dev/null or to a log file.
The only indication that something is wrong is that not all tests in
the script are executed and at the end of the test script's output
there is no "# passed all N tests" message, which are subtle and can
easily go unnoticed, as I had to experience myself.
Send these "bug in the test script" error messages directly to the
test scripts standard error and thus to the terminal, so those bugs
will be much harder to overlook. Instead of updating all ~20 such
'error' calls with a redirection, let's add a BUG() function to
'test-lib.sh', wrapping an 'error' call with the proper redirection
and also including the common prefix of those error messages, and
convert all those call sites [4] to use this new BUG() function
instead.
[1] That particular error message from 'test_expect_success' is
printed in color only when running with or without '--verbose';
with '--tee' or '--verbose-log' the error is printed without
color, but it is printed to the terminal nonetheless.
[2] If such a condition is triggered in a subshell of a test, then
'error' won't be able to abort the whole test script, but only the
subshell, which in turn causes the test to fail in the usual way,
indicating loudly and clearly that something is wrong.
[3] Well, 'error' aborts the test script the same way when run
manually or by 'make' or 'prove', but both 'make' and 'prove' pay
attention to the test script's exit status, and even a silently
aborted test script would then trigger those tools' usual
noticable error messages.
[4] Strictly speaking, not all those 'error' calls need that
redirection to send their output to the terminal, see e.g.
'test_expect_success' in the opening example, but I think it's
better to be consistent.
Signed-off-by: SZEDER Gábor <szeder.dev@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
6 years ago
|
|
|
BUG "not 2 or 3 parameters to test-expect-success"
|
|
|
|
export test_prereq
|
|
|
|
if ! test_skip "$@"
|
|
|
|
then
|
|
|
|
base=$(basename "$0" .sh)
|
|
|
|
echo "$test_count" >>"$perf_results_dir"/$base.subtests
|
|
|
|
echo "$1" >"$perf_results_dir"/$base.$test_count.descr
|
perf-lib.sh: remove GIT_TEST_INSTALLED from perf-lib.sh
Follow-up my preceding change which fixed the immediate "./run
<revisions>" regression in 0baf78e7bc ("perf-lib.sh: rely on
test-lib.sh for --tee handling", 2019-03-15) and entirely get rid of
GIT_TEST_INSTALLED from perf-lib.sh (and aggregate.perl).
As noted in that change the dance we're doing with GIT_TEST_INSTALLED
perf-lib.sh isn't necessary, but there I was doing the most minimal
set of changes to quickly fix a regression.
But it's much simpler to never deal with the "GIT_TEST_INSTALLED" we
were setting in perf-lib.sh at all. Instead the run_dirs_helper() sets
the previously inferred $PERF_RESULTS_PREFIX directly.
Setting this at the callsite that's already best positioned to
exhaustively know about all the different cases we need to handle
where PERF_RESULTS_PREFIX isn't what we want already (the empty
string) makes the most sense. In one-off cases like:
./run ./p0000-perf-lib-sanity.sh
./p0000-perf-lib-sanity.sh
We'll just do the right thing because PERF_RESULTS_PREFIX will be
empty, and test-lib.sh takes care of finding where our git is.
Any refactoring of this code needs to change both the shell code and
the Perl code in aggregate.perl, because when running e.g.:
./run ../../ -- <test>
The "../../" path to a relative bindir needs to be munged to a
filename containing the results, and critically aggregate.perl does
not get passed the path to those aggregations, just "../..".
Let's fix cases where aggregate.perl would print e.g. ".." in its
report output for this, and "git" for "/home/avar/g/git", i.e. it
would always pick the last element. Now'll always print the full path
instead.
This also makes the code sturdier, e.g. you can feed "../.." to
"./run" and then an absolute path to the aggregate.perl script, as
long as the absolute path and "../.." resolved to the same directory
printing the aggregation will work.
Also simplify the "[_*]" on the RHS of "tr -c", we're trimming
everything to "_", so we don't need that.
Signed-off-by: Ævar Arnfjörð Bjarmason <avarab@gmail.com>
6 years ago
|
|
|
base="$perf_results_dir"/"$PERF_RESULTS_PREFIX$(basename "$0" .sh)"."$test_count"
|
|
|
|
"$test_wrapper_func_" "$@"
|
|
|
|
fi
|
|
|
|
|
|
|
|
test_finish_
|
|
|
|
}
|
|
|
|
|
|
|
|
test_perf_ () {
|
|
|
|
if test -z "$verbose"; then
|
|
|
|
printf "%s" "perf $test_count - $1:"
|
|
|
|
else
|
|
|
|
echo "perf $test_count - $1:"
|
|
|
|
fi
|
|
|
|
for i in $(test_seq 1 $GIT_PERF_REPEAT_COUNT); do
|
|
|
|
say >&3 "running: $2"
|
|
|
|
if test_run_perf_ "$2"
|
|
|
|
then
|
|
|
|
if test -z "$verbose"; then
|
|
|
|
printf " %s" "$i"
|
|
|
|
else
|
|
|
|
echo "* timing run $i/$GIT_PERF_REPEAT_COUNT:"
|
|
|
|
fi
|
|
|
|
else
|
|
|
|
test -z "$verbose" && echo
|
|
|
|
test_failure_ "$@"
|
|
|
|
break
|
|
|
|
fi
|
|
|
|
done
|
|
|
|
if test -z "$verbose"; then
|
|
|
|
echo " ok"
|
|
|
|
else
|
|
|
|
test_ok_ "$1"
|
|
|
|
fi
|
|
|
|
"$TEST_DIRECTORY"/perf/min_time.perl test_time.* >"$base".result
|
|
|
|
}
|
|
|
|
|
|
|
|
test_perf () {
|
|
|
|
test_wrapper_ test_perf_ "$@"
|
|
|
|
}
|
|
|
|
|
|
|
|
test_size_ () {
|
|
|
|
say >&3 "running: $2"
|
|
|
|
if test_eval_ "$2" 3>"$base".result; then
|
|
|
|
test_ok_ "$1"
|
|
|
|
else
|
|
|
|
test_failure_ "$@"
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
|
|
|
test_size () {
|
|
|
|
test_wrapper_ test_size_ "$@"
|
|
|
|
}
|
|
|
|
|
|
|
|
# We extend test_done to print timings at the end (./run disables this
|
|
|
|
# and does it after running everything)
|
|
|
|
test_at_end_hook_ () {
|
|
|
|
if test -z "$GIT_PERF_AGGREGATING_LATER"; then
|
|
|
|
( cd "$TEST_DIRECTORY"/perf && ./aggregate.perl $(basename "$0") )
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
|
|
|
test_export () {
|
|
|
|
export "$@"
|
|
|
|
}
|