2016-10-20

test provides a standard way of writing and running tests in Dart.

Writing Tests

Running Tests

Restricting Tests to Certain Platforms

Platform Selectors

Running Tests on Dartium

Asynchronous Tests

Running Tests With Custom HTML

Configuring Tests

Skipping Tests

Timeouts

Platform-Specific Configuration

Whole-Package Configuration

Tagging Tests

Debugging

Testing with barback

Further Reading

Writing Tests

Tests are specified using the top-level test() function, and test
assertions are made using expect():

Tests can be grouped together using the group() function. Each
group's description is added to the beginning of its test's descriptions.

Any matchers from the matcher package can be used with expect()
to do complex validations:

You can use the setUp() and tearDown() functions to
share code between tests. The setUp() callback will run before every test in a
group or test suite, and tearDown() will run after. tearDown() will run even
if a test fails, to ensure that it has a chance to clean up after itself.

Running Tests

A single test file can be run just using pub run test path/to/test.dart.



Many tests can be run at a time using pub run test path/to/dir.



It's also possible to run a test on the Dart VM only by invoking it using dart
path/to/test.dart, but this doesn't load the full test runner and will be
missing some features.

The test runner considers any file that ends with _test.dart to be a test
file. If you don't pass any paths, it will run all the test files in your
test/ directory, making it easy to test your entire application at once.

You can select specific tests cases to run by name using pub run test -n "test
name". The string is interpreted as a regular expression, and only tests whose
description (including any group descriptions) match that regular expression
will be run. You can also use the -N flag to run tests whose names contain a
plain-text string.

By default, tests are run in the Dart VM, but you can run them in the browser as
well by passing pub run test -p chrome path/to/test.dart. test will take
care of starting the browser and loading the tests, and all the results will be
reported on the command line just like for VM tests. In fact, you can even run
tests on both platforms with a single command: pub run test -p "chrome,vm"
path/to/test.dart.

Restricting Tests to Certain Platforms

Some test files only make sense to run on particular platforms. They may use
dart:html or dart:io, they might test Windows' particular filesystem
behavior, or they might use a feature that's only available in Chrome. The
@TestOn annotation makes it easy to declare exactly which platforms
a test file should run on. Just put it at the top of your file, before any
library or import declarations:

The string you pass to @TestOn is what's called a "platform selector", and it
specifies exactly which platforms a test can run on. It can be as simple as the
name of a platform, or a more complex Dart-like boolean expression involving
these platform names.

You can also declare that your entire package only works on certain platforms by
adding a test_on field to your package config file.

Platform Selectors

Platform selectors use the boolean selector syntax defined in the
boolean_selector package, which is a subset of Dart's
expression syntax that only supports boolean operations. The following
identifiers are defined:

vm: Whether the test is running on the command-line Dart VM.

dartium: Whether the test is running on Dartium.

content-shell: Whether the test is running on the headless Dartium content
shell.

chrome: Whether the test is running on Google Chrome.

phantomjs: Whether the test is running on
PhantomJS.

firefox: Whether the test is running on Mozilla Firefox.

safari: Whether the test is running on Apple Safari.

ie: Whether the test is running on Microsoft Internet Explorer.

dart-vm: Whether the test is running on the Dart VM in any context,
including Dartium. It's identical to !js.

browser: Whether the test is running in any browser.

js: Whether the test has been compiled to JS. This is identical to
!dart-vm.

blink: Whether the test is running in a browser that uses the Blink
rendering engine.

windows: Whether the test is running on Windows. If vm is false, this will
be false as well.

mac-os: Whether the test is running on Mac OS. If vm is false, this will
be false as well.

linux: Whether the test is running on Linux. If vm is false, this will be
false as well.

android: Whether the test is running on Android. If vm is false, this will
be false as well, which means that this won't be true if the test is
running on an Android browser.

ios: Whether the test is running on iOS. If vm is false, this will be
false as well, which means that this won't be true if the test is running
on an iOS browser.

posix: Whether the test is running on a POSIX operating system. This is
equivalent to !windows.

For example, if you wanted to run a test on every browser but Chrome, you would
write @TestOn("browser && !chrome").

Running Tests on Dartium

Tests can be run on Dartium by passing the -p dartium flag. If you're
using Mac OS, you can install Dartium using Homebrew. Otherwise,
make sure there's an executable called dartium (on Mac OS or Linux) or
dartium.exe (on Windows) on your system path.

Similarly, tests can be run on the headless Dartium content shell by passing -p
content-shell. The content shell is installed along with Dartium when using
Homebrew. Otherwise, you can downloaded it manually from this
page; if you do, make sure the executable named content_shell
(on Mac OS or Linux) or content_shell.exe (on Windows) is on your system path.

In the future, there will be a more explicit way to configure the
location of both the Dartium and content shell executables.

Asynchronous Tests

Tests written with async/await will work automatically. The test runner
won't consider the test finished until the returned Future completes.

There are also a number of useful functions and matchers for more advanced
asynchrony. The completion() matcher can be used to test
Futures; it ensures that the test doesn't finish until the Future completes,
and runs a matcher against that Future's value.

The throwsA() matcher and the various throwsExceptionType
matchers work with both synchronous callbacks and asynchronous Futures. They
ensure that a particular type of exception is thrown:

The expectAsync() function wraps another function and has two
jobs. First, it asserts that the wrapped function is called a certain number of
times, and will cause the test to fail if it's called too often; second, it
keeps the test from finishing until the function is called the requisite number
of times.

Running Tests With Custom HTML

By default, the test runner will generate its own empty HTML file for browser
tests. However, tests that need custom HTML can create their own files. These
files have three requirements:

They must have the same name as the test, with .dart replaced by .html.

