2020-05-10 03:09:42 +02:00
|
|
|
# Testing
|
|
|
|
|
|
|
|
Deno has a built-in test runner that you can use for testing JavaScript or
|
|
|
|
TypeScript code.
|
|
|
|
|
2021-04-07 03:52:56 -07:00
|
|
|
`deno test` will search in `./*` and `./**/*` recursively, for test files:
|
|
|
|
|
|
|
|
- named `test.{ts, tsx, js, mjs, jsx}`,
|
|
|
|
- or ending with `.test.{ts, tsx, js, mjs, jsx}`,
|
|
|
|
- or ending with `_test.{ts, tsx, js, mjs, jsx}`
|
|
|
|
|
2020-05-10 03:09:42 +02:00
|
|
|
## Writing tests
|
|
|
|
|
|
|
|
To define a test you need to call `Deno.test` with a name and function to be
|
2020-06-18 11:13:56 +01:00
|
|
|
tested. There are two styles you can use.
|
2020-05-10 03:09:42 +02:00
|
|
|
|
|
|
|
```ts
|
2020-12-29 12:11:03 -06:00
|
|
|
import { assertEquals } from "https://deno.land/std@$STD_VERSION/testing/asserts.ts";
|
|
|
|
|
2020-06-18 11:13:56 +01:00
|
|
|
// Simple name and function, compact form, but not configurable
|
|
|
|
Deno.test("hello world #1", () => {
|
2020-05-10 03:09:42 +02:00
|
|
|
const x = 1 + 2;
|
2020-06-18 11:13:56 +01:00
|
|
|
assertEquals(x, 3);
|
|
|
|
});
|
|
|
|
|
|
|
|
// Fully fledged test definition, longer form, but configurable (see below)
|
|
|
|
Deno.test({
|
|
|
|
name: "hello world #2",
|
2020-06-26 00:35:34 +08:00
|
|
|
fn: () => {
|
2020-06-18 11:13:56 +01:00
|
|
|
const x = 1 + 2;
|
|
|
|
assertEquals(x, 3);
|
2020-06-26 00:35:34 +08:00
|
|
|
},
|
2020-05-10 03:09:42 +02:00
|
|
|
});
|
|
|
|
```
|
|
|
|
|
2020-06-18 11:13:56 +01:00
|
|
|
## Assertions
|
|
|
|
|
2020-07-31 11:12:20 +02:00
|
|
|
There are some useful assertion utilities at
|
|
|
|
https://deno.land/std@$STD_VERSION/testing#usage to make testing easier:
|
2020-05-10 03:09:42 +02:00
|
|
|
|
|
|
|
```ts
|
2020-06-18 11:13:56 +01:00
|
|
|
import {
|
2020-10-26 16:03:30 +01:00
|
|
|
assertArrayIncludes,
|
2020-09-27 06:22:32 -04:00
|
|
|
assertEquals,
|
2020-07-31 11:12:20 +02:00
|
|
|
} from "https://deno.land/std@$STD_VERSION/testing/asserts.ts";
|
2020-05-10 03:09:42 +02:00
|
|
|
|
|
|
|
Deno.test("hello world", () => {
|
|
|
|
const x = 1 + 2;
|
|
|
|
assertEquals(x, 3);
|
2020-10-26 16:03:30 +01:00
|
|
|
assertArrayIncludes([1, 2, 3, 4, 5, 6], [3], "Expected 3 to be in the array");
|
2020-05-10 03:09:42 +02:00
|
|
|
});
|
|
|
|
```
|
|
|
|
|
|
|
|
### Async functions
|
|
|
|
|
|
|
|
You can also test asynchronous code by passing a test function that returns a
|
|
|
|
promise. For this you can use the `async` keyword when defining a function:
|
|
|
|
|
|
|
|
```ts
|
2020-07-31 11:12:20 +02:00
|
|
|
import { delay } from "https://deno.land/std@$STD_VERSION/async/delay.ts";
|
2020-05-25 06:12:45 -07:00
|
|
|
|
2020-05-10 03:09:42 +02:00
|
|
|
Deno.test("async hello world", async () => {
|
|
|
|
const x = 1 + 2;
|
|
|
|
|
|
|
|
// await some async task
|
|
|
|
await delay(100);
|
|
|
|
|
|
|
|
if (x !== 3) {
|
|
|
|
throw Error("x should be equal to 3");
|
|
|
|
}
|
|
|
|
});
|
|
|
|
```
|
|
|
|
|
|
|
|
### Resource and async op sanitizers
|
|
|
|
|
|
|
|
Certain actions in Deno create resources in the resource table
|
|
|
|
([learn more here](./contributing/architecture.md)). These resources should be
|
|
|
|
closed after you are done using them.
|
|
|
|
|
2020-06-18 11:13:56 +01:00
|
|
|
For each test definition, the test runner checks that all resources created in
|
2020-05-10 03:09:42 +02:00
|
|
|
this test have been closed. This is to prevent resource 'leaks'. This is enabled
|
|
|
|
by default for all tests, but can be disabled by setting the `sanitizeResources`
|
|
|
|
boolean to false in the test definition.
|
|
|
|
|
|
|
|
The same is true for async operation like interacting with the filesystem. The
|
|
|
|
test runner checks that each operation you start in the test is completed before
|
|
|
|
the end of the test. This is enabled by default for all tests, but can be
|
|
|
|
disabled by setting the `sanitizeOps` boolean to false in the test definition.
|
|
|
|
|
|
|
|
```ts
|
|
|
|
Deno.test({
|
|
|
|
name: "leaky test",
|
|
|
|
fn() {
|
|
|
|
Deno.open("hello.txt");
|
|
|
|
},
|
|
|
|
sanitizeResources: false,
|
|
|
|
sanitizeOps: false,
|
|
|
|
});
|
|
|
|
```
|
|
|
|
|
2021-02-24 20:55:50 +08:00
|
|
|
### Exit sanitizer
|
|
|
|
|
|
|
|
There's also the exit sanitizer which ensures that tested code doesn't call
|
|
|
|
Deno.exit() signaling a false test success.
|
|
|
|
|
|
|
|
This is enabled by default for all tests, but can be disabled by setting the
|
|
|
|
`sanitizeExit` boolean to false in thetest definition.
|
|
|
|
|
|
|
|
```ts
|
|
|
|
Deno.test({
|
|
|
|
name: "false success",
|
|
|
|
fn() {
|
|
|
|
Deno.exit(0);
|
|
|
|
},
|
|
|
|
sanitizeExit: false,
|
|
|
|
});
|
|
|
|
|
|
|
|
Deno.test({
|
|
|
|
name: "failing test",
|
|
|
|
fn() {
|
|
|
|
throw new Error("this test fails");
|
|
|
|
},
|
|
|
|
});
|
|
|
|
```
|
|
|
|
|
2020-06-18 11:13:56 +01:00
|
|
|
## Running tests
|
|
|
|
|
|
|
|
To run the test, call `deno test` with the file that contains your test
|
|
|
|
function. You can also omit the file name, in which case all tests in the
|
|
|
|
current directory (recursively) that match the glob
|
|
|
|
`{*_,*.,}test.{js,mjs,ts,jsx,tsx}` will be run. If you pass a directory, all
|
|
|
|
files in the directory that match this glob will be run.
|
|
|
|
|
|
|
|
```shell
|
2020-08-24 11:36:56 -05:00
|
|
|
# Run all tests in the current directory and all sub-directories
|
2020-06-18 11:13:56 +01:00
|
|
|
deno test
|
|
|
|
|
|
|
|
# Run all tests in the util directory
|
|
|
|
deno test util/
|
|
|
|
|
|
|
|
# Run just my_test.ts
|
|
|
|
deno test my_test.ts
|
|
|
|
```
|
|
|
|
|
|
|
|
`deno test` uses the same permission model as `deno run` and therefore will
|
|
|
|
require, for example, `--allow-write` to write to the file system during
|
|
|
|
testing.
|
|
|
|
|
|
|
|
To see all runtime options with `deno test`, you can reference the command line
|
|
|
|
help:
|
|
|
|
|
|
|
|
```shell
|
|
|
|
deno help test
|
|
|
|
```
|
|
|
|
|
|
|
|
## Filtering
|
|
|
|
|
|
|
|
There are a number of options to filter the tests you are running.
|
|
|
|
|
|
|
|
### Command line filtering
|
|
|
|
|
|
|
|
Tests can be run individually or in groups using the command line `--filter`
|
|
|
|
option.
|
|
|
|
|
2020-07-07 09:13:38 -04:00
|
|
|
The filter flags accept a string or a pattern as value.
|
|
|
|
|
|
|
|
Assuming the following tests:
|
|
|
|
|
|
|
|
```ts
|
|
|
|
Deno.test({ name: "my-test", fn: myTest });
|
|
|
|
Deno.test({ name: "test-1", fn: test1 });
|
|
|
|
Deno.test({ name: "test2", fn: test2 });
|
|
|
|
```
|
|
|
|
|
|
|
|
This command will run all of these tests because they all contain the word
|
|
|
|
"test".
|
|
|
|
|
2020-06-18 11:13:56 +01:00
|
|
|
```shell
|
2020-07-07 09:13:38 -04:00
|
|
|
deno test --filter "test" tests/
|
2020-06-18 11:13:56 +01:00
|
|
|
```
|
|
|
|
|
2020-07-07 09:13:38 -04:00
|
|
|
On the flip side, the following command uses a pattern and will run the second
|
|
|
|
and third tests.
|
|
|
|
|
|
|
|
```shell
|
|
|
|
deno test --filter "/test-*\d/" tests/
|
|
|
|
```
|
|
|
|
|
|
|
|
_To let Deno know that you want to use a pattern, wrap your filter with
|
|
|
|
forward-slashes like the JavaScript syntactic sugar for a REGEX._
|
|
|
|
|
2020-06-18 11:13:56 +01:00
|
|
|
### Test definition filtering
|
|
|
|
|
|
|
|
Within the tests themselves, you have two options for filtering.
|
|
|
|
|
|
|
|
#### Filtering out (Ignoring these tests)
|
2020-05-10 03:09:42 +02:00
|
|
|
|
|
|
|
Sometimes you want to ignore tests based on some sort of condition (for example
|
|
|
|
you only want a test to run on Windows). For this you can use the `ignore`
|
|
|
|
boolean in the test definition. If it is set to true the test will be skipped.
|
|
|
|
|
|
|
|
```ts
|
|
|
|
Deno.test({
|
|
|
|
name: "do macOS feature",
|
|
|
|
ignore: Deno.build.os !== "darwin",
|
|
|
|
fn() {
|
|
|
|
doMacOSFeature();
|
|
|
|
},
|
|
|
|
});
|
|
|
|
```
|
|
|
|
|
2020-06-18 11:13:56 +01:00
|
|
|
#### Filtering in (Only run these tests)
|
2020-05-10 03:09:42 +02:00
|
|
|
|
2020-06-18 11:13:56 +01:00
|
|
|
Sometimes you may be in the middle of a problem within a large test class and
|
|
|
|
you would like to focus on just that test and ignore the rest for now. For this
|
|
|
|
you can use the `only` option to tell the test framework to only run tests with
|
|
|
|
this set to true. Multiple tests can set this option. While the test run will
|
|
|
|
report on the success or failure of each test, the overall test run will always
|
|
|
|
fail if any test is flagged with `only`, as this is a temporary measure only
|
|
|
|
which disables nearly all of your tests.
|
2020-05-10 03:09:42 +02:00
|
|
|
|
2020-06-18 11:13:56 +01:00
|
|
|
```ts
|
|
|
|
Deno.test({
|
|
|
|
name: "Focus on this test only",
|
|
|
|
only: true,
|
|
|
|
fn() {
|
|
|
|
testComplicatedStuff();
|
|
|
|
},
|
|
|
|
});
|
2020-05-10 03:09:42 +02:00
|
|
|
```
|
|
|
|
|
2020-06-18 11:13:56 +01:00
|
|
|
## Failing fast
|
2020-06-16 22:17:00 +01:00
|
|
|
|
2020-06-18 11:13:56 +01:00
|
|
|
If you have a long running test suite and wish for it to stop on the first
|
2020-11-22 14:40:33 +00:00
|
|
|
failure, you can specify the `--fail-fast` flag when running the suite.
|
2020-06-16 22:17:00 +01:00
|
|
|
|
|
|
|
```shell
|
2020-11-22 14:40:33 +00:00
|
|
|
deno test --fail-fast
|
2020-06-16 22:17:00 +01:00
|
|
|
```
|
2020-09-13 15:17:25 +02:00
|
|
|
|
|
|
|
## Test coverage
|
|
|
|
|
2021-02-24 22:27:51 +08:00
|
|
|
Deno will collect test coverage into a directory for your code if you specify
|
|
|
|
the `--coverage` flag when starting `deno test`.
|
2020-09-15 09:23:20 +08:00
|
|
|
|
2021-02-24 22:27:51 +08:00
|
|
|
This coverage information is acquired directly from the JavaScript engine (V8)
|
|
|
|
which is very accurate.
|
2020-09-13 15:17:25 +02:00
|
|
|
|
2021-02-24 22:27:51 +08:00
|
|
|
This can then be further processed from the internal format into well known
|
|
|
|
formats by the `deno coverage` tool.
|
2020-09-13 15:17:25 +02:00
|
|
|
|
|
|
|
```
|
2021-02-24 22:27:51 +08:00
|
|
|
# Go into your project's working directory
|
|
|
|
git clone https://github.com/oakserver/oak && cd oak
|
|
|
|
|
|
|
|
# Collect your coverage profile with deno test --coverage=<output_directory>
|
2021-04-27 12:44:36 +02:00
|
|
|
deno test --coverage=cov_profile
|
2021-02-24 22:27:51 +08:00
|
|
|
|
|
|
|
# From this you can get a pretty printed diff of uncovered lines
|
2021-04-27 12:44:36 +02:00
|
|
|
deno coverage cov_profile
|
2021-02-24 22:27:51 +08:00
|
|
|
|
|
|
|
# Or generate an lcov report
|
2021-04-27 12:44:36 +02:00
|
|
|
deno coverage cov_profile --lcov > cov_profile.lcov
|
2021-02-24 22:27:51 +08:00
|
|
|
|
|
|
|
# Which can then be further processed by tools like genhtml
|
|
|
|
genhtml -o cov_profile/html cov_profile.lcov
|
2020-09-13 15:17:25 +02:00
|
|
|
```
|
2021-02-24 22:27:51 +08:00
|
|
|
|
|
|
|
By default, `deno coverage` will exclude any files matching the regular
|
|
|
|
expression `test\.(js|mjs|ts|jsx|tsx)` and only consider including files
|
|
|
|
matching the regular expression `^file:`.
|
|
|
|
|
2021-05-19 13:22:04 +09:00
|
|
|
These filters can be overridden using the `--exclude` and `--include` flags. A
|
2021-02-24 22:27:51 +08:00
|
|
|
source file's url must match both regular expressions for it to be a part of the
|
|
|
|
report.
|