urchin/urchin

568 lines
16 KiB
Plaintext
Raw Normal View History

#!/bin/sh
2012-10-04 11:24:03 +00:00
2016-02-29 04:09:46 +00:00
# Copyright (c) 2013, 2014, 2015, 2016 Thomas Levine
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU Affero General Public License as
# published by the Free Software Foundation, either version 3 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 Affero General Public License for more details.
#
# You should have received a copy of the GNU Affero General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
2016-02-08 16:09:00 +00:00
2016-02-29 04:09:46 +00:00
# Copyright (c) 2014, Michael Klement
# Copyright (c) 2012, ScraperWiki Limited
# All rights reserved.
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions are met:
#
# Redistributions of source code must retain the above copyright notice, this
# list of conditions and the following disclaimer.
#
# Redistributions in binary form must reproduce the above copyright notice,
# this list of conditions and the following disclaimer in the documentation
# and/or other materials provided with the distribution.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
# POSSIBILITY OF SUCH DAMAGE.
2016-02-28 01:18:55 +00:00
set -e
2016-02-08 16:09:00 +00:00
2016-02-29 02:32:58 +00:00
DEFAULT_SHELLS='sh bash dash mksh zsh'
if [ -n "$ZSH_VERSION" ]; then
# avoid "no matches found: *" error when directories are empty
setopt NULL_GLOB
emulate sh
fi
2016-02-28 14:01:17 +00:00
2016-02-08 16:05:56 +00:00
# Make sure that CDPATH isn't set, as it causes `cd` to behave unpredictably -
2016-02-28 01:18:55 +00:00
# notably, it can produce output.
Make sure that CDPATH isn't set, as it causes `cd` to behave unpredictably - notably, it can produce output, which breaks fullpath(). Also: Improved CLI help, updated URLs in read-me, cleaned up package.json: I've tried to clarify the intent of `-x` in the CLI help, but I haven't touched the read-me in that respect. I don't see any benefit to `-x`: * Just using `#/bin/sh` as the shebang line in combination with `-s <shell>` gives you the same functionality, * When it comes to invoking scripts from _within_ test scripts, nothing can do the work for you: you consciously have to mark the invocation with _something_ to indicate that it should be controlled from the outside; it won't get any easier than `$TEST_SHELL ...` * Finally, using a shebang line such as `#!/usr/bin/env urchin -x` is problematic for two reasons: * Some platforms can handle only *1* argument in a shebang line. * In a _package-local_ installation, `#!/usr/bin/env` may not find the Urchin executable. I'm also not sure how the following (from `readme.md`) fits in the picture: > It might make sense if you do this. export TEST_SHELL=zsh && urchin -x export TEST_SHELL=bash && urchin -x (As an aside: To achieve the same thing, you don't need `export`; `TEST_SHELL=zsh urchin -x` and `TEST_SHELL=bash urchin -x` is the better choice.) How does this relate to use in a _shebang line_? `urchin_help()` now uses a here-doc: easier to maintain, and should work in all Bourne-like shells. `readmeFilename` removed from `package.json`: > "The readmeFilename does not need to ever be in your actual package.json file" - npm/npm#3573
2014-12-03 14:48:49 +00:00
unset CDPATH
2016-02-28 09:08:20 +00:00
# All temporary files go here
2016-02-28 19:18:54 +00:00
urchin_tmp=$(mktemp -d)
> $urchin_tmp/log
2016-02-28 09:08:20 +00:00
urchin_exit() {
2016-02-28 19:18:54 +00:00
rm -Rf "$urchin_tmp"
2016-02-28 09:08:20 +00:00
exit "$@"
}
2016-02-28 13:12:55 +00:00
stdout_file() {
2016-02-28 20:31:17 +00:00
the_test="$1"
the_shell="$2"
x="$urchin_tmp/stdout$(fullpath "$the_test")"
mkdir -p "$x"
echo "$x/$(echo "$the_shell" | md5sum | cut -d\ -f1)"
2016-02-28 13:12:55 +00:00
}
2016-02-28 09:12:57 +00:00
# Expand relative paths
fullpath() {
readlink -f -- "$1"
}
2016-02-28 09:12:57 +00:00
2016-02-28 11:22:51 +00:00
remove_trailing_slash() {
echo "$1" | sed s/\\/$//
}
2016-02-28 11:18:39 +00:00
urchin_root() {
2016-02-28 11:22:51 +00:00
# Call recursively but remember the original argument.
current="$(remove_trailing_slash "$1")"
if test -n "$2"; then
orig="$2"
else
orig="$1"
fi
2016-02-28 11:18:39 +00:00
2016-02-28 22:49:57 +00:00
abscurrent="$(fullpath "$1")"
if test "$abscurrent" = / ||
basename "$abscurrent" | grep '^\.' > /dev/null; then
# Stop traversing upwards at / and at hidden directories.
2016-02-28 11:22:51 +00:00
if test -d "$orig"; then
echo "$orig"
else
2016-02-28 12:59:36 +00:00
dirname -- "$orig"
2016-02-28 11:22:51 +00:00
fi
elif ! test -e "$current"; then
echo "$current: No such file or directory">&2
return 1
elif test -f "$current"; then
2016-02-28 12:59:36 +00:00
urchin_root "$(dirname -- "$current")" "$orig"
2016-02-28 11:27:55 +00:00
elif test -f "$current"/.urchin; then
2016-02-28 11:22:51 +00:00
remove_trailing_slash "$current"
else
urchin_root "$current"/.. "$orig"
fi
2016-02-28 11:18:39 +00:00
}
2016-01-29 17:17:31 +00:00
# Urchin version number
2016-02-28 01:18:55 +00:00
VERSION=0.1.0-rc1
2012-10-11 18:50:03 +00:00
2012-10-11 00:43:13 +00:00
indent() {
level="$1"
2016-02-08 15:14:22 +00:00
if test "$level" -gt 0; then
printf "%$((2 * ${level}))s"
fi
2012-10-11 00:43:13 +00:00
}
2012-10-04 11:29:34 +00:00
recurse() {
2016-02-28 12:14:16 +00:00
requested_path="$1"
potential_test="$(fullpath "$2")"
2016-02-28 14:01:17 +00:00
cycle_shell="$3"
2016-02-28 12:14:16 +00:00
TEST_SHELL="$4"
2016-02-28 10:27:49 +00:00
2016-02-28 10:00:53 +00:00
for ignore in setup_dir teardown_dir setup teardown; do
2016-02-28 18:41:00 +00:00
if test "$(basename "$potential_test")" = $ignore; then
2016-02-28 10:00:53 +00:00
return
fi
done
2012-10-10 18:40:49 +00:00
2016-02-28 12:15:42 +00:00
echo "$requested_path" | grep "^$potential_test" > /dev/null ||
echo "$potential_test" | grep "^$requested_path" > /dev/null ||
2016-02-28 12:14:16 +00:00
return 0
2016-02-28 12:02:57 +00:00
2016-02-28 14:32:33 +00:00
validate_strings "$potential_test" 'Test file names'
2016-02-28 12:02:57 +00:00
2016-02-28 01:18:55 +00:00
if [ -d "$potential_test" ]; then
(
2016-02-28 09:36:48 +00:00
cd -- "$potential_test" > /dev/null
2016-02-28 19:18:54 +00:00
if test -f setup_dir; then . ./setup_dir; fi
2014-11-05 17:38:22 +00:00
2016-02-28 01:18:55 +00:00
for test in *; do
2016-02-28 12:35:27 +00:00
if test "$test" = '*' && ! test -e $test; then
# The directory is empty.
break
fi
2012-10-11 00:43:13 +00:00
2016-02-28 16:05:03 +00:00
(
2016-02-28 19:18:54 +00:00
if test -f setup; then . ./setup; fi
2016-02-29 01:03:17 +00:00
if recurse "$requested_path" "$test" "$cycle_shell" "$TEST_SHELL"; then
exit_code=0
else
exit_code=$?
fi
2016-02-28 19:18:54 +00:00
if test -f teardown; then . ./teardown; fi
2016-02-29 01:07:04 +00:00
exit $exit_code
2016-02-28 16:05:03 +00:00
) &
2016-02-28 16:19:42 +00:00
if $run_in_series; then
2016-02-29 01:03:17 +00:00
if wait $!; then exit_code=0; else exit_code=$?; fi
2016-02-28 16:05:03 +00:00
if $exit_on_not_ok && test $exit_code -ne 0; then
2016-02-28 19:18:54 +00:00
if test -f teardown_dir; then . ./teardown_dir; fi
2016-02-29 01:03:17 +00:00
return 1
2016-02-28 16:05:03 +00:00
fi
fi
2012-10-08 12:50:48 +00:00
done
2016-02-28 16:12:13 +00:00
wait
2016-02-28 19:18:54 +00:00
if test -f teardown_dir; then . ./teardown_dir; fi
2016-02-28 10:00:53 +00:00
)
else
2016-02-28 01:25:36 +00:00
if [ -x "$potential_test" ]; then
2016-02-28 12:59:36 +00:00
cd -- "$(dirname -- "$potential_test")"
2016-02-28 14:01:17 +00:00
# Determine the environment variable to define for test scripts
# that reflects the specified or implied shell to use for shell-code tests.
2016-02-28 16:32:06 +00:00
while read the_test_shell; do
(
2016-02-28 19:18:54 +00:00
if test -f setup; then . ./setup; fi
2016-02-28 16:32:06 +00:00
# Run the test
start=$(date +%s)
set +e
2016-02-28 16:53:49 +00:00
{
if $cycle_shell &&
has_sh_or_no_shebang_line "$potential_test"; then
2016-02-29 03:40:27 +00:00
TEST_SHELL="$the_test_shell" $TIMEOUT \
"$the_test_shell" "$potential_test"
2016-02-28 16:53:49 +00:00
else
2016-02-29 02:13:40 +00:00
# Shell cycling is disabled with -n; use the present value of
# TEST_SHELL or default to /bin/sh
if [ -n "$TEST_SHELL" ]; then
2016-02-29 03:40:27 +00:00
$TIMEOUT "$potential_test"
2016-02-29 02:13:40 +00:00
else
2016-02-29 03:40:27 +00:00
TEST_SHELL=/bin/sh $TIMEOUT "$potential_test"
2016-02-29 02:13:40 +00:00
fi
2016-02-28 16:53:49 +00:00
fi
2016-02-28 20:31:17 +00:00
} > "$(stdout_file "$potential_test" "$the_test_shell")" 2>&1
2016-02-28 16:32:06 +00:00
exit_code="$?"
set -e
finish=$(date +%s)
2016-02-28 19:18:54 +00:00
if test -f teardown; then . ./setup; fi
2016-02-28 16:32:06 +00:00
if [ $exit_code -eq 0 ]; then
result=ok
elif [ $exit_code -eq 3 ]; then
result=skip
else
result=not_ok
fi
elapsed=$(($finish - $start))
printf "${potential_test}\t${the_test_shell}\t${result}\t${elapsed}\n" \
2016-02-28 19:18:54 +00:00
>> $urchin_tmp/log
2016-02-28 16:32:06 +00:00
exit "$exit_code"
) &
if $run_in_series; then
2016-02-29 01:03:17 +00:00
if wait $!; then exit_code=0; else exit_code=$?; fi
2016-02-28 16:32:06 +00:00
if $exit_on_not_ok && test $exit_code -ne 0; then
2016-02-28 19:18:54 +00:00
if test -f teardown_dir; then . ./teardown_dir; fi
2016-02-28 16:32:06 +00:00
return 1
fi
fi
done < $shell_list
wait
2012-10-04 11:29:34 +00:00
else
2016-02-28 15:50:53 +00:00
# Shell is ''
2016-02-28 19:18:54 +00:00
printf "${potential_test}\t\tskip\t0\n" >> $urchin_tmp/log
fi
2016-02-28 09:08:20 +00:00
fi
}
2016-02-28 09:08:20 +00:00
report_outcome() {
2016-02-28 12:41:39 +00:00
root="$1"
tap_format="$2"
log_file="$3"
start="$4"
finish="$5"
escaped_root="$(fullpath "$root" | sed 's/\//\\\//g')"
2016-02-28 10:39:32 +00:00
elapsed=$(($finish - $start))
2016-02-28 10:07:55 +00:00
2016-02-28 12:53:26 +00:00
if "$tap_format"; then
printf \#\
fi
echo Running tests at $(date +%Y-%m-%dT%H:%M:%S)
2016-02-28 09:25:21 +00:00
2016-02-28 10:57:27 +00:00
for number in n oks skips not_oks; do
eval "$number=0"
done
2016-02-28 11:14:23 +00:00
# Use a temporary file rather than a pipe because a pipe starts a sub-shell
# and thus makes the above variables local.
sorted_log_file=$(mktemp)
2016-02-29 00:41:03 +00:00
# Sort in alphabetical order.
# GNU sort requires -m, and BSD sort doesn't.
2016-02-29 01:37:53 +00:00
sort "$log_file" > $sorted_log_file
2016-02-29 00:41:03 +00:00
2016-02-28 11:14:23 +00:00
while read line; do
2016-02-28 14:32:33 +00:00
abspath=$(echo "$line" | cut -f1)
2016-02-28 13:12:55 +00:00
path=$(echo "$abspath" | sed "s/$escaped_root\/\?//")
2016-02-28 14:32:33 +00:00
the_shell=$(echo "$line" | cut -f2)
result=$(echo "$line" | cut -f3)
file_elapsed=$(echo "$line" | cut -f4)
2016-02-28 11:06:54 +00:00
2016-02-28 12:53:26 +00:00
prevdir=$currentdir
2016-02-28 12:59:36 +00:00
currentdir="$(dirname -- "$path")"
2016-02-28 12:53:26 +00:00
2016-02-28 11:06:54 +00:00
# Number of files that have run, including this one
2016-02-28 10:27:49 +00:00
n=$(($n + 1))
2016-02-28 11:06:54 +00:00
# Number of files that have been ok, not ok, and skipped
eval "old_count=${result}s"
eval "${result}s=$(($old_count+1))"
2016-02-28 09:12:57 +00:00
2016-02-28 10:27:49 +00:00
if $tap_format; then
2016-02-28 20:18:22 +00:00
if [ "$result" = not_ok ]; then
2016-02-28 10:27:49 +00:00
not='not '
else
not=''
fi
2016-02-28 20:18:22 +00:00
if [ "$result" = skip ]; then
2016-02-28 10:49:40 +00:00
skip='# SKIP'
2016-02-28 10:27:49 +00:00
else
skip=''
fi
2016-02-28 15:50:53 +00:00
if test -z "$the_shell"; then
the_shell='File is not executable.'
fi
echo "${not}ok $n - ${path} ($the_shell) ${skip}"
2016-02-28 22:41:13 +00:00
if $verbose || [ "$result" = not_ok ]; then
2016-02-28 10:27:49 +00:00
echo '# ------------ Begin output ------------'
2016-02-28 20:31:17 +00:00
sed 's/^/# /' "$(stdout_file "$abspath" "$the_shell")"
2016-02-28 10:27:49 +00:00
echo '# ------------ End output ------------'
fi
2016-02-28 10:39:32 +00:00
echo "# Previous test took $file_elapsed seconds."
2016-02-28 12:45:38 +00:00
else
2016-02-28 12:53:26 +00:00
if test "$prevdir" != "$currentdir"; then
echo
fi
2016-02-28 15:28:56 +00:00
if test "$prevpath" != "$path"; then
printf "$(dirname -- "$path")/\n> $(basename -- "$path")\n"
fi
2016-02-28 12:45:38 +00:00
case "$result" in
ok)
# On success, print a green '✓'
printf '\033[32m✓ \033[0m'
2016-02-28 16:12:13 +00:00
echo "${the_shell} ($file_elapsed $(plural second $file_elapsed))"
2016-02-28 12:45:38 +00:00
;;
not_ok)
# On not_ok, print a red '✗'
printf '\033[31m✗ \033[0m'
2016-02-28 16:12:13 +00:00
echo "${the_shell} ($file_elapsed $(plural second $file_elapsed))"
2016-02-28 12:45:38 +00:00
;;
skip)
2016-02-28 15:50:53 +00:00
if test -z "$the_shell"; then
2016-02-28 18:41:00 +00:00
echo ' (File is not executable.)'
else
echo " ${the_shell} ($file_elapsed $(plural second $file_elapsed))"
2016-02-28 15:50:53 +00:00
fi
2016-02-28 12:45:38 +00:00
;;
esac
2016-02-28 22:41:13 +00:00
if $verbose || test "$result" = not_ok; then
sed 's/^/ # /' "$(stdout_file "$abspath" "$the_shell")"
fi
2016-02-28 10:27:49 +00:00
fi
2016-02-28 09:17:16 +00:00
2016-02-28 15:28:56 +00:00
prevpath="$path"
2016-02-28 11:14:23 +00:00
done < $sorted_log_file
rm $sorted_log_file
2016-02-28 09:17:16 +00:00
if $tap_format; then
2016-02-28 11:14:23 +00:00
echo "# Full test suite took $elapsed $(plural second $elapsed)."
echo 1..$n
2016-02-28 09:17:16 +00:00
else
2016-02-28 12:53:26 +00:00
echo
2016-02-28 09:17:16 +00:00
echo "Done, took $elapsed $(plural second $elapsed)."
2016-02-28 12:45:38 +00:00
printf '%s\n' "$oks $(plural test "$oks") passed."
2016-02-28 12:59:36 +00:00
printf '%s\n' "$skips $(plural test "$skips") skipped."
2016-02-28 09:17:16 +00:00
2016-02-28 10:57:27 +00:00
# If any tests are not ok, print the message in red.
if [ $not_oks -gt 0 ] ; then
printf '\033[31m'
fi
2016-02-28 12:45:38 +00:00
printf '%s\n' "$not_oks $(plural test "$not_oks") failed."
printf '\033[m\n'
2016-02-28 09:17:16 +00:00
fi
2016-02-28 10:57:27 +00:00
test "$not_oks" -eq '0'
2012-10-04 11:29:34 +00:00
}
2012-10-04 16:43:49 +00:00
has_sh_or_no_shebang_line() {
2016-02-08 16:05:56 +00:00
# no shebang line at all
head -n 1 "$1" | grep -vqE '^#!' && return 0
# shebang line is '#!/bin/sh' or legal variations thereof
head -n 1 "$1" | grep -qE '^#![[:blank:]]*/bin/sh($|[[:blank:]])' &&
return 0
return 1
}
USAGE="usage: $0 [<options>] <test directory>"
2012-10-11 05:46:02 +00:00
2012-10-11 06:21:05 +00:00
urchin_help() {
cat <<EOF
$USAGE
2016-02-28 14:01:17 +00:00
By default, Urchin checks for the following shells and runs every
particular test file once per shell.
2016-02-28 15:16:57 +00:00
$(echo "$DEFAULT_SHELLS" | sed 's/ /\n /g')
2016-02-28 14:01:17 +00:00
On each run,
1. The TEST_SHELL environment variable is set to the particular shell.
2. If the test file has a shebang line of "#!/bin/sh" or no shebang
line at all, the test script is also executed in that shell.
2016-02-28 14:12:21 +00:00
The following flags affect how this multiple-shell testing is handled.
-s, --shell <shell> Tell Urchin to use a different list of shells.
(You can pass this flag multiple times.)
-n, --disable-cycling Disable the cycling of shells; Urchin will execute
test files directly, and it will not manipulate the
TEST_SHELL environment variable.
2016-02-28 15:15:09 +00:00
The following flags affect how Urchin processes tests.
2016-02-28 14:12:21 +00:00
2016-02-28 16:00:07 +00:00
-b, --run-in-series Run tests in series. The default is to run tests
in parallel where possible.
2016-02-28 14:12:21 +00:00
-e, --exit-on-fail Stop running if any single test fails.
2016-02-28 15:15:09 +00:00
This is useful if you are running something
configuration files with Urchin.
-T, --timeout <seconds> Kill a test if it runs for longer than the
specified duration. The default is no timeout.
2016-02-28 14:12:21 +00:00
-f, --force Force running even if the test directory's name
does not contain the word "test".
These options affect how results are formatted.
-t, --tap Format output in Test Anything Protocol (TAP)
-v, --verbose Print stdout from all tests, not just failed tests.
The remaining flags provide information about urchin.
-h, --help Display this help.
--version Display the version number.
Go to https://github.com/tlevine/urchin for documentation on writing tests.
EOF
2012-10-11 06:21:05 +00:00
}
2012-10-11 05:46:02 +00:00
2013-06-20 17:56:29 +00:00
plural () {
# Make $1 a plural according to the number $2.
# If $3 is supplied, use that instead of "${1}s".
# Result is written to stdout.
if [ "$2" = 1 ]
then
printf '%s\n' "$1"
else
printf '%s\n' "${3-${1}s}"
fi
}
2016-02-28 14:32:33 +00:00
validate_strings() {
test $(echo "$1" | wc -l) -eq 1 || {
echo '$1 may not contain tab or newline characters.' >&2
echo 'If this is really a problem, tell me, and I may fix it.' >&2
urchin_exit 11
}
}
2016-02-28 14:01:17 +00:00
cycle_shell=true
2016-02-28 19:18:54 +00:00
shell_list=$urchin_tmp/shell_list
2016-02-28 16:00:07 +00:00
run_in_series=false
force=false
2016-02-28 10:57:27 +00:00
exit_on_not_ok=false
2016-01-27 00:32:51 +00:00
tap_format=false
2016-02-28 22:41:13 +00:00
verbose=false
2012-10-11 06:21:05 +00:00
while [ $# -gt 0 ]
do
case "$1" in
2016-02-28 19:45:18 +00:00
-b|--run-in-series) run_in_series=true;;
-e|--exit-on-fail) exit_on_not_ok=true;;
2016-02-28 22:36:45 +00:00
-f|--force) force=true;;
-s|--shell)
shift
shell_for_sh_tests=$1
2016-02-28 14:01:17 +00:00
2016-02-08 16:05:56 +00:00
which "$shell_for_sh_tests" > /dev/null || {
echo "Cannot find specified shell: '$shell_for_sh_tests'" >&2
urchin_help >&2
2016-02-28 09:08:20 +00:00
urchin_exit 11
2016-02-28 14:01:17 +00:00
}
2016-02-28 14:32:33 +00:00
validate_strings "$shell_for_sh_tests" 'Shell paths'
2016-02-28 20:56:21 +00:00
if echo "$shell_for_sh_tests" | grep \ > /dev/null; then
echo "Warning: It is best if shell paths contain no spaces so that
you don't need to quote the TEST_SHELL variable." > /dev/stderr
fi
2016-02-28 14:01:17 +00:00
2016-02-28 14:14:55 +00:00
echo "$shell_for_sh_tests" >> "$shell_list"
2016-02-28 14:01:17 +00:00
;;
2016-02-28 22:36:45 +00:00
-n|--disable-cycling) cycle_shell=false;;
2016-02-28 22:33:47 +00:00
-t|--tap) tap_format=true;;
2016-02-29 03:37:04 +00:00
-T|--timeout) shift; urchin_timeout="$1" ;;
2016-02-28 22:41:13 +00:00
-v|--verbose) verbose=true;;
-h|--help) urchin_help
2016-02-29 01:03:17 +00:00
urchin_exit 0;;
2016-02-28 14:12:21 +00:00
--version) echo "$VERSION"
2016-02-28 09:08:20 +00:00
urchin_exit;;
-*) urchin_help >&2
2016-02-29 01:03:17 +00:00
urchin_exit 11;;
2012-10-11 06:21:05 +00:00
*) break;;
esac
shift
done
if ! $cycle_shell && test -f "$shell_list"; then
2016-02-28 16:53:49 +00:00
echo "The -n/--disable-cycling and -s/--shell options clash with each other." >&2
urchin_exit 11
fi
2016-02-28 15:19:15 +00:00
# If -s was not passed, use the available default shells.
2016-02-28 14:14:55 +00:00
if ! test -f "$shell_list"; then
2016-02-28 16:53:49 +00:00
if $cycle_shell; then
for shell in $DEFAULT_SHELLS; do
if which $shell > /dev/null; then
echo $shell >> "$shell_list"
fi
done
else
echo > "$shell_list"
fi
2016-02-28 14:14:55 +00:00
fi
2012-10-11 06:21:05 +00:00
# Verify argument for main stuff
2016-02-28 12:02:57 +00:00
if [ "$#" != '1' ] || [ ! -e "$1" ]; then
if [ -n "$1" ] && [ ! -e "$1" ]; then
echo "No such file or directory: '$1'" >&2
fi
echo "$USAGE" >&2
2016-02-28 09:08:20 +00:00
urchin_exit 11
2012-10-11 19:47:08 +00:00
fi
2012-10-11 06:21:05 +00:00
2016-02-29 01:37:53 +00:00
# Warn about strange sort commands
weird_string='\n- c\n-- b\n--- c\n---- a\n'
if test $(printf "$weird_string" | sort | tr -d '[ \n-]') != cbca; then
echo 'Your version of sort sorts in dictionary order (-d) by default.
Depending on how you name your tests, your Urchin output may look strange.
If this is a problem, install BusyBox or BSD coreutils.' >&2
fi
2016-02-29 03:37:04 +00:00
if test -n "$urchin_timeout"; then
# Choose the timeout command
if timeout -t 0 true; then
TIMEOUT="timeout -t $urchin_timeout"
elif timeout 0 true; then
TIMEOUT="timeout $urchin_timeout"
else
echo I couldn\'t figure out to use your version of timeout >&2
urchin_exit 1
fi
fi
2016-02-29 01:37:53 +00:00
2016-02-28 16:19:42 +00:00
if $exit_on_not_ok && ! $run_in_series; then
2016-02-28 16:12:13 +00:00
echo 'You must also pass -b/--series in order to use -e/--exit-on-fail.' >&2
urchin_exit 11
fi
2012-10-11 06:21:05 +00:00
# Run or present the Molly guard.
2016-02-28 12:02:57 +00:00
root="$(urchin_root "$1")"
if basename "$(fullpath "$root")" |
2016-02-28 19:43:32 +00:00
grep -i 'test' > /dev/null || $force; then
2016-02-28 09:25:21 +00:00
2016-02-28 10:39:32 +00:00
start=$(date +%s)
2016-02-28 12:14:16 +00:00
# 1 test file or folder to run
# 2 urchin root
2016-02-28 14:01:17 +00:00
# 3 Should we cycle shells?
2016-02-28 12:14:16 +00:00
# 4 TEST_SHELL
2016-02-29 01:03:17 +00:00
recurse "$(fullpath "$1")" "$root" "$cycle_shell" "$TEST_SHELL" || :
2016-02-28 10:39:32 +00:00
finish=$(date +%s)
2016-02-28 10:18:06 +00:00
2016-02-28 19:18:54 +00:00
report_outcome "$root" $tap_format $urchin_tmp/log $start $finish
2016-02-28 21:37:28 +00:00
urchin_exit $?
2012-10-11 06:21:05 +00:00
else
2016-02-28 22:56:49 +00:00
echo 'The root directory of the tests that you are running urchin on
2016-02-28 20:57:55 +00:00
does not contain the word "test", so I am not running,
in case that was an accident. Use the -f flag if you really
want to run urchin on that directory.' >&2
exit_code=1
2012-10-08 14:43:14 +00:00
fi
2016-02-28 20:57:55 +00:00
urchin_exit $exit_code