tasty
Modern and extensible testing framework
https://github.com/feuerbach/tasty
Version on this page: | 0.11.3 |
LTS Haskell 23.8: | 1.5.3 |
Stackage Nightly 2025-02-15: | 1.5.3 |
Latest on Hackage: | 1.5.3 |
tasty-0.11.3@sha256:b00d0afdbd957e77f16fc14c2d657e07392874095dc3c28e77047a045d4e912e,2165
Module documentation for 0.11.3
Tasty
Tasty is a modern testing framework for Haskell.
It lets you combine your unit tests, golden tests, QuickCheck/SmallCheck properties, and any other types of tests into a single test suite.
Features:
- Run tests in parallel but report results in a deterministic order
- Filter the tests to be run using patterns specified on the command line
- Hierarchical, colored display of test results
- Reporting of test statistics
- Acquire and release resources (sockets, temporary files etc.) that can be shared among several tests
- Extensibility: add your own test providers and ingredients (runners) above and beyond those provided
To find out what’s new, read the change log.
Ask any tasty-related questions on the mailing list or IRC channel #tasty at FreeNode (logs & stats).
Example
Here’s how your test.hs
might look like:
import Test.Tasty
import Test.Tasty.SmallCheck as SC
import Test.Tasty.QuickCheck as QC
import Test.Tasty.HUnit
import Data.List
import Data.Ord
main = defaultMain tests
tests :: TestTree
tests = testGroup "Tests" [properties, unitTests]
properties :: TestTree
properties = testGroup "Properties" [scProps, qcProps]
scProps = testGroup "(checked by SmallCheck)"
[ SC.testProperty "sort == sort . reverse" $
\list -> sort (list :: [Int]) == sort (reverse list)
, SC.testProperty "Fermat's little theorem" $
\x -> ((x :: Integer)^7 - x) `mod` 7 == 0
-- the following property does not hold
, SC.testProperty "Fermat's last theorem" $
\x y z n ->
(n :: Integer) >= 3 SC.==> x^n + y^n /= (z^n :: Integer)
]
qcProps = testGroup "(checked by QuickCheck)"
[ QC.testProperty "sort == sort . reverse" $
\list -> sort (list :: [Int]) == sort (reverse list)
, QC.testProperty "Fermat's little theorem" $
\x -> ((x :: Integer)^7 - x) `mod` 7 == 0
-- the following property does not hold
, QC.testProperty "Fermat's last theorem" $
\x y z n ->
(n :: Integer) >= 3 QC.==> x^n + y^n /= (z^n :: Integer)
]
unitTests = testGroup "Unit tests"
[ testCase "List comparison (different length)" $
[1, 2, 3] `compare` [1,2] @?= GT
-- the following test does not hold
, testCase "List comparison (same length)" $
[1, 2, 3] `compare` [1,2,2] @?= LT
]
And here is the output of the above program:
(Note that whether QuickCheck finds a counterexample to the third property is determined by chance.)
Packages
tasty is the core package. It contains basic definitions and APIs and a console runner.
In order to create a test suite, you also need to install one or more «providers» (see below).
Providers
The following providers exist:
- tasty-hunit — for unit tests (based on HUnit)
- tasty-golden — for golden tests, which are unit tests whose results are kept in files
- tasty-smallcheck — exhaustive property-based testing (based on smallcheck)
- tasty-quickcheck — for randomized property-based testing (based on QuickCheck)
- tasty-hspec — for Hspec tests
- tasty-program — run external program and test whether it terminates successfully
It’s easy to create custom providers using the API from Test.Tasty.Providers
.
Ingredients
Ingredients represent different actions that you can perform on your test suite. One obvious ingredient that you want to include is one that runs tests and reports the progress and results.
Another standard ingredient is one that simply prints the names of all tests.
It is possible to write custom ingredients using the API from Test.Tasty.Runners
.
Some ingredients that can enhance your test suite are:
- tasty-ant-xml adds a possibility to write the test results in a machine-readable XML format, which is understood by various CI systems and IDEs
- tasty-rerun adds support for minimal test reruns by recording previous test runs and using this information to filter the test tree. For example, you can use this ingredient to only run failed tests, or only run tests that threw an exception.
- tasty-html adds the possibility to write the test results as a HTML file
- tasty-stats adds the possibility to collect statistics of the test suite in a CSV file.
Other packages
- tasty-th automatically discovers tests based on the function names and generate the boilerplate code for you
- tasty-hunit-adapter converts existing HUnit test suites into tasty test suites
- tasty-discover automatically discovers your tests.
- tasty-expected-failure provides test markers for when you expect failures or wish to ignore tests.
Options
Options allow one to customize the run-time behavior of the test suite, such as:
- mode of operation (run tests, list tests, run tests quietly etc.)
- which tests are run (see «Patterns» below)
- parameters of individual providers (like depth of search for SmallCheck)
Setting options
There are two main ways to set options:
Runtime
When using the standard console runner, the options can be passed on the
command line or via environment variables. To see the available options, run
your test suite with the --help
flag. The output will look something like this
(depending on which ingredients and providers the test suite uses):
% ./test --help
Mmm... tasty test suite
Usage: test [-p|--pattern ARG] [-t|--timeout ARG] [-l|--list-tests]
[-j|--num-threads ARG] [-q|--quiet] [--hide-successes] [--color ARG]
[--quickcheck-tests ARG] [--quickcheck-replay ARG]
[--quickcheck-show-replay ARG] [--quickcheck-max-size ARG]
[--quickcheck-max-ratio ARG] [--quickcheck-verbose]
[--smallcheck-depth ARG]
Available options:
-h,--help Show this help text
-p,--pattern ARG Select only tests that match pattern
-t,--timeout ARG Timeout for individual tests (suffixes: ms,s,m,h;
default: s)
-l,--list-tests Do not run the tests; just print their names
-j,--num-threads ARG Number of threads to use for tests execution
-q,--quiet Do not produce any output; indicate success only by
the exit code
--hide-successes Do not print tests that passed successfully
--color ARG When to use colored output. Options are 'never',
'always' and 'auto' (default: 'auto')
--quickcheck-tests ARG Number of test cases for QuickCheck to generate
--quickcheck-replay ARG Replay token to use for replaying a previous test run
--quickcheck-show-replay ARG
Show a replay token for replaying tests
--quickcheck-max-size ARG
Size of the biggest test cases quickcheck generates
--quickcheck-max-ratio ARG
Maximum number of discared tests per successful test
before giving up
--quickcheck-verbose Show the generated test cases
--smallcheck-depth ARG Depth to use for smallcheck tests
Every option can be passed via environment. To obtain the environment variable
name from the option name, replace hyphens -
with underscores _
, capitalize
all letters, and prepend TASTY_
. For example, the environment equivalent of
--smallcheck-depth
is TASTY_SMALLCHECK_DEPTH
. To turn on a switch (such as
TASTY_HIDE_SUCCESSES
), set the variable to True
.
If you’re using a non-console runner, please refer to its documentation to find out how to configure options during the run time.
Compile-time
You can also specify options in the test suite itself, using
localOption
. It can be applied not only to the whole test tree, but also to
individual tests or subgroups, so that different tests can be run with
different options.
It is possible to combine run-time and compile-time options, too, by using
adjustOption
. For example, make the overall testing depth configurable
during the run time, but increase or decrease it slightly for individual
tests.
This method currently doesn’t work for ingredient options, such as --quiet
or
--num-threads
. You can set them by setting the corresponding environment
variable before calling defaultMain
:
import Test.Tasty
import System.Environment
main = do
setEnv "TASTY_NUM_THREADS" "1"
defaultMain _
Patterns
It is possible to restrict the set of executed tests using the --pattern
option. The syntax of patterns is the same as for test-framework, namely:
- An optional prefixed bang
!
negates the pattern. - If the pattern ends with a slash, it is removed for the purpose of
the following description, but it would only find a match with a
test group. In other words,
foo/
will match a group calledfoo
and any tests underneath it, but will not match a regular testfoo
. - If the pattern does not contain a slash
/
, the framework checks for a match against any single component of the path. - Otherwise, the pattern is treated as a glob, where:
- The wildcard
*
matches anything within a single path component (i.e.foo
but notfoo/bar
). - Two wildcards
**
matches anything (i.e.foo
andfoo/bar
). - Anything else matches exactly that text in the path (i.e.
foo
would only match a component of the test path calledfoo
(or a substring of that form).
- The wildcard
For example, group/*1
matches group/test1
but not
group/subgroup/test1
, whereas both examples would be matched by
group/**1
. A leading slash matches the beginning of the test path; for
example, /test*
matches test1
but not group/test1
.
Running tests in parallel
In order to run tests in parallel, you have to do the following:
- Compile (or, more precisely, link) your test program with the
-threaded
flag; - Launch the program with
+RTS -N -RTS
.
Timeout
To apply timeout to individual tests, use the --timeout
(or -t
) command-line
option, or set the option in your test suite using the mkTimeout
function.
Timeouts can be fractional, and can be optionally followed by a suffix ms
(milliseconds), s
(seconds), m
(minutes), or h
(hours). When there’s no
suffix, seconds are assumed.
Example:
./test --timeout=0.5m
sets a 30 seconds timeout for each individual test.
Options controlling console output
The following options control behavior of the standard console interface:
Custom options
It is possible to add custom options, too.
To do that,
- Define a datatype to represent the option, and make it an instance of
IsOption
- Register the options with the
includingOptions
ingredient - To query the option value, use
askOption
.
See the Custom options in Tasty article for some examples.
Project organization and integration with Cabal
There may be several ways to organize your project. What follows is not Tasty’s requirements but my recommendations.
Tests for a library
Place your test suite sources in a dedicated subdirectory (called tests
here) instead of putting them among the main library sources.
The directory structure will be as follows:
my-project/
my-project.cabal
src/
...
tests/
test.hs
Mod1.hs
Mod2.hs
...
test.hs
is where your main
function is defined. The tests may be
contained in test.hs
or spread across multiple modules (Mod1.hs
, Mod2.hs
,
…) which are then imported by test.hs
.
Add the following section to the cabal file (my-project.cabal
):
test-suite test
default-language:
Haskell2010
type:
exitcode-stdio-1.0
hs-source-dirs:
tests
main-is:
test.hs
build-depends:
base >= 4 && < 5
, tasty >= 0.7 -- insert the current version here
, my-project -- depend on the library we're testing
, ...
Tests for a program
All the above applies, except you can’t depend on the library if there’s no library. You have two options:
- Re-organize the project into a library and a program, so that both the program and the test suite depend on this new library. The library can be declared in the same cabal file.
- Add your program sources directory to the
Hs-source-dirs
. Note that this will lead to double compilation (once for the program and once for the test suite).
FAQ
-
How do I make some tests execute after others?
Currently, your only option is to make all tests execute sequentially by setting the number of tasty threads to 1 (example). See #48 for the discussion.
Press
Blog posts and other publications related to tasty. If you wrote or just found something not mentioned here, send a pull request!
- Holy Haskell Project Starter
- First time testing, also with FP Complete (tasty has been added to stackage since then)
- 24 Days of Hackage: tasty
- Resources in Tasty
- Custom options in Tasty
- Resources in Tasty (update)
- Announcing tasty-rerun
- Code testing in Haskell revisited (with Tasty)
Background
Tasty is heavily influenced by test-framework.
The problems with test-framework are:
- Poor code style (some lines of the code wouldn’t even fit in a twitter message!)
- Poor architecture — e.g. relying on laziness for IO and control flow. The
whole story with
:~>
andImprovingIO
is really obscure. - Non-extensible options. For example, when I integrated SmallCheck with
test-framework (in the form of the
test-framework-smallcheck
package), I still had to submit patches to the main package to make SmallCheck depth customizable by the user. - The project is effectively unmaintained.
So I decided to recreate everything that I liked in test-framework from scratch in this package.
Maintainers
Roman Cheplyaka is the primary maintainer.
Oliver Charles is the backup maintainer. Please get in touch with him if the primary maintainer cannot be reached.
Changes
Changes
Version 0.11.3
Expose and document several of the internals of
Test.Tasty.Ingredients.ConsoleReporter
.
Version 0.11.2.5
Fix compatibility with GHC 7.4
Version 0.11.2.4
- Make the
--quiet
mode more efficient on a large number of tests - Fix a bug where a cursor would disappear if the test suite was terminated by a signal other than SIGINT.
Version 0.11.2.3
Make filtering tests (-p
) work faster
Version 0.11.2.2
Fix a critical bug in the quiet mode (-q
/--quiet
):
the exit status could be wrong or the test suite could hang.
Version 0.11.2.1
Fix compatibility with the latest unbounded-delays
Version 0.11.2
Add composeReporters
, a function to run multiple reporter ingredients
Version 0.11.1
Introduce mkOptionCLParser
and mkFlagCLParser
Version 0.11.0.4
Fix compatibility with optparse-applicative-0.13
Version 0.11.0.3
Switch from regex-tdfa-rc
to regex-tdfa
, which got a new maintainer.
Version 0.11.0.2
Clarify IsTest
’s specification with regard to exceptions
Version 0.11.0.1
Use monotonic clock when measuring durations.
Version 0.11
New field resultShortDescription
of Result
Version 0.10.1.2
- Improve the docs
- Fix compatibility with GHC HEAD
Version 0.10.1.1
- Prevent parsing non-positive number of threads via program options (#104)
- Buffer output to avoid slowdowns when printing test results (#101)
- Default to using the maximum number of available cores for test execution
Version 0.10.1
Export Test.Tasty.Runners.formatMessage
Version 0.10.0.4
Don’t output ANSI codes for the Emacs terminal emulator
Version 0.10.0.3
Better handle the situation when there are no ingredients to run
Version 0.10.0.2
Split the changelog into per-project changelogs
Version 0.10.0.1
Update to optparse-applicative 0.11
Version 0.10
- Add the
--color
option - Timings
- Introduce the
Time
type synonym - Change the types of
launchTestTree
andTestReporter
to accept the total run time consoleTestReporter
now displays the timings
- Introduce the
Version 0.9.0.1
Upgrade to optparse-applicative-0.10.
Version 0.8.1.3
Be careful not to export the Show (a -> b)
instance, see
https://github.com/feuerbach/tasty/issues/71
Version 0.8.1.2
Hide cursor when running tests
Version 0.8.1.1
Fix for GHC 7.9
Version 0.8.0.4
Remove the old ‘colors’ flag description from the cabal file
Version 0.8.0.2
Make ansi-terminal an unconditional dependency
Version 0.8
Test.Tasty.Ingredients
is now exposedTest.Tasty.Ingredients.Basic
is added, which exports the ingredients defined in thetasty
package. These exports should now be used instead of ones exported fromTest.Tasty.Runners
- The
Result
type is now structured a bit differently. Providers now should usetestPassed
andtestFailed
functions instead of constructingResult
s directly. - Add «quiet mode» (see README)
- Add «hide successes» mode (see README)
- Add short command-line options:
-j
for--num-threads
,-p
for--pattern
- Add timeout support
AppMonoid
is renamed toTraversal
for consistency with the ‘reducers’ package. Another similar wrapper,Ap
, is introduced.- Fix a resources bug (resources were not released if the test suite was interrupted)
- The type of
launchTestTree
is changed. It now takes a continuation as an argument. This is necessary to fix the bug mentioned above. - Add
flagCLParser
to be used as theoptionCLParser
implementation for boolean options. - Add the ability to pass options via environment
Version 0.7
- Use
regex-tdfa
instead ofregex-posix
(which is a native implementation, and as such is more portable) foldTestTree
now takes the algebra in the form of a record rather than multiple arguments, to minimize breakage when new nodes are added or existing ones changewithResource
now passes the IO action to get the resource to the inner test tree
Version 0.6
- Better handling of exceptions that arise during resource creation or disposal
- Expose the
AppMonoid
wrapper - Add
askOption
andinludingOptions
Version 0.5.2.1
Depend on ansi-terminal >= 0.6.1. This fixes some issues with colors on Windows.
Version 0.5.2
- Export
Result
andProgress
fromTest.Tasty.Runners
- Make it clear that only GHC 7.4+ is supported
Version 0.5.1
Export ResourceSpec
from Test.Tasty.Runners
Version 0.5
Add a capability to acquire and release resources. See the «Resources» section
in the Test.Tasty
docs.
For the end users, the API is backwards-compatible.
Test runners may have to be adjusted — there is a new constructor of TestTree
and a new argument of foldTestTree
.
Version 0.4.2
Add defaultIngredients
Version 0.4.1.1
Print the failure description in red
Version 0.4.0.1
Fix a bug (#25)
Version 0.4
The big change in this release is introduction of ingredients, which is a replacement for runners. But unless you have a custom runner, this is unlikely to affect you much.
The Ingredient
data type has replaced the Runner
type.
The following functions have been renamed and possibly changed their types:
defaultMainWithRunner
→defaultMainWithIngredients
treeOptionParser
→suiteOptionParser
getTreeOptions
→treeOptions
runUI
→consoleTestReporter
Added in this release:
suiteOptions
optionParser
- functions operating on ingredients
testsNames
- the
listingTests
ingredient and its option,ListTests
NumThreads
is no longer a core option, but is automatically included in the
test reporting ingredients (see its haddock).
Version 0.3.1
- Proper reporting of (some) non-terminating tests (#15)
- Upgrade to optparse-applicative 0.6
Version 0.3
- Restrict dependency versions
- Fix a bug where non-terminating test would lead to a deadlock (#15)
Version 0.2
- Add an
execRunner
function - Make
Runner
returnIO Bool
Version 0.1.1
Set lower bound on optparse-applicative dependency version