Important: This documentation covers Yarn 1 (Classic).
For Yarn 2+ docs and migration guide, see yarnpkg.com.

Package detail

fancy-test

oclif529.2kMITdeprecated3.0.16TypeScript support: included

Package no longer supported. Contact Support at https://www.npmjs.com/support for more info.

extendable utilities for testing

mocha

readme

fancy-test

extendable utilities for testing

Version Known Vulnerabilities Downloads/week License

Why

Mocha out of the box often requires a lot of setup and teardown code in beforeEach/afterEach filters. Using this library, you can get rid of those entirely and build your tests declaratively by chaining functionality together. Using the builtin plugins and your own, you create bits of functionality and chain them together with a concise syntax. It will greatly reduce the amount of repetition in your codebase.

It should be compatible with other testing libraries as well (e.g. jest), but may require a couple small changes. If you're interested, try it out and let me know if it works.

As an example, here is what a test file might look like for an application setup with fancy-test. This chain could partially be stored to a variable for reuse.

describe('api', () => {
  fancy
  // [custom plugin] initializes the db
  .initDB({withUser: mockDBUser})

  // [custom plugin] uses nock to mock out github API
  .mockGithubAPI({user: mockGithubUser})

  // [custom plugin] that calls the API of the app
  .call('POST', '/api/user/foo', {id: mockDBUser.id})

  // add adds to the context object
  // fetch the newly created data from the API (can return a promise)
  .add('user', ctx => ctx.db.fetchUserAsync(mockDBUser.id))

  // do just runs arbitary code
  // check to ensure the operation was successful
  .do(ctx => expect(ctx.user.foo).to.equal('bar'))

  // it is essentially mocha's it(expectation, callback)
  // start the test and provide a description
  .it('POST /api/user/foo updates the user')
})

V3 Breaking Changes

Version 3 now uses sinon under the hood to manage stubs. Because of this stubs are now set like this:

import * as os from 'os'

describe('stub tests', () => {
  fancy
  .stub(os, 'platform', stub => stub.returns('foobar'))
  .it('sets os', () => {
    expect(os.platform()).to.equal('foobar')
  })
})

Usage

Setup is pretty easy, just install mocha and fancy-test, then you can use any of the examples below.

Assume the following is before all the examples:

import {fancy} from 'fancy-test'
import {expect} from 'chai'

Stub

Stub any object. Like all fancy plugins, it ensures that it is reset to normal after the test runs.

import * as os from 'os'

describe('stub tests', () => {
  fancy
  .stub(os, 'platform', stub => stub.returns('foobar'))
  .it('sets os', () => {
    expect(os.platform()).to.equal('foobar')
  })
})

Catch

catch errors in a declarative way. By default, ensures they are actually thrown as well.

describe('catch tests', () => {
  fancy
  .do(() => { throw new Error('foobar') })
  .catch(/foo/)
  .it('uses regex')

  fancy
  .do(() => { throw new Error('foobar') })
  .catch('foobar')
  .it('uses string')

  fancy
  .do(() => { throw new Error('foobar') })
  .catch(err => expect(err.message).to.match(/foo/))
  .it('uses function')

  fancy
  // this would normally raise because there is no error being thrown
  .catch('foobar', {raiseIfNotThrown: false})
  .it('do not error if not thrown')
})

Without fancy, you could check an error like this:

it('dont do this', () => {
  try {
    myfunc()
  } catch (err) {
    expect(err.message).to.match(/my custom errorr/)
  }
})

But this has a common flaw, if the test does not error, the test will still pass. Chai and other assertion libraries have helpers for this, but they still end up with somewhat messy code.

Finally

Run a task even if the test errors out.

describe('finally tests', () => {
  fancy
  .do(() => { throw new Error('x') })
  .finally(() => { /* always called */ })
  .end('always calls finally')
})

Nock

Uses nock to mock out HTTP calls to external APIs. You'll need to also install nock in your devDependencies. Automatically calls done() to ensure the calls were made and cleanAll() to remove any pending requests.

