Compare commits
	
		
			39 Commits
		
	
	
		
			v0.0.2
			...
			exit-on-fa
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 1941bf716a | ||
|  | 9f175f19b3 | ||
|  | 9043430750 | ||
|  | f2d63b2953 | ||
|  | d01e993041 | ||
|  | b5c6464eab | ||
|  | 329fc27929 | ||
|  | 48e5090091 | ||
|  | 5abf088eb8 | ||
|  | c84851cb5c | ||
|  | 510476f924 | ||
|  | f01869fb97 | ||
|  | 0385dcd86e | ||
|  | 2847b020b4 | ||
|  | fae24e926a | ||
|  | 2ab070c353 | ||
|  | 292663dd73 | ||
|  | 90abc2640d | ||
|  | 19ed3b02e8 | ||
|  | b0429315ec | ||
|  | 5e06ffb1c1 | ||
|  | bf8c404784 | ||
|  | bb41d5a156 | ||
|  | ed09351df6 | ||
|  | 1e2d4cc8a3 | ||
|  | f8d921b5e5 | ||
|  | 1f115df222 | ||
|  | eafed6ac59 | ||
|  | 8273661440 | ||
|  | f57e99be6f | ||
|  | afc02582fb | ||
|  | 2f97421164 | ||
|  | 1e9611e732 | ||
|  | deb77cb5de | ||
|  | f12636e029 | ||
|  | 0587a615f3 | ||
|  | b95b010c0d | ||
|  | 98c23a2ab8 | ||
|  | fc711672d6 | 
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1 +0,0 @@ | ||||
| .urchin_stdout | ||||
							
								
								
									
										33
									
								
								HISTORY
									
									
									
									
									
								
							
							
						
						
									
										33
									
								
								HISTORY
									
									
									
									
									
								
							| @@ -1,18 +1,31 @@ | ||||
| History | ||||
| ====== | ||||
| HISTORY | ||||
| ------- | ||||
|  | ||||
| Version 0.0.5 | ||||
| --------------------- | ||||
| * urchin now unsets `CDPATH`. | ||||
| * The documentation for `urchin -x` was removed because it was confusing. | ||||
|  | ||||
| Not done | ||||
| ------ | ||||
| We made the colors brighter. | ||||
| Version 0.0.4 | ||||
| --------------------- | ||||
| * Switch urchin -x to urchin -sh and fix some problems with it | ||||
| * Documentation | ||||
|  | ||||
| We allowed for multiple tests to be written in a file | ||||
| Version 0.0.3 | ||||
| --------------------- | ||||
| General tidying | ||||
|  | ||||
| We added flags for verbosity and for Molly guards | ||||
| Run with different shells in three ways | ||||
|  | ||||
| Done | ||||
| ----- | ||||
| * urchin -s | ||||
| * $TEST_SHELL variable with $TEST_SHELL | ||||
| * $TEST_SHELL variable with urchin -sh | ||||
|  | ||||
| We adjusted the input parameters so it is harder to accidentally run all executible files in your home directory. | ||||
| Set NULL_GLOB so zsh doesn't print a warning. | ||||
|  | ||||
| Before version 0.0.3 | ||||
| ---------------------- | ||||
| We adjusted the input parameters so it is harder to accidentally run all executable files in you | ||||
| r home directory. | ||||
|  | ||||
| We added directory-based indents. | ||||
|   | ||||
							
								
								
									
										2
									
								
								alternatives
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								alternatives
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| Totally different syntax and similar features, plus TAP output | ||||
| https://github.com/sstephenson/bats | ||||
| @@ -6,7 +6,7 @@ for shell in dash bash ksh zsh; do | ||||
|   if which $shell > /dev/null 2> /dev/null; then | ||||
|     echo | ||||
|     echo Running urchin tests in $shell | ||||
|     $shell urchin tests | tail -n 2 | ||||
|     $shell urchin -s $shell tests | tail -n 3 | ||||
|   else | ||||
|     echo | ||||
|     echo Skipping $shell because it is not in the PATH | ||||
|   | ||||
							
								
								
									
										20
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										20
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,26 +1,32 @@ | ||||
| { | ||||
|   "name": "urchin", | ||||
|   "version": "0.0.2", | ||||
|   "version": "0.0.5", | ||||
|   "description": "Test framework for shell", | ||||
|   "main": "urchin", | ||||
|   "directories": { | ||||
|     "test": "tests" | ||||
|   }, | ||||
|   "scripts": { | ||||
|     "test": "./urchin tests" | ||||
|     "test": "export PATH=\"$PWD:$PATH\" && urchin tests" | ||||
|   }, | ||||
|   "bin": "./urchin", | ||||
|   "repository": { | ||||
|     "type": "git", | ||||
|     "url": "git://github.com/scraperwiki/urchin.git" | ||||
|     "url": "git://github.com/tlevine/urchin.git" | ||||
|   }, | ||||
|   "keywords": [ | ||||
|     "shell", | ||||
|     "test", | ||||
|     "testing" | ||||
|   ], | ||||
|   "author": "ScraperWiki Limited <feedback@scraperwiki.com>", | ||||
|   "license": "BSD", | ||||
|   "readmeFilename": "readme.md", | ||||
|   "gitHead": "015052dcfd7152a0df037bef1bc5452e0c956695" | ||||
|   "author": "Thomas Levine <_@thomaslevine.com>", | ||||
|   "contributors": [ | ||||
|     {"name": "Thomas Levine", "email": "_@thomaslevine.com"}, | ||||
|     {"name": "David Jones", "email": "drj@pobox.com"}, | ||||
|     {"name": "Francis Irving", "email": "francis@flourish.org"}, | ||||
|     {"name": "Zarino Zappia", "email": "mail@zarino.co.uk"}, | ||||
|     {"name": "Tom Mortimer-Jones", "email": "tom@morty.co.uk"}, | ||||
|     {"name": "Michael Klement", "email": "mklement0@gmail.com"} | ||||
|   ], | ||||
|   "license": "BSD" | ||||
| } | ||||
|   | ||||
							
								
								
									
										103
									
								
								readme.md
									
									
									
									
									
								
							
							
						
						
									
										103
									
								
								readme.md
									
									
									
									
									
								
							| @@ -12,16 +12,16 @@ other Unix platforms. | ||||
