assert.verifySteps()

version added: 2.2.0.

Description

verifySteps( steps, message = "" )

Verify the presence and exact order of previously marked steps in a test.

name description
steps (array) List of strings
message (string) Short description

The Step API provides an easy way to verify execution logic to a high degree of accuracy and precision, whether for asynchronous code, event-driven code, or callback-driven code.

For example, you can mark steps to observe and validate whether parts of your code are reached correctly, or to check the frequency (how often) an asynchronous code path is executed. You can also capture any unexpected steps, which are automatically detected and shown as part of the test failure.

This assertion compares a given array of string values to a list of steps recorded via calls to assert.step().

Calling assert.verifySteps() will clear and reset the internal list of steps. This allows multiple independent sequences of assert.step() to exist within the same test.

Refer to the below examples to learn how to use the Step API in your test suite.

Changelog

UNRELEASED assert.expect() now counts assert.verifySteps() as one assertion. Steps no longer count separately.

Examples

Test event-based interface

These two examples test classes that emits event via an on() method. It might be based on an EventEmitter, such as the one found in Node.js and in other environments:

QUnit.test('good example', async function (assert) {
  MyVoice.on('noun', function (word) {
    assert.step(word);
  });
  const song = await MyVoice.sing('My Favorite Things', { lines: 1 });

  assert.true(song.finished, 'finished');
  assert.verifySteps([
    'Raindrops',
    'roses',
    'whiskers',
    'kittens'
  ]);
});
QUnit.test('good example', async function (assert) {
  const finder = new WordFinder();
  finder.on('start', () => {
    assert.step('start');
  });
  finder.on('data', (word) => {
    assert.step(word);
  });
  finder.on('end', () => {
    assert.step('end');
  });
  finder.on('error', message => {
    assert.step('error: ' + message);
  });

  await finder.process('Hello, 3.1. Great!');

  assert.verifySteps(['start', 'Hello', 'Great', 'end']);
});

If you approach this scenario without the Step API, one might be tempted to place assertions directly inside event callbacks. This is risky, because it assumes the code you’re testing has no bugs. For example, some callbacks might not run, or run multiple times, or run out of order. It is therefore generally recommended to only run assertions directly in the test function. Avoid assertions in callbacks that the test does not have control over. Such loose assurances easily produce false positives (because the subset of assertions that did run, passed!). It also offers virtually no debugging information during failures.

// WARNING: This is a BAD example
QUnit.test('bad example 1', async function (assert) {
  const finder = new WordFinder();
  finder.on('start', () => {
    assert.true(true, 'start');
  });
  finder.on('middle', () => {
    assert.true(true, 'middle');
  });
  finder.on('end', () => {
    assert.true(true, 'end');
  });
  finder.on('error', () => {
    assert.true(false, 'error');
  });

  await maker.process();
});

A less fragile approach could involve a local array that we check afterwards with deepEqual. This catches out-of-order issues, unexpected values, and duplicate values. It also places the assertion under the test’s direct control, leaving no room for it to be skipped, run too early, too late, or multiple times. The array approach also provides ample debugging information during failures. The below is in essence how the Step API works:

QUnit.test('manual example without Step API', async function (assert) {
  const values = [];

  const finder = new WordFinder();
  finder.on('start', () => {
    values.push('start');
  });
  finder.on('middle', () => {
    values.push('middle');
  });
  finder.on('end', () => {
    values.push('end');
  });
  finder.on('error', () => {
    values.push('error');
  });

  await finder.process();

  assert.deepEqual(values, ['start', 'middle', 'end']);
});

Test publish/subscribe system

Use the Step API to verify messages received in a Pub-Sub channel or topic.

QUnit.test('good example', function (assert) {
  const publisher = new Publisher();

  const subscriber1 = (message) => assert.step(`Sub 1: ${message}`);
  const subscriber2 = (message) => assert.step(`Sub 2: ${message}`);

  publisher.subscribe(subscriber1);
  publisher.subscribe(subscriber2);
  publisher.publish('Hello!');

  publisher.unsubscribe(subscriber1);
  publisher.publish('World!');

  assert.verifySteps([
    'Sub 1: Hello!',
    'Sub 2: Hello!',
    'Sub 2: World!'
  ]);
});

Multiple step verifications in one test

The internal buffer of observed steps is automatically reset when calling verifySteps().

QUnit.test('multiple verifications example', function (assert) {
  assert.step('one');
  assert.step('two');
  assert.verifySteps(['one', 'two']);

  assert.step('three');
  assert.step('four');
  assert.verifySteps(['three', 'four']);
});