const fancy = require('fancy-test')

describe('nock tests', () => {
  fancy
  .nock('https://api.github.com', api => api
    .get('/me')
    .reply(200, {name: 'jdxcode'})
  )
  .it('mocks http call to github', async () => {
    const {body: user} = await HTTP.get('https://api.github.com/me')
    expect(user).to.have.property('name', 'jdxcode')
  })
})

Environment Variables

Sometimes it's helpful to clear out environment variables before running tests or override them to something common.

describe('env tests', () => {
  fancy
  .env({FOO: 'BAR'})
  .it('mocks FOO', () => {
    expect(process.env.FOO).to.equal('BAR')
    expect(process.env).to.not.deep.equal({FOO: 'BAR'})
  })

  fancy
  .env({FOO: 'BAR'}, {clear: true})
  .it('clears all env vars', () => {
    expect(process.env).to.deep.equal({FOO: 'BAR'})
  })
})

Do

Run some arbitrary code within the pipeline. Useful to create custom logic and debugging.

describe('run', () => {
  fancy
  .stdout()
  .do(() => console.log('foo'))
  .do(({stdout}) => expect(stdout).to.equal('foo\n'))
  .it('runs this callback last', () => {
    // test code
  })

  // add to context object
  fancy
  .add('a', () => 1)
  .add('b', () => 2)
  // context will be {a: 1, b: 2}
  .it('does something with context', context => {
    // test code
  })
})

Add

Similar to run, but extends the context object with a new property. Can return a promise or not.

describe('add', () => {
  fancy
  .add('foo', () => 'foo')
  .add('bar', () => Promise.resolve('bar'))
  .do(ctx => expect(ctx).to.include({foo: 'foo', bar: 'bar'}))
  .it('adds the properties')
})

Stdin Mocking

Mocks stdin. You may have to pass a delay to have it wait a bit until it sends the event.

describe('stdin test', () => {
  fancy
  .stdin('whoa there!\n')
  .stdout()
  .it('mocks', () => {
    process.stdin.setEncoding('utf8')
    process.stdin.once('data', data => {
      // data === 'whoa there!\n'
    })
  })
})

Stdout/Stderr Mocking

This is used for tests that ensure that certain stdout/stderr messages are made. By default this also trims the output from the screen. See the output by setting TEST_OUTPUT=1, or by setting {print: true} in the options passed.

You can use the library stdout-stderr directly for doing this, but you have to be careful to always reset it after the tests run. We do that work for you so you don't have to worry about mocha's output being hidden.

describe('stdmock tests', () => {
  fancy
  .stdout()
  .it('mocks stdout', output => {
    console.log('foobar')
    expect(output.stdout).to.equal('foobar\n')
  })

  fancy
  .stderr()
  .it('mocks stderr', output => {
    console.error('foobar')
    expect(output.stderr).to.equal('foobar\n')
  })

  fancy
  .stdout()
  .stderr()
  .it('mocks stdout and stderr', output => {
    console.log('foo')
    console.error('bar')
    expect(output.stdout).to.equal('foo\n')
    expect(output.stderr).to.equal('bar\n')
  })
})

Done

You can get the mocha done() callback by passing in a second argument.

describe('calls done', () => {
  fancy
  .it('expects FOO=bar', (_, done) => {
    done()
  })
})

Retries

Retry the test n times.

let count = 3

describe('test retries', () => {
  fancy
  .retries(2)
  .do(() => {
    count--
    if (count > 0) throw new Error('x')
  })
  .it('retries 3 times')
})

Timeout

Set mocha timeout duration.

const wait = (ms = 10) => new Promise(resolve => setTimeout(resolve, ms))

describe('timeout', () => {
  fancy
  .timeout(50)
  .it('times out after 50ms', async () => {
    await wait(100)
  })
})

Chai

This library includes chai for convenience:

