QUnit Compatible (mostly! 🙈)
pnpm install @warp-drive/diagnostic
Tagged Releases
@warp-drive/ diagnostic is a ground-up revisiting of the APIs QUnit popularized and Ember polished.
- 💜 Fully Typed
- Universal
- ⚡️ Fast
- ✅ Easy to use
@warp-drive/ diagnostic is also a test launcher/runner inspired by the likes of Testem, ember exam and the ember test command. It is similarly flexible, but faster and more lightweight while somehow bringing a more robust feature set to the table.
- 🚀 Easy Browser Setup Included
- Runs without fuss on Github Actions
- 📦 Out of the box randomization, parallelization, load balancing, and more.
But don't worry, if you're not ready to leave your existing stack the launcher/runner portion is optional. Out of the box, it comes ready with a Testem integration, or you can add your own.
- Writing Tests
- Running Tests
- Using the DOM Reporter
- Concurrency
- Using The Launcher
- Adding A Sidecar
- 🔜 Parallelism
- 🔜 Randomization
- Why Is It Fast?
- Migration From QUnit
- Using with Ember
import { module, test } from '@warp-drive/diagnostic';
module('My Module', function(hooks) {
hooks.beforeEach(async function() {
// do setup
});
test('It Works!', async function(assert) {
assert.ok('We are up and running');
});
});
Tests and hooks may be async or sync.
The this
context and assert
instance passed to a beforeEach
or afterEach
hook is the same as will be used for the given test but is not shared across tests.
This makes this
a convenient pattern for accessing or stashing state during setup/teardown in a manner that is safe for test concurrency.
Global and module level state that is not safely shared between multiple tests potentially running simultaneously should be avoided.
When augmenting this
, import TestContext
.
import { type TestContext } from '@warp-drive/diagnostic';
interface ModuleContext extends TestContext {
some: 'state';
}
module('My Module', function(hooks) {
hooks.beforeEach(async function(this: ModuleContext) {
this.some = 'state';
});
test('It Works!', async function(this: ModuleContext, assert) {
assert.equal(this.some, 'state', 'We are up and running');
});
});
Alternatively, key some state to a WeakMap and avoid the type gymnastics.
interface ModuleState {
some: 'state';
}
const STATES = new WeakMap<object, ModuleState>();
export function setState(key: object, state: ModuleState) {
STATES.set(key, state);
}
export function getState(key: object) {
const state = STATES.get(key);
if (!state) {
throw new Error(`Failed to setup state`);
}
return state;
}
Now all we need to do is use the this
we already have!
import { setState, getState } from './helpers';
module('My Module', function(hooks) {
hooks.beforeEach(async function() {
setState(this, { some: 'state' });
});
test('It Works!', async function(assert) {
const state = getState(this);
assert.equal(state.some, 'state', 'We are up and running');
});
});
Note This section is about how to setup your tests to run once launched. To learn about launching tests, read Using The Launcher
Warning This section is nuanced, read carefully!
To run your tests, import and run start
.
import { start } from '@warp-drive/diagnostic';
start();
Start will immediately begin running any tests it knows about, so when you call start matters.
For instance, if your tests require DOM to be setup, making sure start
is called only once DOM exists is important.
If there are global hooks that need configured, that configuration needs to happen before you call start
. Similar with any reporters, registerReporter
must be called first.
import { registerReporter, setupGlobalHooks, start } from '@warp-drive/diagnostic';
import CustomReporter from './my-custom-reporter';
setupGlobalHooks((hooks) => {
hooks.beforeEach(() => {
// .. some setup
});
hooks.afterEach(() => {
// .. some teardown
});
});
registerReporter(new CustomReporter());
start();
For convenience, a DOMReporter
is provided. When using the DOMReporter
it expects to be given an element to render the report into.
import { registerReporter, start } from '@warp-drive/diagnostic';
import { DOMReporter } from '@warp-drive/diagnostic/reporters/dom';
const container = document.getElementById('warp-drive__diagnostic');
registerReporter(new DOMReporter(container));
start();
When using this reporter you will likely want to include the css
for it, which can be imported from @warp-drive/diagnostic/dist/styles/dom-reporter.css
The specific container element id
of warp-drive__diagnostic
only matters if using the provided dom-reporter CSS, custom CSS may be used.
For convenience, the above code can be condensed by using the DOM runner
.
import { start } from '@warp-drive/diagnostic/runners/dom';
start();
By default, diagnostic will only run tests one at a time, waiting for all beforeEach
and afterEach
hooks to be called for a test before moving on to the next.
This is exactly as QUnit would have run the tests. For most this linear mode is likely a requirement due to state having been stored in module scope or global scope.
But if you are starting fresh, or have a test suite and program that is very well encapsulated, you may benefit from using test concurrency.
Emphasis on may because concurrency will only help if there is significany empty time
during each test due to things such as requestAnimationFrame
, setTimeout
or a
fetch
request.
Concurrency is activated by providing a concurrency option in your test suite config. The option should be a positive integer
greater than 1
for it to have any effect.
import { configure, start } from '@warp-drive/diagnostic';
configure({
concurrency: 10
});
start();
Skip to Advanced
First, we need to add a configuration file for the launcher to our project.
If our build assets are located in <dir>/dist-test/*
and the entry point for tests is dist-test/tests/index.html
, then the default configuration will get us setup with no further effort.
<dir>/diagnostic.js
import launch from '@warp-drive/diagnostic/server/default-setup.js';
await launch();
Next, adjust the configuration for start
to tell the runner to emit test information to the diagnostic server.
start({
groupLogs: false,
instrument: true,
hideReport: false,
+ useDiagnostic: true,
});
Next, we will want to install bun
. (We intend to pre-bundle the runner as an executable in the near future, but until then this is required).
For github-actions, use the official bun action
- uses: oven-sh/setup-bun@v1
with:
bun-version: latest
Finally, give your tests a run to make sure they still work as expected.
bun ./diagnostic.js
And update any necessary scripts in package.json
{
"scripts": {
"build" "ember build",
- "test": "ember test"
+ "test": "bun run build && bun ./diagnostic.js"
}
}
✅ That's all! You're ready to test! 💜
Diagnostic's launcher supports running additional services alongside your test suite when they are necessary for your tests to run correctly. For instance, you may want to start a local API instance, http mock service, or a build process.
@warp-drive/holodeck is an http mock service for test suites. We can start and stop the holodeck server along side our test server with an easy integration.
[Coming Soon]
[Coming Soon]
There's a number of micro-optimizations, but the primary difference is in "yielding".
QUnit
and ember-qunit
both schedule async checks using setTimeout
. Even if no work needs to happen and the thread is free, setTimeout
will delay ~4.5ms
before executing its callback.
When you delay in this manner multiple times per test, and have lots of tests, things add up.
In our experience working on EmberData, most of our tests, even our more complicated ones, had
completion times in the 4-30ms
range, the duration of which was dominated by free-time spent
waiting for setTimeout
callbacks. We did some math and realized that most of our tests run in
less than 0.5ms
, and even our average was <4ms
, smaller than the time for even a single setTimeout
callback.
@warp-drive/diagnostic
runs tests as microtasks. Yielding out of the microtask queue only occurs if
the test itself needs to do so.
Note soon we will elect to periodically yield just to allow the DOMReporter to show results, currently its so fast though that the tests are done before you'd care.
Next, diagnostic, uses several singleton patterns internally to keep allocations to a minimum while running tests.
By not waiting for DOMComplete and by being more intelligent about yielding, we start running tests sooner. In most situations this means test runs start 100-200ms quicker.
We further noticed that the qunit DOM Reporter was its own bottleneck for both memory and compute time. For our version we made a few tweaks to reduce this cost, which should especially help test suites with thousands or tens of thousands of tests.
Lastly, we noticed that the serialization and storage of objects being reported had a high cost.
This was a problem shared between the launcher (Testem) and what QUnit was providing to it. For this,
we opted to reduce the amount of information shared to Testem by default to the bare minimum, but with a fast debug
toggle to switch into the more verbose mode.
- Replace
qunit
with@warp-drive/diagnostic
index 2fbga6a55..c9537dd37 100644
--- a/package.json
+++ b/package.json
@@ -23,5 +23,5 @@
- "qunit": "2.20.0",
+ "@warp-drive/diagnostic": "latest",
- Update imports from
qunit
to@warp-drive/diagnostic
--- a/tests/example.ts
+++ b/tests/example.ts
@@ -1,0 +1,0 @@
- import { module, test } from 'qunit';
+ import { module, test } from '@warp-drive/diagnostic';
- Use
equal
andnotEqual
Diagnostic has no loose comparison mode. So instead of strictEqual
and notStrictEqual
we can just use equal
and notEqual
which are already strict.
- Update module hooks
beforeEach
and afterEach
are unchanged.
before
and after
become beforeModule
and afterModule
.
module('My Module', function(hooks) {
- hooks.before(function(assert) {
+ hooks.beforeModule(function(assert) {
// ...
});
- hooks.after(function(assert) {
+ hooks.afterModule(function(assert) {
// ...
});
});
- Update global hooks
QUnit.begin
and QUnit.done
become onSuiteStart
and onSuiteFinish
respectively.
QUnit.hooks
becomes setupGlobalHooks
.
+ import { setupGlobalHooks } from '@warp-drive/diagnostic';
- QUnit.begin(function() {});
- QUnit.done(function() {});
- QUnit.hooks.beforeEach(function() {});
+ setupGlobalHooks(function(hooks) {
+ hooks.onSuiteStart(function() {});
+ hooks.onSuiteFinish(function() {});
+ hooks.beforeEach(function() {});
+ });
- Add the following peer-deps to your app:
+ "@ember/test-helpers": ">= 3.3.0",
+ "ember-cli-test-loader": ">= 3.1.0",
+ "@embroider/addon-shim": ">= 1.8.6"
- Configure for ember in
test-helper.js
import { configure } from '@warp-drive/diagnostic/ember';
configure();
- Use setup helpers
import { module, test } from '@warp-drive/diagnostic';
import { setupTest } from '@warp-drive/diagnostic/ember';
module('My Module', function (hooks) {
setupTest(hooks);
});