You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 

199 lines
5.0 KiB

'use strict';
const assert = require('core-assert');
const deepEqual = require('lodash.isequal');
const observableToPromise = require('observable-to-promise');
const indentString = require('indent-string');
const isObservable = require('is-observable');
const isPromise = require('is-promise');
const jestSnapshot = require('jest-snapshot');
const snapshotState = require('./snapshot-state');
const x = module.exports;
const noop = () => {};
Object.defineProperty(x, 'AssertionError', {value: assert.AssertionError});
function create(val, expected, operator, msg, fn) {
return {
actual: val,
expected,
message: msg || ' ',
operator,
stackStartFunction: fn
};
}
function test(ok, opts) {
if (!ok) {
const err = new assert.AssertionError(opts);
err.showOutput = ['fail', 'throws', 'notThrows'].indexOf(err.operator) === -1;
throw err;
}
}
x.pass = msg => {
test(true, create(true, true, 'pass', msg, x.pass));
};
x.fail = msg => {
msg = msg || 'Test failed via t.fail()';
test(false, create(false, false, 'fail', msg, x.fail));
};
x.truthy = (val, msg) => {
test(val, create(val, true, '==', msg, x.truthy));
};
x.falsy = (val, msg) => {
test(!val, create(val, false, '==', msg, x.falsy));
};
x.true = (val, msg) => {
test(val === true, create(val, true, '===', msg, x.true));
};
x.false = (val, msg) => {
test(val === false, create(val, false, '===', msg, x.false));
};
x.is = (val, expected, msg) => {
test(val === expected, create(val, expected, '===', msg, x.is));
};
x.not = (val, expected, msg) => {
test(val !== expected, create(val, expected, '!==', msg, x.not));
};
x.deepEqual = (val, expected, msg) => {
test(deepEqual(val, expected), create(val, expected, '===', msg, x.deepEqual));
};
x.notDeepEqual = (val, expected, msg) => {
test(!deepEqual(val, expected), create(val, expected, '!==', msg, x.notDeepEqual));
};
x.throws = (fn, err, msg) => {
if (isObservable(fn)) {
fn = observableToPromise(fn);
}
if (isPromise(fn)) {
return fn
.then(() => {
x.throws(noop, err, msg);
}, fnErr => {
return x.throws(() => {
throw fnErr;
}, err, msg);
});
}
if (typeof fn !== 'function') {
throw new TypeError('t.throws must be called with a function, Promise, or Observable');
}
try {
if (typeof err === 'string') {
const errMsg = err;
err = err => err.message === errMsg;
}
let result;
assert.throws(() => {
try {
fn();
} catch (err) {
result = err;
throw err;
}
}, err, msg);
return result;
} catch (err) {
test(false, create(err.actual, err.expected, 'throws', err.message, x.throws));
}
};
x.notThrows = (fn, msg) => {
if (isObservable(fn)) {
fn = observableToPromise(fn);
}
if (isPromise(fn)) {
return fn
.catch(err => {
x.notThrows(() => {
throw err;
}, msg);
});
}
if (typeof fn !== 'function') {
throw new TypeError('t.notThrows must be called with a function, Promise, or Observable');
}
try {
assert.doesNotThrow(fn, msg);
} catch (err) {
test(false, create(err.actual, err.expected, 'notThrows', err.message, x.notThrows));
}
};
x.regex = (contents, regex, msg) => {
test(regex.test(contents), create(regex, contents, '===', msg, x.regex));
};
x.notRegex = (contents, regex, msg) => {
test(!regex.test(contents), create(regex, contents, '!==', msg, x.notRegex));
};
x.ifError = (err, msg) => {
test(!err, create(err, 'Error', '!==', msg, x.ifError));
};
x._snapshot = function (tree, optionalMessage, match, snapshotStateGetter) {
// Set defaults - this allows tests to mock deps easily
const toMatchSnapshot = match || jestSnapshot.toMatchSnapshot;
const getState = snapshotStateGetter || snapshotState.get;
const state = getState();
const context = {
dontThrow() {},
currentTestName: this.title,
snapshotState: state
};
// Symbols can't be serialized and saved in a snapshot, that's why tree
// is saved in the `__ava_react_jsx` prop, so that JSX can be detected later
const serializedTree = tree.$$typeof === Symbol.for('react.test.json') ? {__ava_react_jsx: tree} : tree; // eslint-disable-line camelcase
const result = toMatchSnapshot.call(context, JSON.stringify(serializedTree));
let message = 'Please check your code or --update-snapshots';
if (optionalMessage) {
message += '\n\n' + indentString(optionalMessage, 2);
}
state.save();
let expected;
if (result.expected) {
// JSON in a snapshot is surrounded with `"`, because jest-snapshot
// serializes snapshot values too, so it ends up double JSON encoded
expected = JSON.parse(result.expected.slice(1).slice(0, -1));
// Define a `$$typeof` symbol, so that pretty-format detects it as React tree
if (expected.__ava_react_jsx) { // eslint-disable-line camelcase
expected = expected.__ava_react_jsx; // eslint-disable-line camelcase
Object.defineProperty(expected, '$$typeof', {value: Symbol.for('react.test.json')});
}
}
test(result.pass, create(tree, expected, 'snapshot', message, x.snapshot));
};
x.snapshot = function (tree, optionalMessage) {
x._snapshot.call(this, tree, optionalMessage);
};