They must contain a link tag with rel="x-dart-test" and an href
attribute pointing to the test script.

They must contain <script src="packages/test/dart.js"></script>.

For example, if you had a test called custom_html_test.dart, you might write
the following HTML file:

Configuring Tests

Skipping Tests

If a test, group, or entire suite isn't working yet and you just want it to stop
complaining, you can mark it as "skipped". The test or tests won't be run, and,
if you supply a reason why, that reason will be printed. In general, skipping
tests indicates that they should run but is temporarily not working. If they're
is fundamentally incompatible with a platform, @TestOn/testOn
should be used instead.

To skip a test suite, put a @Skip annotation at the top of the file:

The string you pass should describe why the test is skipped. You don't have to
include it, but it's a good idea to document why the test isn't running.

Groups and individual tests can be skipped by passing the skip parameter. This
can be either true or a String describing why the test is skipped. For example:

Timeouts

By default, tests will time out after 30 seconds of inactivity. However, this
can be configured on a per-test, -group, or -suite basis. To change the timeout
for a test suite, put a @Timeout annotation at the top of the file:

In addition to setting an absolute timeout, you can set the timeout relative to
the default using @Timeout.factor. For example, @Timeout.factor(1.5) will
set the timeout to one and a half times as long as the default—45 seconds.

Timeouts can be set for tests and groups using the timeout parameter. This
parameter takes a Timeout object just like the annotation. For example:

Nested timeouts apply in order from outermost to innermost. That means that
"even slower test" will take two minutes to time out, since it multiplies the
group's timeout by 2.

Platform-Specific Configuration

Sometimes a test may need to be configured differently for different platforms.
Windows might run your code slower than other platforms, or your DOM
manipulation might not work right on Safari yet. For these cases, you can use
the @OnPlatform annotation and the onPlatform named parameter to test()
and group(). For example:

Both the annotation and the parameter take a map. The map's keys are platform
selectors which describe the platforms for which the
specialized configuration applies. Its values are instances of some of the same
annotation classes that can be used for a suite: Skip and Timeout. A value
can also be a list of these values.

If multiple platforms match, the configuration is applied in order from first to
last, just as they would in nested groups. This means that for configuration
like duration-based timeouts, the last matching value wins.

You can also set up global platform-specific configuration using the
package configuration file.

Tagging Tests

Tags are short strings that you can associate with tests, groups, and suites.
They don't have any built-in meaning, but they're very useful nonetheless: you
can associate your own custom configuration with them, or you can use them to
easily filter tests so you only run the ones you need to.

Tags are defined using the @Tags annotation for suites and the tags named
parameter to test() and group(). For example:

If the test runner encounters a tag that wasn't declared in the
package configuration file, it'll print a warning, so be
sure to include all your tags there. You can also use the file to provide
default configuration for tags, like giving all browser tests twice as much
time before they time out.

Tests can be filtered based on their tags by passing command line flags. The
--tags or -t flag will cause the test runner to only run tests with the
given tags, and the --exclude-tags or -x flag will cause it to only run
tests without the given tags. These flags also support
boolean selector syntax. For example, you can pass --tags "(chrome ||
firefox) && !slow" to select quick Chrome or Firefox tests.

Note that tags must be valid Dart identifiers, although they may also contain
hyphens.

Whole-Package Configuration

For configuration that applies across multiple files, or even the entire
package, test supports a configuration file called dart_test.yaml. At its
simplest, this file can contain the same sort of configuration that can be
passed as command-line arguments:

The configuration file sets new defaults. These defaults can still be overridden
by command-line arguments, just like the built-in defaults. In the example
above, you could pass --platform chrome to run on Chrome instead of the
Dartium content shell.

A configuration file can do much more than just set global defaults. See
the full documentation for more details.

Debugging

Tests can be debugged interactively using browsers' built-in development tools,
including Observatory when you're using Dartium. Currently there's no support
for interactively debugging command-line VM tests, but it will be added
in the future.

The first step when debugging is to pass the --pause-after-load flag to the
test runner. This pauses the browser after each test suite has loaded, so that
you have time to open the development tools and set breakpoints. For Dartium,
the test runner will print the Observatory URL for you. For PhantomJS, it will
print the remote debugger URL. For content shell, it'll print both!

Once you've set breakpoints, either click the big arrow in the middle of the web
page or press Enter in your terminal to start the tests running. When you hit a
breakpoint, the runner will open its own debugging console in the terminal that
controls how tests are run. You can type "restart" there to re-run your test as
many times as you need to figure out what's going on.

Normally, browser tests are run in hidden iframes. However, when debugging, the
iframe for the current test suite is expanded to fill the browser window so you
can see and interact with any HTML it renders. Note that the Dart animation may
still be visible behind the iframe; to hide it, just add a background-color to
the page's HTML.

Testing With barback

Packages using the barback transformer system may need to test code that's
created or modified using transformers. The test runner handles this using the
--pub-serve option, which tells it to load the test code from a pub serve
instance rather than from the filesystem.

Before using the --pub-serve option, add the test/pub_serve transformer to
your pubspec.yaml. This transformer adds the necessary bootstrapping code that
allows the test runner to load your tests properly:

Note that if you're using the test runner along with polymer, you
have to make sure that the test/pub_serve transformer comes after the
polymer transformer:

Then, start up pub serve. Make sure to pay attention to which port it's using
to serve your test/ directory:

In this case, the port is 8081. In another terminal, pass this port to
--pub-serve and otherwise invoke pub run test as normal:

Further Reading

Check out the API docs for detailed information about all the functions
available to tests.

The test runner also supports a machine-readable JSON-based reporter. This
reporter allows the test runner to be wrapped and its progress presented in
custom ways (for example, in an IDE). See the protocol documentation for
more details.

Show more