| Urchin's tests are written in Urchin, so you can run them to see what Urchin | ||||
| is like. Clone the repository | ||||
|  | ||||
|     git clone git://github.com/scraperwiki/urchin.git | ||||
|     git clone git://github.com/tlevine/urchin.git | ||||
|  | ||||
| Run the tests | ||||
|  | ||||
|     cd urchin | ||||
|     ./urchin tests | ||||
|  | ||||
| The above command will run the tests in your systems default | ||||
| The above command will run the tests in your system's default | ||||
| shell, /bin/sh (on recent Ubuntu this is dash, but it could be | ||||
| ksh or bash on other systems); to test cross-shell compatibility, | ||||
| ksh or bash on other systems); to test urchin's cross-shell compatibility, | ||||
| run this: | ||||
|  | ||||
|     cd urchin | ||||
| @@ -31,7 +31,7 @@ run this: | ||||
| Download Urchin like so (as root) (or use npm, below): | ||||
|  | ||||
|     cd /usr/local/bin | ||||
|     wget https://raw.github.com/scraperwiki/urchin/master/urchin | ||||
|     wget https://raw.github.com/tlevine/urchin/master/urchin | ||||
|     chmod +x urchin | ||||
|  | ||||
| Can be installed with npm too: | ||||
| @@ -42,6 +42,8 @@ Now you can run it. | ||||
|  | ||||
|     urchin <test directory> | ||||
|  | ||||
| Run `urchin -h` to get command-line help. | ||||
|  | ||||
| ## Writing tests | ||||
| Make a root directory for your tests. Inside it, put executable files that | ||||
| exit `0` on success and something else on fail. Non-executable files and hidden | ||||
| @@ -83,3 +85,96 @@ within the particular directory, and the `teardown` file is run right after. | ||||
| Files are only run if they are executable, and files beginning with `.` are | ||||
| ignored. Thus, fixtures and libraries can be included sloppily within the test | ||||
| directory tree. The test passes if the file exits 0; otherwise, it fails. | ||||
|  | ||||
| In case you care about the order in which your tests execute, consider that | ||||
| urchin looks for files within a directory in the following manner. | ||||
|  | ||||
|     for file in *; do | ||||
|       do_something_with_test_file $file | ||||
|     done | ||||
|  | ||||
| Tests within a directory are executed in whatever order `*` returns. | ||||
|  | ||||
| ### Writing cross-shell compatibility tests for testing shell code | ||||
|  | ||||
| While you could write your test scripts to explicitly invoke the functionality | ||||
| to test with various shells, Urchin facilitates a more flexible approach. | ||||
|  | ||||
| The specific approach depends on your test scenario: | ||||
|  | ||||
| * (a) Your test scripts _invoke_ scripts containing portable shell code. | ||||
| * (b) Your scripts _source_ scripts containing portable shell code. | ||||
|  | ||||
| #### (a) Cross-shell tests with test scripts that _invoke_ shell scripts | ||||
|  | ||||
| First, consider using [shall](https://github.com/mklement0/shall). | ||||
|  | ||||
|     #!/usr/bin/env shall | ||||
|     echo This is a test file. | ||||
|  | ||||
| Alternatively, you can use urchin's built-in recognition of the | ||||
| `TEST_SHELL` environment variable. | ||||
| In your test scripts, invoke the shell scripts to test via the shell | ||||
| specified in environment variable `TEST_SHELL` rather than directly; | ||||
| e.g.: `$TEST_SHELL ../foo bar` (rather than just `../foo bar`).   | ||||
| Note that if you alsow want your test scripts to work when run directly, | ||||
| outside of Urchin, be sure to target scripts that happen to be in the  | ||||
| current directory with prefix `./`; e.g., `$TEST_SHELL ./baz` | ||||
| (rather than `$TEST_SHELL baz`). | ||||
|  | ||||
| Then, on invocation of Urchin, prepend a definition of environment variable | ||||
| `TEST_SHELL` specifying the shell to test with, e.g.: `TEST_SHELL=zsh urchin ./tests`.   | ||||
| To test with multiple shells in sequence, use something like: | ||||
|  | ||||
|     for shell in sh bash ksh zsh; do | ||||
|       TEST_SHELL=$shell urchin ./tests | ||||
|     done | ||||
|  | ||||
| If `TEST_SHELL` has no value, Urchin defines it as `/bin/sh`, so the test | ||||
| scripts can rely on `$TEST_SHELL` always containing a value. | ||||
|  | ||||
| #### (b) Cross-shell tests with test scripts that _source_ shell scripts | ||||
|  | ||||
| If you _source_ shell code in your test scripts, it is the test scripts | ||||
| themselves that must be run with the shell specified. | ||||
|  | ||||
| To that end, Urchin supports the `-s <shell>` option, which instructs | ||||
| Urchin to invoke the test scripts with the specified shell; e.g., `-s bash`.   | ||||
| (In addition, Urchin sets environment variable `TEST_SHELL` to the specified | ||||
| shell.) | ||||
|  | ||||
| Note that only test scripts that either have no shebang line at all or | ||||
| have shebang line `#!/bin/sh` are invoked with the specified shell. | ||||
| This allows non-shell test scripts or test scripts for specific | ||||
| shells to coexist with those whose invocation should be controlled by `-s`. | ||||
|  | ||||
| To test with multiple shells in sequence, use something like: | ||||
|  | ||||
|     for shell in sh bash ksh zsh; do | ||||
|       urchin -s $shell ./tests | ||||
|     done | ||||
|  | ||||
| <!-- | ||||
| #### (c) Cross shell tests with `urchin -x` (experimental) | ||||
| If you run urchin with the `-x` flag, it will be as if you ran | ||||
| `$TEST_SHELL`. Unless `$TEST_SHELL` isn't set, in which case it'll | ||||
| be as if you ran `/bin/sh`. Putting this in she shebang line might | ||||
| eventually work out to be a cleaner way of doing cross-shell testing. | ||||
|  | ||||
|     #!/usr/bin/env urchin -x | ||||
|     test a = a | ||||
|  | ||||
| It might make sense if you do this. | ||||
|  | ||||
|     export TEST_SHELL=zsh && urchin -x | ||||
|     export TEST_SHELL=bash && urchin -x | ||||
| --> | ||||
| ## Alternatives to Urchin | ||||
| Alternatives to Urchin are discussed in | ||||
| [this blog post](https://blog.scraperwiki.com/2012/12/how-to-test-shell-scripts/). | ||||
|  | ||||
| ## Ideas for new features | ||||
|  | ||||
| * Support [Nagios plugins](https://nagios-plugins.org/doc/guidelines.html) | ||||
| * Stop running if a test fails so one can use Urchin as a | ||||
|     [setup framework](https://github.com/tlevine/urchin/issues/16). | ||||
|   | ||||
							
								
								
									
										1
									
								
								tests/.die-on-fail/2 should run.
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										1
									
								
								tests/.die-on-fail/2 should run.
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1 @@ | ||||
| false | ||||
							
								
								
									
										1
									
								
								tests/.die-on-fail/3 should not run.
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										1
									
								
								tests/.die-on-fail/3 should not run.
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1 @@ | ||||
| false | ||||
							
								
								
									
										1
									
								
								tests/.die-on-fail/4 should not run.
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										1
									
								
								tests/.die-on-fail/4 should not run.
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1 @@ | ||||
| false | ||||
							
								
								
									
										1
									
								
								tests/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								tests/.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1 +1,2 @@ | ||||
| .urchin.log | ||||
| .urchin_stdout | ||||
|   | ||||
							
								
								
									
										1
									
								
								tests/.print-arg-3
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								tests/.print-arg-3
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| echo $3 | ||||
							
								
								
									
										6
									
								
								tests/A nonempty CDPATH should not break urchin.
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										6
									
								
								tests/A nonempty CDPATH should not break urchin.
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,6 @@ | ||||
| #!/bin/sh | ||||
|  | ||||
| cd .. | ||||
| export CDPATH=$PWD | ||||
| ./urchin -f 'tests/urchin exit code' >/dev/null | ||||
|  | ||||
							
								
								
									
										3
									
								
								tests/Command-line help contents/The -s option should be documented.
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										3
									
								
								tests/Command-line help contents/The -s option should be documented.
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| #!/bin/sh | ||||
|  | ||||
| ../../urchin -h | grep -- -s | ||||
| @@ -1,3 +0,0 @@ | ||||
| #!/bin/sh | ||||
|  | ||||
| ../../urchin -h|grep -i xsd | ||||
| @@ -0,0 +1,7 @@ | ||||
| #!/bin/sh | ||||
|  | ||||
| # Assuming that urchin was invoked with `TEST_SHELL=bash urchin ...`, $TEST_SHELL should contain 'bash'. | ||||
|  | ||||
| echo "\$TEST_SHELL: $TEST_SHELL" | ||||
|  | ||||
| [ "$TEST_SHELL" = 'bash' ] | ||||
| @@ -0,0 +1,4 @@ | ||||
| #!/bin/sh | ||||
|  | ||||
| # Invoke a simple test command with $TEST_SHELL as the executable. | ||||
| [ "$($TEST_SHELL -c 'echo $0')" = "$TEST_SHELL" ] | ||||
| @@ -0,0 +1,5 @@ | ||||
| #!/usr/bin/awk -f | ||||
|  | ||||
| # This script will only succeed if it is indeed processed by awk. | ||||
|  | ||||
| BEGIN { print "ok" } | ||||
| @@ -0,0 +1,10 @@ | ||||
| #!/bin/sh | ||||
|  | ||||
| # Assuming that urchin was invoked with `-s bash`, this script should be being run with bash. | ||||
|  | ||||
| this_shell=$(ps -o comm= -p $$ && :) | ||||
|  | ||||
| echo "Running shell: $this_shell" | ||||
|  | ||||
| [  "$this_shell" = 'bash' ] | ||||
|  | ||||
| @@ -0,0 +1,10 @@ | ||||
| # By design, this file has no shebang line. | ||||
|  | ||||
| # Assuming that urchin was invoked with `-s bash`, this script should be being run with bash. | ||||
|  | ||||
| this_shell=$(ps -o comm= -p $$ && :) | ||||
|  | ||||
| echo "Running shell: $this_shell" | ||||
|  | ||||
| [  "$this_shell" = 'bash' ] | ||||
|  | ||||
| @@ -0,0 +1,9 @@ | ||||
| #!/bin/sh | ||||
|  | ||||
| # Assuming that urchin was invoked with `-s bash`, $TEST_SHELL should contain 'bash'. | ||||
|  | ||||
| echo "Running shell: $(ps -o comm= -p $$ && :)" | ||||
|  | ||||
| echo "\$TEST_SHELL: $TEST_SHELL" | ||||
|  | ||||
| [  "$TEST_SHELL" = 'bash' ] | ||||
| @@ -0,0 +1,9 @@ | ||||
| #!/bin/sh | ||||
|  | ||||
| # Tests support for either passing through or defining a default value for environment variable TEST_SHELL. | ||||
| # (for test scripts that want to invoke shell scripts with a specified shell). | ||||
|  | ||||
| which bash 2>/dev/null || { echo "Cannot test -s option: bash cannot be located." >&2; exit 1; } | ||||
|  | ||||
| # Test if $TEST_SHELL, when placed in urchin's environment, is passed through to the test scripts. | ||||
| TEST_SHELL=bash ../../urchin ./.test-TEST_SHELL-passed-through | ||||
| @@ -0,0 +1,8 @@ | ||||
| #!/bin/sh | ||||
|  | ||||
| # Tests support for either passing through or defining a default value for environment variable TEST_SHELL. | ||||
| # (for test scripts that want to invoke shell scripts with a specified shell). | ||||
|  | ||||
| # Test if $TEST_SHELL - if *defined, but empty* - is exported with value '/bin/sh' by urchin | ||||
| # and thus has that value inside the scripts. | ||||
| TEST_SHELL= ../../urchin ./.test-TEST_SHELL-undefined_or_empty | ||||
| @@ -0,0 +1,9 @@ | ||||
| #!/bin/sh | ||||
|  | ||||
| # Tests support for either passing through or defining a default value for environment variable TEST_SHELL. | ||||
| # (for test scripts that want to invoke shell scripts with a specified shell). | ||||
|  | ||||
| # Test if $TEST_SHELL - if *undefined* - is exported with value '/bin/sh' by urchin | ||||
| # and thus has that value inside test scripts. | ||||
| unset -v TEST_SHELL | ||||
| ../../urchin ./.test-TEST_SHELL-undefined_or_empty | ||||
| @@ -0,0 +1,8 @@ | ||||
| #!/bin/sh | ||||
|  | ||||
| # Tests the `-s <shell> option, which invokes shebang-less and sh-shebang-line test scripts with the specified shell (for testing *sourced* shell code). | ||||
|  | ||||
| which bash >/dev/null || { echo "Cannot test -s option: bash cannot be located." >&2; exit 2; } | ||||
| which /usr/bin/awk >/dev/null || { echo "Cannot test -s option: /usr/bin/awk not found." >&2; exit 2; } | ||||
|  | ||||
| ../../urchin -s bash ./.test-run-by-specified-shell | ||||
							
								
								
									
										11
									
								
								tests/Die on fail.
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										11
									
								
								tests/Die on fail.
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| tmp=$(mktemp) | ||||
| ../urchin -e -f ./.die-on-fail > $tmp | ||||
| result=$? | ||||
|  | ||||
| grep '1 should run.' $tmp | ||||
| grep '2 should run.' $tmp | ||||
| grep -v '3 should not run.' $tmp | ||||
| grep -v '4 should not run.' $tmp | ||||
|  | ||||
| rm $tmp | ||||
| exit $result | ||||
| @@ -1,4 +1,3 @@ | ||||
| #!/bin/sh | ||||
|  | ||||
| mkdir /tmp/urchintmp | ||||
| ! ../../urchin /tmp/urchintmp | ||||
| ! ../../urchin ./.chainsaw | ||||
|   | ||||
| @@ -1,3 +0,0 @@ | ||||
| #!/bin/sh | ||||
|  | ||||
| exit 1 | ||||
| @@ -1,3 +0,0 @@ | ||||
| #!/bin/sh | ||||
|  | ||||
| exit | ||||
| @@ -1,3 +0,0 @@ | ||||
| #!/bin/sh | ||||
|  | ||||
| ../../urchin .fixture | grep 'http://www.w3.org/2001/XMLSchema' | ||||
| @@ -1,3 +0,0 @@ | ||||
| #!/bin/sh | ||||
|  | ||||
| export variable_set_in_setup='value' | ||||
							
								
								
									
										0
									
								
								tests/urchin exit code/.test/This test passes
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										0
									
								
								tests/urchin exit code/.test/This test passes
									
									
									
									
									
										Executable file
									
								
							
							
								
								
									
										13
									
								
								tests/urchin exit code/non zero exit code when a test fails
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										13
									
								
								tests/urchin exit code/non zero exit code when a test fails
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| #!/bin/sh | ||||
|  | ||||
| # Borrow this test which has a failure in it (3 in fact), and | ||||
| # run it using urchin: | ||||
| ../../urchin "../Counts should be kept of successes and failures./.test" > /dev/null | ||||
|  | ||||
| # Now invert the result, we want _this_ test to fail if the exit | ||||
| # code was 0. | ||||
| case $? in | ||||
|   0) exit 8;; | ||||
|   *) exit 0;; | ||||
| esac | ||||
|  | ||||
							
								
								
									
										1
									
								
								tests/urchin exit code/zero exit code when tests all pass
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										1
									
								
								tests/urchin exit code/zero exit code when tests all pass
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1 @@ | ||||
| ../../urchin .test > /dev/null | ||||
							
								
								
									
										147
									
								
								urchin
									
									
									
									
									
								
							
							
						
						
									
										147
									
								
								urchin
									
									
									
									
									
								
							| @@ -1,5 +1,9 @@ | ||||
| #!/bin/sh | ||||
|  | ||||
| # Make sure that CDPATH isn't set, as it causes `cd` to behave unpredictably - notably, it can produce output, | ||||
| # which breaks fullpath(). | ||||
| unset CDPATH | ||||
|  | ||||
| fullpath() { | ||||
|   ( | ||||
|     cd -- "$1" | ||||
| @@ -15,13 +19,14 @@ indent() { | ||||
| recurse() { | ||||
|   potential_test="$1" | ||||
|   indent_level="$2" | ||||
|   shell_for_sh_tests="$3" | ||||
|  | ||||
|   [ "$potential_test" = 'setup_dir' ] && return | ||||
|   [ "$potential_test" = 'teardown_dir' ] && return | ||||
|   [ "$potential_test" = 'setup' ] && return | ||||
|   [ "$potential_test" = 'teardown' ] && return | ||||
|  | ||||
|   echo > "$stdout_file" | ||||
|   [ $indent_level -eq 0 ] && : > "$stdout_file" | ||||
|  | ||||
|   if [ -d "$potential_test" ] | ||||
|     then | ||||
| @@ -30,12 +35,25 @@ recurse() { | ||||
|       echo "  ${potential_test}" | ||||
|       cd -- "$potential_test" | ||||
|       [ -f setup_dir ] && [ -x setup_dir ] && ./setup_dir >> "$stdout_file" | ||||
|  | ||||
|       if [ -n "$ZSH_VERSION" ]; then | ||||
|         # avoid "no matches found: *" error when directories are empty | ||||
|         setopt NULL_GLOB | ||||
|       fi | ||||
|  | ||||
|       for test in * | ||||
|         do | ||||
|         [ -f setup ] && [ -x setup ] && ./setup >> "$stdout_file" | ||||
|  | ||||
|         # $2 instead of $indent_level so it doesn't clash | ||||
|         recurse "${test}" $(( $2 + 1 )) | ||||
|         recurse "${test}" $(( $2 + 1 )) "$shell_for_sh_tests" | ||||
|         exit_code=$? | ||||
|  | ||||
|         if $exit_on_fail && test $exit_code -ne 0; then | ||||
|           [ -f teardown ] && [ -x teardown ] && ./teardown >> "$stdout_file" | ||||
|           [ -f teardown_dir ] && [ -x teardown_dir ] && ./teardown_dir >> "$stdout_file" | ||||
|           return 1 | ||||
|         fi | ||||
|  | ||||
|         [ -f teardown ] && [ -x teardown ] && ./teardown >> "$stdout_file" | ||||
|       done | ||||
| @@ -48,15 +66,20 @@ recurse() { | ||||
|     [ -f setup ] && [ -x setup ] && ./setup >> "$stdout_file" | ||||
|  | ||||
|     # Run the test | ||||
|     ./"$potential_test" > "$stdout_file" 2>&1 | ||||
|     if [ -n "$shell_for_sh_tests" ] && has_sh_or_no_shebang_line ./"$potential_test" | ||||
|       then     | ||||
|       TEST_SHELL="$TEST_SHELL" "$shell_for_sh_tests" ./"$potential_test" > "$stdout_file" 2>&1 | ||||
|     else | ||||
|       TEST_SHELL="$TEST_SHELL" ./"$potential_test" > "$stdout_file" 2>&1 | ||||
|     fi | ||||
|     exit_code="$?" | ||||
|  | ||||
|     [ -f teardown ] && [ -x teardown ] && ./teardown >> "$stdout_file" | ||||
|  | ||||
|     indent $indent_level | ||||
|     if [ "$exit_code" = '0' ] | ||||
|     if [ $exit_code -eq 0 ] | ||||
|       then | ||||
|       # On success, print a '✓' | ||||
|       # On success, print a green '✓' | ||||
|       printf '\033[32m✓ \033[0m' | ||||
|       printf '%s\n' "${potential_test}" | ||||
|       printf '%s\n' "${potential_test} passed" >> "$logfile" | ||||
| @@ -65,25 +88,43 @@ recurse() { | ||||
|       printf '\033[31m✗ \033[0m' | ||||
|       printf '%s\n' "${potential_test}" | ||||
|       printf '%s\n' "${potential_test} failed" >> "$logfile" | ||||
|       printf '\033[31m' # Print output captured from failed test in red. | ||||
|       cat "$stdout_file" | ||||
|       printf '\033[0m' | ||||
|     fi | ||||
|  | ||||
|     if $exit_on_fail && test 0 -ne $exit_code; then | ||||
|       return 1 | ||||
|     fi | ||||
|     rm "$stdout_file" | ||||
|   fi | ||||
|   [ $indent_level -eq 0 ] && rm "$stdout_file" | ||||
| } | ||||
|  | ||||
| USAGE="usage: $0 <test directory>" | ||||
| has_sh_or_no_shebang_line() { | ||||
|   head -n 1 "$1" | grep -vqE '^#!' && return 0 # no shebang line at all | ||||
|   head -n 1 "$1" | grep -qE '^#![[:blank:]]*/bin/sh($|[[:blank:]])' && return 0  # shebang line is '#!/bin/sh' or legal variations thereof | ||||
|   return 1 | ||||
| } | ||||
|  | ||||
| USAGE="usage: $0 [<options>] <test directory>" | ||||
|  | ||||
| urchin_help() { | ||||
|   echo | ||||
|   echo "$USAGE" | ||||
|   echo | ||||
|   echo '-f      Force urchin to run on directories not named "test".' | ||||
|   echo '-h      This help' | ||||
|   echo | ||||
|   echo '--xsd   Output xUnit XML schema for an integration server.' | ||||
|   echo | ||||
|   echo 'Go to http://www.urchin.sh for documentation on writing tests.' | ||||
|   echo | ||||
|   cat <<EOF | ||||
|  | ||||
| $USAGE | ||||
|  | ||||
| -s <shell>  Invoke test scripts that either have no shebang line at all or | ||||
|             have shebang line "#!/bin/sh" with the specified shell. | ||||
| -e          Stop running if any single test fails. This is helpful if you want | ||||
|             to use Urchin to run things other than tests, such as a set of | ||||
|             configuration scripts. | ||||
| -f          Force running even if the test directory's name does not | ||||
|             contain the word "test". | ||||
| -h          This help. | ||||
|  | ||||
| Go to https://github.com/tlevine/urchin for documentation on writing tests. | ||||
|  | ||||
| EOF | ||||
| } | ||||
|  | ||||
| plural () { | ||||
| @@ -99,61 +140,85 @@ plural () { | ||||
| } | ||||
|  | ||||
| urchin_go() { | ||||
|   echo Running tests at $(date +%Y-%m-%dT%H:%M:%S) | ||||
|   echo Running tests at $(date +%Y-%m-%dT%H:%M:%S) | tee "$logfile" | ||||
|   start=$(date +%s) | ||||
|  | ||||
|   echo > "$logfile" | ||||
|   recurse "$1" 0 | ||||
|   # Determine the environment variable to define for test scripts | ||||
|   # that reflects the specified or implied shell to use for shell-code tests. | ||||
|   #  - Set it to the shell specified via -s, if any. | ||||
|   #  - Otherwise, use its present value, if non-empty. | ||||
|   #  - Otherwise, default to '/bin/sh'. | ||||
|   if [ -n "$2" ] | ||||
|     then | ||||
|     TEST_SHELL="$2" | ||||
|   elif [ -z "$TEST_SHELL" ] | ||||
|     then | ||||
|     TEST_SHELL='/bin/sh' | ||||
|   fi | ||||
|  | ||||
|   recurse "$1" 0 "$2"  # test folder -- indentation level -- [shell to invoke test scripts with] | ||||
|  | ||||
|   finish=$(date +%s) | ||||
|   elapsed=$(($finish - $start)) | ||||
|   echo Done, took $elapsed $(plural second $elapsed) | ||||
|   echo "Done, took $elapsed $(plural second $elapsed)." | ||||
|   set -- $(grep -e 'passed$' "$logfile"|wc -l) $(grep -e 'failed$' "$logfile"|wc -l) | ||||
|   printf '%s\n' "$1 $(plural test "$1") passed." | ||||
|   [ $2 -gt 0 ] && printf '\033[31m' || printf '\033[32m' # If tests failed, print the message in red, otherwise in green. | ||||
|   printf '%s\n' "$2 $(plural test "$2") failed." | ||||
|   printf '\033[m' | ||||
|   return "$2" | ||||
| } | ||||
|  | ||||
| urchin_molly_guard() { | ||||
|   echo | ||||
|   echo 'The directory on which you are running urchin is not' | ||||
|   echo 'called "test", so I am not running in case that' | ||||
|   echo 'was an accident. Use the -f flag if you really want' | ||||
|   echo 'to run urchin on that directory.' | ||||
|   echo | ||||
|   { | ||||
|     echo | ||||
|     echo 'The name of the directory on which you are running urchin' | ||||
|     echo 'does not contain the word "test", so I am not running,' | ||||
|     echo 'in case that was an accident. Use the -f flag if you really' | ||||
|     echo 'want to run urchin on that directory.' | ||||
|     echo | ||||
|   } >&2 | ||||
|   exit 1 | ||||
| } | ||||
|  | ||||
| FORCE=false | ||||
| shell_for_sh_tests= | ||||
| force=false | ||||
| exit_on_fail=false | ||||
| while [ $# -gt 0 ] | ||||
| do | ||||
|     case "$1" in | ||||
|         -f) FORCE=true;; | ||||
|         -e) exit_on_fail=true;; | ||||
|         -f) force=true;; | ||||
|         -s) | ||||
|           shift | ||||
|           shell_for_sh_tests=$1 | ||||
|           which "$shell_for_sh_tests" >/dev/null || { echo "Cannot find specified shell: '$shell_for_sh_tests'" >&2; urchin_help >&2; exit 2; } | ||||
|           ;; | ||||
|         -h|--help) urchin_help | ||||
|           exit 0;; | ||||
| #       --xsd) action=testsuite;; | ||||
| #       --) shift; break;; | ||||
|         -*) urchin_help 1>&2 | ||||
|         -*) urchin_help >&2 | ||||
|             exit 1;; | ||||
|         *)  break;; | ||||
|     esac | ||||
|     shift | ||||
| done | ||||
|  | ||||
| # Verify argument for main stuff | ||||
| if [ "$#" != '1' ] || [ ! -d "$1" ] | ||||
|   then | ||||
|   [ -n "$1" ] && [ ! -d "$1" ] && echo "Not a directory: '$1'" >&2 | ||||
|   echo "$USAGE" >&2 | ||||
|   exit 2 | ||||
| fi | ||||
|  | ||||
| # Constants | ||||
| logfile=$(fullpath "$1")/.urchin.log | ||||
| stdout_file=$(fullpath "$1")/.urchin_stdout | ||||
|  | ||||
| # Verify argument for main stuff | ||||
| if [ "$#" != '1' ] && [ ! -d "$1" ] | ||||
|   then | ||||
|   echo "$USAGE" | ||||
|   exit 1 | ||||
| fi | ||||
|  | ||||
| # Run or present the Molly guard. | ||||
| if basename "$(fullpath "$1")" | grep test > /dev/null || $FORCE | ||||
| if basename "$(fullpath "$1")" | grep -Fi 'test' > /dev/null || $force | ||||
|   then | ||||
|   urchin_go "$1" | ||||
|   urchin_go "$1" "$shell_for_sh_tests" | ||||
| else | ||||
|   urchin_molly_guard | ||||
| fi | ||||
|   | ||||
		Reference in New Issue
	
	Block a user