import {expect, fancy} from 'fancy-test'

describe('has chai', () => {
  fancy
  .env({FOO: 'BAR'})
  .it('expects FOO=bar', () => {
    expect(process.env.FOO).to.equal('BAR')
  })
})

Chaining

Everything here is chainable. You can also store parts of a chain to re-use later on.

For example:

describe('my suite', () => {
  let setupDB = fancy
                .do(() => setupDB())
                .env({FOO: 'FOO'})

  setupDB
  .stdout()
  .it('tests with stdout mocked', () => {
    // test code
  })

  setupDB
  .env({BAR: 'BAR'})
  .it('also mocks the BAR environment variable', () => {
    // test code
  })
})

Using do you can really maximize this ability. In fact, you don't even need to pass a callback to it if you prefer this syntax:

describe('my suite', () => {
  let setupDB = fancy
                .do(() => setupDB())
                .catch(/spurious db error/)
                .do(() => setupDeps())

  let testMyApp = testInfo => {
    return setupDB.run()
    .do(context => myApp(testInfo, context))
  }

  testMyApp({info: 'test run a'})
  .it('tests a')

  testMyApp({info: 'test run b'})
  .it('tests b')
})

Custom Plugins

It's easy to create your own plugins to extend fancy. In oclif we use fancy to create custom command testers.

Here is an example that creates a counter that could be used to label each test run. See the actual test to see the TypeScript types needed.

let count = 0

fancy = fancy
.register('count', prefix => {
  return {
    run(ctx) {
      ctx.count = ++count
      ctx.testLabel = `${prefix}${count}`
    }
  }
})

describe('register', () => {
  fancy
  .count('test-')
  .it('is test #1', context => {
    expect(context.count).to.equal(1)
    expect(context.testLabel).to.equal('test-1')
  })

  fancy
  .count('test-')
  .it('is test #2', context => {
    expect(context.count).to.equal(2)
    expect(context.testLabel).to.equal('test-2')
  })
})

TypeScript

This module is built in typescript and exports the typings. Doing something with dynamic chaining like this was not easy, but it should be fully typed throughout. Look at the internal plugins to get an idea of how to keep typings for your custom plugins.

changelog

3.0.16 (2024-05-19)

