Lean-Test
A testing framework for when you want to test without adding hundreds of dependencies.
Runs tests in NodeJS and/or in browsers.
Running in NodeJS
npx lean-test
Running in browsers
npx lean-test --target chrome --target firefox
Features
- Run tests from the commandline against NodeJS and/or browsers;
- Natively supports ES6 modules and promises (
async
tests); - Transpile source with your chosen tooling before running;
- Fluent and matcher-style expectations, easy to add custom matchers;
- Parallel test running;
- Low overhead (fast tests);
- No dependencies;
- Highly extensible plugin and reporter architecture:
stdout
/ stderr
/ console
capturing;- lifecycle methods (
beforeAll
/ beforeEach
/ afterEach
/ afterAll
); - repeated tests, failure tolerance;
- auto retry failing tests;
- parameterised tests;
- sequential test execution with stop at first failure (for flow testing);
- optional pseudo-random test execution ordering (with a seed to allow repetition);
- configurable test timeout.
Usage
Install with:
npm install --save-dev lean-test
Or run without installing:
npx lean-test
Automatically discovers all .test.js
/ .spec.js
(and .mjs
) files by default.
Tests can be writen in a variety of ways; the classic "globals" approach:
describe('my thing', () => {
it('does something', () => {
expect(3 * 3).equals(9);
});
it('does another thing', () => {
expect(2 + 2, equals(4));
});
it('does a thing with promises', async () => {
const value = await Promise.resolve(7);
expect(value, equals(7));
});
});
Or with shorthand describe
syntax:
describe('my other thing', {
'does stuff'() {
expect(2 + 2, equals(4));
},
async 'does async stuff'() {
const value = await Promise.resolve(7);
expect(value, equals(7));
},
'sub-block': {
'more stuff'() {
expect(1 * 2, isGreaterThan(1));
},
},
});
Or if you prefer, you can avoid globals by using the export
approach:
export default ({ describe, it, expect, equals }) => {
describe('my thing', () => {
it('does a thing', () => {
expect(3 * 3, equals(9));
});
});
};
Or with shorthand describe
syntax:
export default ({ describe, expect }) => describe('my thing', {
'does a thing'() {
expect(3 * 3).equals(9);
},
});
Features
Most features are provided by plugins. The standard plugins are enabled by default, and
offer the following features. You can also create your own plugins if you have bespoke
needs.
fail
fail();
fail('message');
Throws a TestAssertionError
(marking the test as failed).
skip
skip();
skip('message');
Throws a TestAssumptionError
(marking the test as skipped).
expect
expect(2, equals(2));
expect(2).equals(2);
Checks a condition, throwing a TestAssertionError
if it fails (marking the test as
failed).
assume
assume(2, equals(2));
assume(2).equals(2);
Checks a condition, throwing a TestAssumptionError
if it fails (marking the test as
skipped). Can use all the same matchers as expect
.
expect.extend
const isSeven = () => (actual) => {
if (actual === 7) {
return {
pass: true,
message: 'Expected value not to be 7, but was.',
};
} else {
return {
pass: false,
message: `Expected 7, but got ${actual}.`,
};
}
};
expect(7, isSeven());
expect.extend({ isSeven });
expect(7).isSeven();
Globally registers new fluent checks.
expect.poll
await expect.poll(() => getThing(), equals(6));
await expect.poll(() => getThing(), equals(6), {
timeout: 1000,
interval: 100,
});
Polls a condition repeatedly until it passes, throwing a TestAssertionError
if the
timeout is reached before the condition passes (marking the test as failed).
The first parameter should be a function with no side-effects (as it will be invoked
several times). The second parameter can be any matcher, including custom matchers.
Note that expect.poll
does not support fluent matcher syntax.
You can optionally provide configuration for the polling behaviour. By default, it
will poll every 50 milliseconds for up to 5 seconds.
mock
const mockedFunc = mock();
const namedMockFunc = mock('my mock');
const spyLog = mock(console, 'log');
const myMock = mock()
.whenCalledWith(1, 'foo').thenReturn(10)
.whenCalledWith(greaterThan(5)).thenThrow(new Error('too much!'));
Creates a mock function, or spies on an existing method.
Mocked functions can be configured to return specific values when invoked,
and can be checked to see if they were called with particular arguments (see
hasBeenCalled
/ hasBeenCalledWith
below).
The extra methods available on mocks and spies are:
-
whenCalled()
:
Begins a context for configuring behaviour when the function is called.
The returned object has several fluent-API methods:
-
with(...arguments)
:
Filters for invocations with matching arguments (can be literal values,
matchers, or a combination). By default, the arguments are not checked.
-
times(n)
:
Limits the current configuration to a fixed number of invocations, after
which it is removed. This can be useful for configuring return values which
change in subsequent invocations. By default, there is no limit.
-
once()
:
Shorthand for .times(1)
.
-
then(func)
:
Configures the mock to invoke the given function when an invocation matches
the current configuration. The function will be called with all provided
arguments, and its return value will be returned, so this acts as a
pass-through.
As a convenience, this returns the original mock function, so multiple
configurations can be chained easily.
-
thenReturn(value)
:
Shorthand for .then(() => value)
-
thenThrow(error)
:
Shorthand for .then(() => { throw error; })
-
thenResolve(value)
:
Shorthand for .thenReturn(Promise.resolve(value))
-
thenReject(error)
:
Shorthand for .thenReturn(Promise.reject(error))
-
thenCallThrough()
:
Configures the spy to invoke the original method when an invocation matches
the current configuration. This is the default for spies.
As a convenience, this returns the original mock function, so multiple
configurations can be chained easily.
-
whenCalledWith(...arguments)
:
Shorthand for .whenCalled().with(...arguments)
.
-
whenCalledNext()
:
Shorthand for .whenCalled().times(1)
.
-
returning(value)
:
Shorthand for .whenCalled().thenReturn(value)
.
-
throwing(error)
:
Shorthand for .whenCalled().thenThrow(error)
.
-
reset()
:
Resets the mock configuration and recorded invocations.
-
revert()
:
Removes the spy, returning the original function (note that this only exists
for spies; it does not exist for mock functions).
-
getInvocation(index?)
and getLatestInvocation()
:
Returns an object containing the parameters the mock has been called with.
The object contains:
arguments
: a list of arguments passed when the function was calledstack
: a stacktrace for the invocation which can be used for debugging
(currently generated by new Error().stack
, but this could change in
future versions, and the exact format can change between Node / browser
versions) - can be undefined
if the platform does not support stack
traces.
If multiple whenCalled*
configurations match an invocation, the first one is
chosen. For example:
const fn = mock('my mocked function')
.whenCalledWith(greaterThan(2)).once().thenReturn('a')
.whenCalledWith(lessThan(6)).thenReturn('b')
.whenCalled().thenReturn('c');
fn(1);
fn(4);
fn(4);
fn(8);
getStdout / getStderr / getOutput
const textStdout = getStdout();
const textStderr = getStderr();
const binaryValue = getStdout(true);
const allOutput = getOutput();
Returns the content of stdout
/ stderr
captured from the current test so far.
getOutput
returns all content to both stdout
and stderr
in the order it was
written.
In the browser, only getOutput()
is available, which returns all content printed
to the console as a string. Note that the exact format of logged content is not
guaranteed (in particular, the format of printed objects may vary and the output
may include ANSI escape sequences for setting colours).
Also note that these may not capture all content; the capturing relies on inspecting
stack traces, which will not work inside event callbacks such as setTimeout
etc.
In NodeJS, console.*
will produce content in stdout
.
ignore
it.ignore('will not run', () => { });
describe.ignore('will not run', () => { });
it('will not run', { ignore: true }, () => { });
describe('will not run', { ignore: true }, () => { });
it('will not run', () => { }, { ignore: true });
describe('will not run', () => { }, { ignore: true });
Ignores a test or block. This will be reported as a skipped test.
focus
it.focus('only this will run', () => { });
describe.focus('only this will run', () => { });
it('only this will run', { focus: true }, () => { });
describe('only this will run', { focus: true }, () => { });
it('only this will run', () => { }, { focus: true });
describe('only this will run', () => { }, { focus: true });
Focuses a test or block. If any tests or blocks are focused, only the marked tests
will run, and the rest will be reported as skipped.
repeat
it('will run multiple times', { repeat: 3 }, () => { });
it('will run multiple times', () => { }, { repeat: 3 });
Runs a test multiple times, expecting every run to succeed. Can also be configured with
a failure tolerance:
it('will run multiple times', { repeat: { total: 3, maxFailures: 1 } }, () => {
});
retry
it('will retry on failure', { retry: 3 }, () => { });
it('will retry on failure', () => { }, { retry: 3 });
Runs a test multiple times until it succeeds. If any attempt succeeds, the test is
considered a success.
parameters
it('will run with multiple parameters', { parameters: [1, 2] }, (v) => { });
it('will run with multiple parameters', (v) => { }, { parameters: [1, 2] });
Runs a test multiple times with different parameters. There are a variety of ways
to set parameters:
{ parameters: [1, 2] }
{ parameters: [[1, 2], [3, 4]] }
{ parameters: [new Set([1, 2]), new Set([3, 4])] }
{ parameters: [new Set([[1, 'a'], [2, 'b']]), new Set([3, 4])] }
You can also set a parameterFilter
to exclude specific combinations of
parameters:
{
parameters: [new Set([1, 2, 3]), new Set([1, 2, 3])],
parameterFilter: (a, b) => (a !== b),
}
By default, the tests will be named using a stringified version of all the
parameters, but if you provide an object with a name
property, that name
will be used instead:
{ parameters: [
{ name: 'my first test', v1: 1, v2: 2 },
{ name: 'my second test', v1: 1, v2: 2 },
] }
timeout
it('will time out', { timeout: 1000 }, () => { });
it('will time out', () => { }, { timeout: 1000 });
Fails the test if it takes longer than the configured time (in milliseconds) to run.
Note that this will not be able to prevent "busy loops" such as while (true) {}
,
and will not terminate tasks which are running (so the test code may continue to
execute even though the timeout has triggered), but any further exceptions will be
ignored.
stopAtFirstFailure
describe('my flow test', { stopAtFirstFailure: true }, () => {
});
describe('my flow test', () => {
}, { stopAtFirstFailure: true });
Stops executing tests within the current block if one fails (subsequent tests will be
marked as skipped).
Lifecycle Hooks
describe('lifecycle', () => {
beforeAll('optional name', () => {
});
beforeEach('optional name', () => {
});
afterEach('optional name', () => {
});
afterAll('optional name', () => {
});
});
Registers execution listeners which will run before and after the whole block, or before
and after each test within the block. Multiple hooks will be executed in the order they
are defined. Nested blocks will be executed from outermost to innermost for before
, and
innermost to outermost for after
.
All methods can be asynchronous.
before
hooks can also return a function which will act like a corresponding after
hook:
describe('lifecycle', () => {
let server;
beforeAll('launch server', async () => {
server = await runServer();
return async () => {
await server.close();
};
});
});
You can also set test parameters from a beforeAll
or beforeEach
hook.
These parameters will be available to all tests which are inside the hook's
scope.
describe('lifecycle', () => {
const SERVER = beforeEach('launch server', async ({ setParameter }) => {
const server = await runServer();
setParameter(server);
return () => server.close();
});
it('does a thing', ({ [SERVER]: server }) => {
server.get('foobar');
});
});
This pattern can be useful for fully decoupling tests from global state, allowing
them to run in parallel.
These parameters are available in the first argument passed to the tests (see the
destructuring example above). They are also available to other lifecycle hooks in
the same way.
To make parameters type-safe in TypeScript, you can use:
describe('lifecycle', () => {
const SERVER = beforeEach<Server>('launch server', async ({ setParameter }) => {
const server = await runServer();
setParameter(server);
return () => server.close();
});
it('does a thing', ({ getTyped }) => {
const server = getTyped(SERVER);
server.get('foobar');
});
});
The legacy method addTestParameter
is also available, but should be avoided as
it is not possible to make type-safe, is not made available to other lifecycle
hooks, and will be removed in the future.
Finally, it is possible to get the current test path in beforeEach
and
afterEach
, and as much of the path as is known in beforeAll
and afterAll
:
describe('lifecycle', () => {
beforeEach('record info', ({ testPath }) => {
myLog.append(testPath.join(' > '));
});
it('my test', () => { });
});
Standard Matchers
-
equals(value)
:
Recursively checks for equality.
-
same(value)
:
Checks strict (===
) identity.
-
isInstanceOf(class)
:
Checks instanceof
.
-
not(expectation)
:
Negates another matcher.
e.g. expect(7, not(equals(4)))
-
any()
:
Always matches. The negation not(any())
always fails. Useful as a sub-matcher.
-
matches(regexp)
:
Checks if a string matches the given regular expression.
-
withMessage(message, expectation)
:
Customises the error message of another matcher.
e.g. expect(7, withMessage('hmm, not 7', equals(7)))
-
isTrue()
:
Checks if === true
.
-
isFalse()
:
Checks if === false
.
-
isTruthy()
:
Checks if the value is truthy (Boolean(value) === true
).
-
isFalsy()
:
Checks if the value is falsy (Boolean(value) === false
).
-
isNull()
:
Checks if === null
.
-
isUndefined()
:
Checks if === undefined
.
-
isNullish()
:
Checks if the value is nullish value === null || value === undefined
.
-
isGreaterThan(value)
:
Checks if > value
.
-
isLessThan(value)
:
Checks if < value
.
-
isGreaterThanOrEqual(value)
:
Checks if >= value
.
-
isLessThanOrEqual(value)
:
Checks if <= value
.
-
isNear(value[, precision])
:
Checks if near value
. By default, the comparison checks to 2 decimal places, but
you can configure this by providing an explicit precision. The types of precision
supported are:
{ tolerance: n }
sets an explicit permitted range (+/- n
){ decimalPlaces: n }
sets an explicit number of decimal places to check
(+/- 0.5 * 10^-n
)
-
resolves(expectation)
:
Checks if the given function or promise returns a value which matches the given
expectation (sub-matcher). expectation
can also be a literal value, in which case
it behaves as if equals(expectation)
were used. If no expectation
is given, this
just checks that the funtion returns (does not throw).
Note that if promises are involved, the expect
call should be awaited:
await expect(myPromise, resolves(equals(7)));
-
throws(expectation)
:
Checks if the given function or promise throws a value which matches the given
expectation (sub-matcher). expectation
can also be a literal string, in which case
it checks if the thrown Error
message contains the given string. If no expectation
is given, this just checks that the funtion throws.
Note that if promises are involved, the expect
call should be awaited:
await expect(myPromise, throws('oops'));
-
hasLength(expectation)
:
Checks if the value (an array, Set
, Map
, etc.) has a length matching the given
expectation (sub-matcher). expectation
can also be a literal number, in which case
it behaves as if equals(expectation)
were used. If no expectation
is given, this
just checks that the value has a length
or size
property.
-
isEmpty()
:
Checks if the value (an array, Set
, Map
, etc.) has no items.
-
contains(sub)
:
Checks if the value (a string, array, or Set
) contains the given substring or
sub-element.
-
startsWith(sub)
:
Checks if a string starts with the given substring.
-
endsWith(sub)
:
Checks if a string ends with the given substring.
-
isListOf(...elements)
:
Checks if the value contains the given elements in the listed order. Elements can be
literal or matchers (these can be mixed).
-
hasProperty(name[, expectation])
:
Checks if the value (any type) contains a property of the given name, optionally
matching the given expectation. If no expectation
is given, this just checks that
the property exists on the object (using hasOwnProperty
).
-
hasBeenCalled()
:
Checks that a mocked function has been invoked since being mocked.
-
hasBeenCalledWith(...arguments)
:
Checks that a mocked function has been invoked since being mocked, with the given
arguments (which can be literal, matchers, or a mix).
CLI flags
The lean-test
executable can be configured in various ways:
-
--preprocess <tool>
/ -c <tool>
:
Applies the specified tool as a preprocessor for all source files (excluding
node_modules
sources).
Current supported tooling:
-
babel
:
The Babel transpiler. Looks for a babel.config.*
or .babelrc.*
file, or the
babel
section of a package.json
for configuration.
Requires babel
(npm install --save-dev @babel/core
).
-
rollup
:
The Rollup bundler. Looks for a rollup.config.js
file for configuration (uses
the first entry if multiple configurations are defined).
Requires rollup
(npm install --save-dev rollup
).
-
tsc
:
The Typescript transpiler. Looks for a tsconfig.json
file for configuration.
Requires typescript
(npm install --save-dev typescript
).
-
webpack
:
The webpack bundler. Looks for a .webpack/webpackfile
,
.webpack/webpack.config.*
, or webpack.config.*
file for configuration (uses
the first entry if multiple configurations are defined).
Requires webpack
(npm install --save-dev webpack
).
-
--target <name>
/ -t <name>
/ environment TARGET=<name>
:
Runs the tests in the chosen target. Currently node
, chrome
and firefox
are
supported, or use url
then open the printed URL in any browser to start the tests.
You can also use a WebDriver-compatible server (e.g. Selenium) by setting the
WEBDRIVER_HOST
environment variable, or WEBDRIVER_HOST_<BROWSER>
to set it for
a specific browser. For example:
docker run -d -p 4444:4444 --shm-size="2g" selenium/standalone-chrome
export WEBDRIVER_HOST=localhost:4444
lean-test --target=chrome
docker run -d -p 4444:4444 --shm-size="2g" selenium/standalone-chrome
docker run -d -p 4445:4444 --shm-size="2g" selenium/standalone-firefox
export WEBDRIVER_HOST_CHROME=localhost:4444
export WEBDRIVER_HOST_FIREFOX=localhost:4445
lean-test --target=chrome --target=firefox
If you are using a remote browser, you will also need to set
--host 0.0.0.0
(or equivalently TESTRUNNER_HOST=0.0.0.0
) so that the test server
is accessible to the browser.
If you specify more than one target, the tests will be run in all targets in
parallel, with the results containing one section per target:
lean-test --target=node,chrome,firefox
lean-test --target node --target chrome --target firefox
-
--port <number>
/ environment TESTRUNNER_PORT=<number>
:
Sets an explicit port number for the browser-based tests to use. By default this is
0
(pick random available port). This only takes effect if the tests are running in a
browser.
-
--host <name>
/ environment TESTRUNNER_HOST=<name>
:
Sets an explicit host name for the browser-based tests to use. By default this is
127.0.0.1
(local loopback). This only takes effect if the tests are running in a
browser.
You may want to change this setting if you need to run tests in a browser running on a
different computer on the same network (e.g. by specifying 0.0.0.0
to make it
available over the network).
-
--import-map
/ environment IMPORT_MAP=true
:
Generates an import map for node_modules
imports. This only takes effect if the tests are running in a browser.
This allows non-relative imports like import foo from 'foo';
, which will be
resolved by looking in node_modules
, which means some projects can be tested
without needing a compilation / transpilation stage.
Note that import maps are currently supported by Chrome, Firefox, and Edge.
-
--parallel-suites
/ --parallel
/ -p
/ environment PARALLEL_SUITES=true
:
Runs test suites in parallel. This is generally recommended unless the code being
tested may cause tests in different files to interfere with each other (e.g. uses
singletons or global state).
-
--random-seed <seed>
/ -s <seed>
/ environment RANDOM_SEED=<seed>
:
Randomise the order of the test in a deterministic (repeatable) way. The seed can
be a 32-character hexadecimal string, or the word random
to pick a random seed
(the chosen seed will be printed at the end of the test run). If you want to
ensure that your tests do not depend on execution order, you can add
--random-seed=random
to your standard test run command, then if an error appears,
you can use the seed it prints to re-run the tests in that order during debugging.
-
--include <pattern>
/ -i <pattern>
:
Configures the search pattern glob. Can be set multiple times. By default, this is
**/*.{spec|test}.*
.
-
--exclude <pattern>
/ -x <pattern>
:
Configures the exclusion pattern glob. Can be set multiple times.
Note that **/node_modules
and **/.*
will always be excluded unless
--no-default-exclude
is specified.
-
--no-default-exclude
:
By default, **/node_modules
and **/.*
are always excluded. Setting this flag
allows them.
-
--parallel-discovery
/ -P
/ environment PARALLEL_DISCOVERY=true
:
Runs test discovery in parallel. This may be slightly faster than the default
(synchronous) discovery, but may fail with an error depending on the environment
and the test complexity.
After the flags, you can provide one or more directories which will be used as starting
points for scanning for tests (by default the current working directory is used).
Example:
lean-test --parallel --target chrome -i '**/*.{js|mjs}' tests
Troubleshooting
Chrome crashes with "session deleted because of page crash"
This typically means that Chrome is running in a docker instance which has limited
shared memory ("shm") available. If possible, configure the container with more space:
docker run -d -p 4444:4444 --shm-size="2g" selenium/standalone-chrome
If this is not possible (e.g. when using GitLab CI), you can set the
WEBDRIVER_DISABLE_SHM
environment variable, which will add --disable-dev-shm-usage
to the requested Chrome capabilities.
CI Examples for Browser testing
These examples assume that package.json
contains something like:
{
"scripts": {
"test": "lean-test --target=chrome,firefox"
}
}
GitLab CI/CD
build_and_test:
image: node:16
services:
- name: selenium/standalone-firefox
alias: firefox
- name: selenium/standalone-chrome
alias: chrome
variables:
WEBDRIVER_DISABLE_SHM: 'true'
WEBDRIVER_HOST_CHROME: chrome:4444
WEBDRIVER_HOST_FIREFOX: firefox:4444
TESTRUNNER_HOST: '0.0.0.0'
script:
- npm install-test
GitHub Actions
name: Test
on: [push]
jobs:
build_and_test:
runs-on: ubuntu-latest
steps:
- name: Checkout
uses: actions/checkout@v2
- name: Install Node
uses: actions/setup-node@v2
with:
node-version: '16'
- name: Test
run: npm install-test