.command()
allowed you to run a command from your CLI. This can now be achieved with the runCommand
function.
Before
import {expect, test} from '@oclif/test'
describe('my cli', () => {
test
.stdout()
.command(['hello:world'])
.it('runs hello world cmd', (ctx) => {
expect(ctx.stdout).to.contain('hello world!')
})
})
After
import {expect} from 'chai'
import {runCommand} from '@oclif/test'
describe('my cli', () => {
it('should run the command', async () => {
const {stdout} = await runCommand<{name: string}>(['hello:world'])
expect(stdout).to.contain('hello world')
})
})
Before (Single Command CLI)
import {expect, test} from '@oclif/test'
describe('my cli', () => {
test
.stdout()
.command(['.'])
.it('runs hello world cmd', (ctx) => {
expect(ctx.stdout).to.contain('hello world!')
})
})
After (Single Command CLI)
import {expect} from 'chai'
import {runCommand} from '@oclif/test'
describe('my cli', () => {
it('should run the command', async () => {
const {stdout} = await runCommand<{name: string}>(['.'])
expect(stdout).to.contain('hello world')
})
})
.exit()
allowed you to test that command exited with a certain exit code. This can now be done by inspecting the error
that's returned by runCommand
Before
import {join} from 'node:path'
import {expect, test} from '@oclif/test'
describe('exit', () => {
test
.loadConfig()
.stdout()
.command(['hello:world', '--code=101'])
.exit(101)
.do((output) => expect(output.stdout).to.equal('exiting with code 101\n'))
.it('should exit with code 101')
})
After
import {expect} from 'chai'
import {runCommand} from '@oclif/test'
describe('exit', () => {
it('should exit with code 101', async () => {
const {error} = await runCommand<{name: string}>(['hello:world', '--code=101'])
expect(error?.oclif?.exit).to.equal(101)
})
})
.hook()
allowed you to test a hook in your CLI. This can now be accomplished using the runHook
function.
Before
import {join} from 'node:path'
import {expect, test} from '@oclif/test'
const root = join(__dirname, 'fixtures/test-cli')
describe('hooks', () => {
test
.loadConfig({root})
.stdout()
.hook('foo', {argv: ['arg']}, {root})
.do((output) => expect(output.stdout).to.equal('foo hook args: arg\n'))
.it('should run hook')
})
After
import {join} from 'node:path'
import {expect} from 'chai'
import {runHook} from '@oclif/test'
const root = join(__dirname, 'fixtures/test-cli')
describe('my cli', () => {
it('should run hook', async () => {
const {stdout} = await runHook('foo', {argv: ['arg']}, {root})
expect(stdout).to.equal('foo hook args: arg\n')
})
})
Version 3 allowed you to access the output in stdout and stderr by attaching it to the context
. This is now replaced by the stdout
and stderr
strings that are returned by runCommand
, runHook
, and captureOutput
Before
describe('stdmock tests', () => {
fancy
.stdout()
.stderr()
.it('mocks stdout and stderr', (context) => {
console.log('foo')
console.error('bar')
expect(context.stdout).to.equal('foo\n')
expect(context.stderr).to.equal('bar\n')
})
})
After
import {expect} from 'chai'
import {captureOutput} from '@oclif/test'
describe('stdmock tests', () => {
it('mocks stdout and stderr', async () => {
const {stdout, stderr} = await captureOutput(async () => {
console.log('foobar')
console.error('bar')
})
expect(stdout).to.equal('foo\n')
expect(stderr).to.equal('bar\n')
})
})
.loadConfig()
allowed you to explicitly set the root of the CLI to be tested. This can now be achieved by passing the path into the runCommand
function.
Before
import {join} from 'node:path'
import {expect, test} from '@oclif/test'
const root = join(__dirname, 'fixtures/test-cli')
describe('my cli', () => {
test
.loadConfig({root})
.stdout()
.command(['foo:bar'])
.it('should run the command from the given directory', (ctx) => {
expect(ctx.stdout).to.equal('hello world!\n')
const {name} = ctx.returned as {name: string}
expect(name).to.equal('world')
})
})
After
import {join} from 'node:path'
import {expect} from 'chai'
import {runCommand} from '@oclif/test'
const root = join(__dirname, 'fixtures/test-cli')
describe('my cli', () => {
it('should run the command from the given directory', async () => {
const {result, stdout} = await runCommand<{name: string}>(['foo:bar'], {root})
expect(result.name).to.equal('world')
})
})
.do()
allowed you to execute some arbitrary code within the test pipeline. There's not a direct replacement in version 4, however, you are still able to execute arbitrary code within your chosen test framework. For example, mocha exposes the beforeEach
and before
hooks.
.catch()
allowed you to catch errors in a declarative way and ensure that the error was actually thrown. We encourage you to use the utilities provided by your preferred testing framework to accomplish this.
.finally()
allowed you to run a task at the end of a test, even if it failed. We encourage you to use the utilities provided by your preferred testing framework to accomplish this (for instance, mocha provided afterEach
and after
lifecycle hooks).
.env()
allowed you to set the environment variables before running the test. If you need this, you can easily implement it yourself.
Before
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'})
})
})
After
describe('env tests', () => {
let originalEnv
beforeEach(() => {
originalEnv = {...process.env}
process.env = {
...originalEnv
FOO: 'BAR'
}
})
afterEach(() => {
process.env = originalEnv
})
it('mocks FOO', () => {
expect(process.env.FOO).to.equal('BAR')
expect(process.env).to.not.deep.equal({FOO: 'BAR'})
})
})
.stub()
allowed you to stub any object and ensure that the stubs were cleared before the next test. We encourage you to use a dedicated library like sinon for this.
.add()
allowed you to extend the context
object that was used throughout fancy-tests. There is no direct replacement for this in version 4.
.stdin()
allowed you mock stdin. There is no direct replacement for this in version 4. You can use mock-stdin directly if you need this functionality.
Before
describe('stdin test', () => {
fancy
.stdin('whoa there!\n')
.stdout()
.it('mocks', () => {
process.stdin.setEncoding('utf8')
process.stdin.once('data', (data) => {
// data === 'whoa there!\n'
})
})
})
After
import {captureOutput} from '@oclif/test'
import {expect} from 'chai'
import {stdin as fstdin} from 'mock-stdin'
import {MyCommand} from '../../src/commands/my/command.js'
const stdin = fstdin()
describe('stdin test', () => {
it('mocks', async () => {
const {result} = await captureOutput(async () => {
setTimeout(() => stdin.send('whoa there!\n'), 10)
return MyCommand.run()
})
expect(result).to.equal('whoa there!')
})
})
.retries()
allowed you to retry the test. There is no direct replacement for this but most testing frameworks have functionality for this builtin.
.timeout
allowed to set a timeout for a test. There is no direct replacement for this but most testing frameworks have functionality for this builtin.
.nock
allowed you to use nock to mock HTTP requests. There is no direct replacement for since you can use nock directly if you need to.