Bug Fixes

  • deps: bump @types/lodash from 4.17.1 to 4.17.4 (#379) (7f95e30)

3.0.15 (2024-05-05)

Bug Fixes

  • deps: bump @types/lodash from 4.17.0 to 4.17.1 (#375) (0b8dce3)

3.0.14 (2024-03-17)

Bug Fixes

  • deps: bump @types/lodash from 4.14.202 to 4.17.0 (#364) (8fde628)

3.0.13 (2024-03-11)

Bug Fixes

3.0.12 (2024-03-03)

Bug Fixes

  • deps: bump nock from 13.5.3 to 13.5.4 (#359) (62277f2)

3.0.11 (2024-02-18)

Bug Fixes

  • deps: bump nock from 13.5.1 to 13.5.3 (#356) (18b17f7)

3.0.10 (2024-02-04)

Bug Fixes

  • deps: bump nock from 13.5.0 to 13.5.1 (#355) (f488aad)

3.0.9 (2024-01-21)

Bug Fixes

  • deps: bump nock from 13.4.0 to 13.5.0 (#354) (8413f25)

3.0.8 (2023-12-24)

Bug Fixes

  • deps: bump @types/chai from 4.3.6 to 4.3.11 (#350) (1cbac0e)

3.0.7 (2023-12-24)

Bug Fixes

  • deps: bump sinon from 16.0.0 to 16.1.3 (#351) (6bed7ce)

3.0.6 (2023-12-17)

Bug Fixes

  • deps: bump @types/node from 14.18.61 to 14.18.63 (#335) (df09200)

3.0.5 (2023-12-17)

Bug Fixes

  • deps: bump nock from 13.3.3 to 13.4.0 (#347) (251196f)

3.0.4 (2023-12-16)

Bug Fixes

  • deps: bump minimist from 1.2.0 to 1.2.8 (#342) (9f86f44)

3.0.3 (2023-12-16)

Bug Fixes

  • deps: bump minimatch from 3.0.4 to 3.1.2 (#343) (883738b)

3.0.2 (2023-12-16)

Bug Fixes

  • deps: bump @types/lodash from 4.14.198 to 4.14.202 (#345) (5c1df0c)

3.0.1 (2023-09-25)

2.0.42 (2023-09-17)

Bug Fixes

  • deps: bump @types/node from 14.18.59 to 14.18.61 (#331) (44e5f6f)

2.0.41 (2023-09-10)

Bug Fixes

  • deps: bump @types/lodash from 4.14.197 to 4.14.198 (#328) (558a899)

2.0.40 (2023-09-10)

Bug Fixes

  • deps: bump @types/node from 14.18.58 to 14.18.59 (#329) (e77afe2)

2.0.39 (2023-09-10)

Bug Fixes

  • deps: bump @types/chai from 4.3.5 to 4.3.6 (#330) (95d6f22)

2.0.38 (2023-09-03)

Bug Fixes

  • deps: bump @types/node from 14.18.56 to 14.18.58 (#326) (a506f29)

2.0.37 (2023-08-27)

Bug Fixes

2.0.36 (2023-08-27)

Bug Fixes

  • deps: bump @types/node from 14.18.54 to 14.18.56 (#325) (b0f5e8b)

2.0.35 (2023-08-20)

Bug Fixes

  • deps: bump nock from 13.3.2 to 13.3.3 (#322) (dd78798)

2.0.34 (2023-08-13)

Bug Fixes

  • deps: bump @types/lodash from 4.14.196 to 4.14.197 (#321) (24fc84f)

2.0.33 (2023-07-30)

Bug Fixes

  • deps: bump @types/lodash from 4.14.195 to 4.14.196 (#320) (9b0e36e)

2.0.32 (2023-07-23)

Bug Fixes

  • deps: bump @types/node from 14.18.53 to 14.18.54 (#318) (6381f89)

2.0.31 (2023-07-19)

Bug Fixes

  • deps: bump word-wrap from 1.2.3 to 1.2.4 (#317) (ed53061)

2.0.30 (2023-07-16)

Bug Fixes

  • deps: bump nock from 13.3.1 to 13.3.2 (#316) (f25e78f)

2.0.29 (2023-07-12)

Bug Fixes

  • deps: bump semver from 5.7.1 to 5.7.2 (#315) (c99450b)

2.0.28 (2023-07-02)

Bug Fixes

  • deps: bump @types/node from 14.18.51 to 14.18.53 (#314) (6764f30)

2.0.27 (2023-06-18)

Bug Fixes

  • deps: bump @types/node from 14.18.50 to 14.18.51 (#312) (c79f2ab)

2.0.26 (2023-06-11)

Bug Fixes

  • deps: bump @types/node from 14.18.48 to 14.18.50 (#311) (8b427d1)

2.0.25 (2023-05-28)

Bug Fixes

  • deps: bump @types/node from 14.18.47 to 14.18.48 (#308) (7e47e7e)

2.0.24 (2023-05-28)

Bug Fixes

  • deps: bump @types/lodash from 4.14.194 to 4.14.195 (#309) (beaf78b)

2.0.23 (2023-05-14)

Bug Fixes

  • deps: bump @types/node from 14.18.45 to 14.18.47 (#306) (43075ab)

2.0.22 (2023-05-07)

Bug Fixes

  • deps: bump @types/node from 14.18.43 to 14.18.45 (#305) (6fc4c29)

2.0.21 (2023-04-30)

Bug Fixes

  • deps: bump @types/node from 14.18.42 to 14.18.43 (#303) (8683e62)

2.0.20 (2023-04-30)

Bug Fixes

  • deps: bump nock from 13.3.0 to 13.3.1 (#304) (4f5ef50)

2.0.19 (2023-04-16)

Bug Fixes

  • deps: bump @types/lodash from 4.14.192 to 4.14.194 (#302) (56b088e)

2.0.18 (2023-04-02)

Bug Fixes

  • deps: bump @types/lodash from 4.14.191 to 4.14.192 (#300) (2ceec3e)

2.0.17 (2023-04-02)

Bug Fixes

  • deps: bump @types/node from 14.18.41 to 14.18.42 (#301) (2d69c76)

2.0.16 (2023-03-26)

Bug Fixes

  • deps: bump @types/node from 14.18.38 to 14.18.41 (#299) (6379a2d)

2.0.15 (2023-03-21)

Bug Fixes

  • deps: bump remarkable from 1.7.1 to 1.7.4 (#298) (d9b2c55)

2.0.14 (2023-03-19)

Bug Fixes

  • deps: bump @types/node from 14.18.37 to 14.18.38 (#297) (5d5038f)

2.0.13 (2023-03-05)

Bug Fixes

  • deps: bump @types/node from 14.18.36 to 14.18.37 (#296) (40d6318)

2.0.12 (2023-01-15)

Bug Fixes

  • deps: bump nock from 13.1.0 to 13.3.0 (#293) (edd9c6c)

2.0.11 (2023-01-01)

Bug Fixes

  • deps: bump @types/node from 14.18.35 to 14.18.36 (#292) (435970e)

2.0.10 (2022-12-18)

Bug Fixes

  • deps: bump @types/node from 14.18.34 to 14.18.35 (#291) (69c0f4b)

2.0.9 (2022-12-04)

Bug Fixes

  • deps: bump @types/lodash from 4.14.188 to 4.14.191 (#288) (9159f78)

2.0.8 (2022-12-04)

Bug Fixes

  • deps: bump @types/node from 14.18.33 to 14.18.34 (#289) (3d83234)

2.0.7 (2022-11-06)

Bug Fixes

  • deps: bump @types/lodash from 4.14.186 to 4.14.188 (#281) (1607c33)

2.0.6 (2022-10-30)

Bug Fixes

  • deps: bump @types/node from 14.18.32 to 14.18.33 (#280) (0064355)

2.0.5 (2022-10-16)

Bug Fixes

  • deps: bump @types/node from 14.18.31 to 14.18.32 (#278) (7aead8b)

2.0.4 (2022-10-01)

Bug Fixes

  • deps: bump @types/lodash from 4.14.185 to 4.14.186 (#274) (94a340a)

2.0.3 (2022-10-01)

Bug Fixes

  • deps: bump @types/node from 14.18.30 to 14.18.31 (#276) (096fd0a)

2.0.2 (2022-09-29)

Bug Fixes

  • deps: bump @types/node from 14.14.31 to 14.18.30 (#270) (4fda69c)

2.0.1 (2022-09-29)

Bug Fixes

  • deps: bump @types/lodash from 4.14.170 to 4.14.185 (#272) (8738b82)

2.0.0 (2021-10-04)

1.4.10 (2020-12-02)

Bug Fixes

1.4.9 (2020-07-02)

Bug Fixes

1.4.8 (2020-05-15)

1.4.7 (2020-01-27)

1.4.6 (2020-01-20)

1.4.5 (2020-01-20)

1.4.4 (2019-08-02)

1.4.3 (2019-01-10)

1.4.2 (2019-01-10)

Bug Fixes

1.4.1 (2018-09-14)

Bug Fixes

  • use generic tuples from ts3 (d89fed0)

1.4.0 (2018-08-31)

Features

1.3.0 (2018-08-17)

Features

1.2.0 (2018-06-14)

Features

1.1.4 (2018-06-01)

Bug Fixes

1.1.3 (2018-06-01)

Bug Fixes

1.1.2 (2018-06-01)

Bug Fixes

  • gracefully degrade with no nock (f673db8)

1.1.1 (2018-06-01)

1.1.0 (2018-05-31)

Features

1.0.10 (2018-05-31)

1.0.9 (2018-05-31)

Bug Fixes

1.0.8 (2018-05-07)

Bug Fixes

  • remove mocha type reference (3998b46)

1.0.7 (2018-05-07)

Bug Fixes

  • bring ITestCallbackContext in from mocha (d011192)

1.0.6 (2018-05-01)

Bug Fixes

1.0.5 (2018-04-18)

Bug Fixes

1.0.4 (2018-04-06)

Bug Fixes

  • remove mention of DEBUG for stdout/stderr (f511c16)

1.0.3 (2018-04-06)

Bug Fixes

1.0.2 (2018-04-06)

Bug Fixes

  • rename run to do in the docs (4f6f5d9)

1.0.1 (2018-02-05)

Bug Fixes

1.0.0 (2018-02-05)

Bug Fixes

  • move nock to other module (d88f065)

BREAKING CHANGES

  • nock is now in @fancy-test/nock

0.6.10 (2018-02-03)

Bug Fixes

0.6.9 (2018-02-03)

Bug Fixes

0.6.8 (2018-02-03)

Bug Fixes

  • fixed skipping when first task in chain (1c40e4c)

0.6.7 (2018-02-03)

Bug Fixes

  • clone context before running plugin (81a13b7)

0.6.6 (2018-02-01)

Bug Fixes

0.6.5 (2018-01-29)

Bug Fixes

0.6.4 (2018-01-28)

Bug Fixes

0.6.3 (2018-01-28)

Bug Fixes

  • updated semantic-release (33db7ed)

0.6.2 (2018-01-28)

Bug Fixes

  • updated semantic-release (460043e)

0.6.1 (2018-01-28)

Bug Fixes

0.6.0 (2018-01-28)

Features

  • remove undefined/nulls from env (b40e4bd)

0.5.5 (2018-01-27)

Bug Fixes

0.5.4 (2018-01-27)

Bug Fixes

  • always reset full context when retrying (72c3588)

0.5.3 (2018-01-27)

Bug Fixes

0.5.2 (2018-01-27)

Bug Fixes

  • set this on mocha callback (5968028)

0.5.1 (2018-01-26)

Bug Fixes

0.5.0 (2018-01-26)

Features

0.4.0 (2018-01-26)

Features

0.3.10 (2018-01-26)

Bug Fixes

  • allow do to modify context (d997f3a)

0.3.9 (2018-01-26)

Bug Fixes

0.3.8 (2018-01-26)

Bug Fixes

  • export types as FancyTypes (716d3b9)

0.3.7 (2018-01-26)

Bug Fixes

0.3.6 (2018-01-26)

Bug Fixes

  • simplify exported types (01401df)

0.3.5 (2018-01-26)

Bug Fixes

  • refactor to save an export (00a8333)

0.3.4 (2018-01-26)

Bug Fixes

0.3.3 (2018-01-26)

Bug Fixes

0.3.3 (2018-01-25)

Bug Fixes

0.3.2 (2018-01-25)

Bug Fixes

0.3.1 (2018-01-25)

Bug Fixes

  • nock: added nock.cleanAll() (5c40945)

0.3.0 (2018-01-25)

Features

0.2.0 (2018-01-25)

Features

  • added options to stdout/stderr (c94e36f)

0.1.0 (2018-01-25)

Features

0.0.6 (2018-01-21)

Bug Fixes

0.0.5 (2018-01-21)

Bug Fixes

  • move chai to non-required (692f325)

0.0.4 (2018-01-21)

Bug Fixes

  • made syntax typedoc compatible (ec27d0b)

0.0.3 (2018-01-21)

Bug Fixes

0.0.2 (2018-01-21)

Bug Fixes

0.0.1 (2018-01-21)

Bug Fixes

0.0.0 (2018-01-21)