'use strict'; describe('minErr', function() { var supportStackTraces = function() { var e = new Error(); return isDefined(e.stack); }; var emptyTestError = minErr(), testError = minErr('test'); it('should return an Error factory', function() { var myError = testError('test', 'Oops'); expect(myError instanceof Error).toBe(true); }); it('should generate stack trace at the frame where the minErr instance was called', function() { var myError; function someFn() { function nestedFn() { myError = testError('fail', "I fail!"); } nestedFn(); } someFn(); // only Chrome, Firefox have stack if (!supportStackTraces()) return; expect(myError.stack).toMatch(/^[.\s\S]+nestedFn[.\s\S]+someFn.+/); }); it('should interpolate string arguments without quotes', function() { var myError = testError('1', 'This {0} is "{1}"', 'foo', 'bar'); expect(myError.message).toMatch(/^\[test:1\] This foo is "bar"/); }); it('should interpolate non-string arguments', function() { var arr = [1, 2, 3], obj = {a: 123, b: 'baar'}, anonFn = function(something) { return something; }, namedFn = function foo(something) { return something; }, myError; myError = testError('26', 'arr: {0}; obj: {1}; anonFn: {2}; namedFn: {3}', arr, obj, anonFn, namedFn); expect(myError.message).toContain('[test:26] arr: [1,2,3]; obj: {"a":123,"b":"baar"};'); // IE does not add space after "function" expect(myError.message).toMatch(/anonFn: function\s?\(something\);/); expect(myError.message).toContain('namedFn: function foo(something)'); }); it('should not suppress falsy objects', function() { var myError = testError('26', 'false: {0}; zero: {1}; null: {2}; undefined: {3}; emptyStr: {4}', false, 0, null, undefined, ''); expect(myError.message). toMatch(/^\[test:26\] false: false; zero: 0; null: null; undefined: undefined; emptyStr: /); }); it('should handle arguments that are objects with cyclic references', function() { var a = { b: { } }; a.b.a = a; var myError = testError('26', 'a is {0}', a); expect(myError.message).toMatch(/a is {"b":{"a":"<>"}}/); }); it('should preserve interpolation markers when fewer arguments than needed are provided', function() { // this way we can easily see if we are passing fewer args than needed var foo = 'Fooooo', myError = testError('26', 'This {0} is {1} on {2}', foo); expect(myError.message).toMatch(/^\[test:26\] This Fooooo is \{1\} on \{2\}/); }); it('should pass through the message if no interpolation is needed', function() { var myError = testError('26', 'Something horrible happened!'); expect(myError.message).toMatch(/^\[test:26\] Something horrible happened!/); }); it('should include a namespace in the message only if it is namespaced', function() { var myError = emptyTestError('26', 'This is a {0}', 'Foo'); var myNamespacedError = testError('26', 'That is a {0}', 'Bar'); expect(myError.message).toMatch(/^\[26\] This is a Foo/); expect(myNamespacedError.message).toMatch(/^\[test:26\] That is a Bar/); }); it('should accept an optional 2nd argument to construct custom errors', function() { var normalMinErr = minErr('normal'); expect(normalMinErr('acode', 'aproblem') instanceof TypeError).toBe(false); var typeMinErr = minErr('type', TypeError); expect(typeMinErr('acode', 'aproblem') instanceof TypeError).toBe(true); }); });