To check that a git command fails and to inspect its error message we
usually execute a command like this throughout our test suite:
test_must_fail git command --option 2>output.err
Note that this command doesn't limit the redirection to the git
command, but it redirects the standard error of the 'test_must_fail'
helper function as well. This is wrong for several reasons:
- If that git command were to succeed or die for an unexpected
reason e.g. signal, then 'test_must_fail's helpful error message
would end up in the given file instead of on the terminal or in
't/test-results/$T.out', when the test is run with '-v' or
'--verbose-log', respectively.
- If the test is run with '-x', the trace of executed commands in
'test_must_fail' will go to stderr as well (except for more recent
Bash versions supporting $BASH_XTRACEFD), and then in turn will
end up in the given file.
- If the git command's error message is checked verbatim with
'test_cmp', then the '-x' trace will cause the test to fail.
- Though rather unlikely, if the git command's error message is
checked with 'test_i18ngrep', then its pattern might match
'test_must_fail's error message (or '-x' trace) instead of the
given git command's, potentially hiding a bug.
Teach the 'test_must_fail' helper the 'stderr=<file>' option to save
the executed git command's standard error to that file, so we can
avoid all the bad side effects of redirecting the whole thing's
standard error.
The same issues apply to the 'test_might_fail' helper function as
well, but since it's implemented as a thin wrapper around
'test_must_fail', no modifications were necessary and 'test_might_fail
stderr=output.err git command' will just work.
Signed-off-by: SZEDER Gábor <[email protected]>
---
Notes:
I considered adding an analogous 'stdout=<file>' option as well, but in
the end haven't, because:
- 'test_must_fail' doesn't print anything to its standard output, so a
plain '>out' redirection at the end of the line doesn't have any
undesirable side effects.
- We would need four conditions to cover all possible combinations of
'stdout=' and 'stderr='. Considering the above point, I don't think
it's worth it.
t/README | 6 ++++++
t/test-lib-functions.sh | 35 +++++++++++++++++++++++++----------
2 files changed, 31 insertions(+), 10 deletions(-)
diff --git a/t/README b/t/README
index 1a1361a806..2528359e9d 100644
--- a/t/README
+++ b/t/README
@@ -430,6 +430,10 @@ Don't:
use 'test_must_fail git cmd'. This will signal a failure if git
dies in an unexpected way (e.g. segfault).
+ Don't redirect 'test_must_fail's standard error like
+ 'test_must_fail git cmd 2>err'. Instead, run 'test_must_fail
+ stderr=err git cmd'.
+
On the other hand, don't use test_must_fail for running regular
platform commands; just use '! cmd'. We are not in the business
of verifying that the world given to us sanely works.
@@ -670,6 +674,8 @@ library for your script to use.
Multiple signals can be specified as a comma separated list.
Currently recognized signal names are: sigpipe, success.
(Don't use 'success', use 'test_might_fail' instead.)
+ stderr=<file>:
+ Save <git-command>'s standard error to <file>.
- test_might_fail [<options>] <git-command>
diff --git a/t/test-lib-functions.sh b/t/test-lib-functions.sh
index 26b149ac1d..d2f561477c 100644
--- a/t/test-lib-functions.sh
+++ b/t/test-lib-functions.sh
@@ -618,18 +618,33 @@ list_contains () {
# Multiple signals can be specified as a comma separated list.
# Currently recognized signal names are: sigpipe, success.
# (Don't use 'success', use 'test_might_fail' instead.)
+# stderr=<file>:
+# Save the git command's standard error to <file>.
test_must_fail () {
- case "$1" in
- ok=*)
- _test_ok=${1#ok=}
- shift
- ;;
- *)
- _test_ok=
- ;;
- esac
- "$@"
+ _test_ok= _test_stderr=
+ while test $# -gt 0
+ do
+ case "$1" in
+ stderr=*)
+ _test_stderr=${1#stderr=}
+ shift
+ ;;
+ ok=*)
+ _test_ok=${1#ok=}
+ shift
+ ;;
+ *)
+ break
+ ;;
+ esac
+ done
+ if test -n "$_test_stderr"
+ then
+ "$@" 2>"$_test_stderr"
+ else
+ "$@"
+ fi
exit_code=$?
if test $exit_code -eq 0 && ! list_contains "$_test_ok" success
then
--
2.16.1.180.g07550b0b1b