// Type definitions for JsMockito 1.0.4 // Project: http://github.com/chrisleishman/jsmockito // Definitions by: Karl Bennett // Definitions: https://github.com/DefinitelyTyped/DefinitelyTyped /** * Top-level module for the JsMockito mocking library. * * @author Karl Bennett */ /** *

Contents

* *
    *
  1. Let's verify some behaviour!
  2. *
  3. How about some stubbing?
  4. *
  5. Matching Arguments
  6. *
  7. Verifying exact number of invocations / at least once / * never
  8. *
  9. Matching the context ('this')
  10. *
  11. Making sure interactions never happened on a mock
  12. *
  13. Finding redundant invocations
  14. *
* *

In the following examples object mocking is done with Array as this is * well understood, although you probably wouldn't mock this in normal test * development.

* *

1. Let's verify some behaviour!

* *

For an object:

*
 * //mock creation
 * var mockedArray = mock(Array);
 *
 * //using mock object
 * mockedArray.push("one");
 * mockedArray.reverse();
 *
 * //verification
 * verify(mockedArray).push("one");
 * verify(mockedArray).reverse();
 * 
* *

For a function:

*
 * //mock creation
 * var mockedFunc = mockFunction();
 *
 * //using mock function
 * mockedFunc('hello world');
 * mockedFunc.call(this, 'foobar');
 * mockedFunc.apply(this, [ 'barfoo' ]);
 *
 * //verification
 * verify(mockedFunc)('hello world');
 * verify(mockedFunc)('foobar');
 * verify(mockedFunc)('barfoo');
 * 
* *

Once created a mock will remember all interactions. Then you selectively * verify whatever interactions you are interested in.

* *

2. How about some stubbing?

* *

For an object:

*
 * var mockedArray = mock(Array);
 *
 * //stubbing
 * when(mockedArray).slice(0).thenReturn('f');
 * when(mockedArray).slice(1).thenThrow('An exception');
 * when(mockedArray).slice(2).then(function() { return 1+2 });
 *
 * //the following returns "f"
 * assertThat(mockedArray.slice(0), equalTo('f'));
 *
 * //the following throws exception 'An exception'
 * var ex = undefined;
 * try {
 *   mockedArray.slice(1);
 * } catch (e) {
 *   ex = e;
 * }
 * assertThat(ex, equalTo('An exception');
 *
 * //the following invokes the stub method, which returns 3
 * assertThat(mockedArray.slice(2), equalTo(3));
 *
 * //the following returns undefined as slice(999) was not stubbed
 * assertThat(mockedArray.slice(999), typeOf('undefined'));
 *
 * //stubs can take multiple values to return in order (same for 'thenThrow' and 'then' as well)
 * when(mockedArray).pop().thenReturn('a', 'b', 'c');
 * assertThat(mockedArray.pop(), equalTo('a'));
 * assertThat(mockedArray.pop(), equalTo('b'));
 * assertThat(mockedArray.pop(), equalTo('c'));
 * assertThat(mockedArray.pop(), equalTo('c'));
 *
 * //stubs can also be chained to return values in order
 * when(mockedArray).unshift().thenReturn('a').thenReturn('b').then(function() { return 'c' });
 * assertThat(mockedArray.unshift(), equalTo('a'));
 * assertThat(mockedArray.unshift(), equalTo('b'));
 * assertThat(mockedArray.unshift(), equalTo('c'));
 * assertThat(mockedArray.unshift(), equalTo('c'));
 *
 * //stub matching can overlap, allowing for specific cases and defaults
 * when(mockedArray).slice(3).thenReturn('abcde');
 * when(mockedArray).slice(3, lessThan(0)).thenReturn('edcba');
 * assertThat(mockedArray.slice(3, -1), equalTo('edcba'));
 * assertThat(mockedArray.slice(3, 1), equalTo('abcde'));
 * assertThat(mockedArray.slice(3), equalTo('abcde'));
 *
 * //can also verify a stubbed invocation, although this is usually redundant
 * verify(mockedArray).slice(0);
 * 
* *

For a function:

*
 * var mockedFunc = mockFunction();
 *
 * //stubbing
 * when(mockedFunc)(0).thenReturn('f');
 * when(mockedFunc)(1).thenThrow('An exception');
 * when(mockedFunc)(2).then(function() { return 1+2 });
 *
 * //the following returns "f"
 * assertThat(mockedFunc(0), equalTo('f'))
 *
 * //following throws exception 'An exception'
 * mockedFunc(1);
 * //the following throws exception 'An exception'
 * var ex = undefined;
 * try {
 *   mockedFunc(1);
 * } catch (e) {
 *   ex = e;
 * }
 * assertThat(ex, equalTo('An exception');
 *
 * //the following invokes the stub method, which returns 3
 * assertThat(mockedFunc(2), equalTo(3));
 *
 * //following returns undefined as mockedFunc(999) was not stubbed
 * assertThat(mockedFunc(999), typeOf('undefined'));
 *
 * //stubs can take multiple values to return in order (same for 'thenThrow' and 'then' as well)
 * when(mockedFunc)(3).thenReturn('a', 'b', 'c');
 * assertThat(mockedFunc(3), equalTo('a'));
 * assertThat(mockedFunc(3), equalTo('b'));
 * assertThat(mockedFunc(3), equalTo('c'));
 * assertThat(mockedFunc(3), equalTo('c'));
 *
 * //stubs can also be chained to return values in order
 * when(mockedFunc)(4).thenReturn('a').thenReturn('b').then(function() { return 'c' });
 * assertThat(mockedFunc(4), equalTo('a'));
 * assertThat(mockedFunc(4), equalTo('b'));
 * assertThat(mockedFunc(4), equalTo('c'));
 * assertThat(mockedFunc(4), equalTo('c'));
 *
 * //stub matching can overlap, allowing for specific cases and defaults
 * when(mockedFunc)(5).thenReturn('abcde')
 * when(mockedFunc)(5, lessThan(0)).thenReturn('edcba')
 * assertThat(mockedFunc(5, -1), equalTo('edcba'))
 * assertThat(mockedFunc(5, 1), equalTo('abcde'))
 * assertThat(mockedFunc(5), equalTo('abcde'))
 *
 * //can also verify a stubbed invocation, although this is usually redundant
 * verify(mockedFunc)(0);
 * 
* * * *

3. Matching Arguments

* *

JsMockito verifies arguments using * JsHamcrest matchers. * *

 * var mockedArray = mock(Array);
 * var mockedFunc = mockFunction();
 *
 * //stubbing using JsHamcrest
 * when(mockedArray).slice(lessThan(10)).thenReturn('f');
 * when(mockedFunc)(containsString('world')).thenReturn('foobar');
 *
 * //following returns "f"
 * mockedArray.slice(5);
 *
 * //following returns "foobar"
 * mockedFunc('hello world');
 *
 * //you can also use matchers in verification
 * verify(mockedArray).slice(greaterThan(4));
 * verify(mockedFunc)(equalTo('hello world'));
 *
 * //if not specified then the matcher is anything(), thus either of these
 * //will match an invocation with a single argument
 * verify(mockedFunc)();
 * verify(mockedFunc)(anything());
 * 
* * * *

4. Verifying exact number of invocations / at least once / * never

* *
 * var mockedArray = mock(Array);
 * var mockedFunc = mockFunction();
 *
 * mockedArray.slice(5);
 * mockedArray.slice(6);
 * mockedFunc('a');
 * mockedFunc('b');
 *
 * //verification of multiple matching invocations
 * verify(mockedArray, times(2)).slice(anything());
 * verify(mockedFunc, times(2))(anything());
 *
 * //the default is times(1), making these are equivalent
 * verify(mockedArray, times(1)).slice(5);
 * verify(mockedArray).slice(5);
 * 
* *

5. Matching the context ('this')

* * Functions can be invoked with a specific context, using the 'call' or * 'apply' methods. JsMockito mock functions (and mock object methods) * will remember this context and verification/stubbing can match on it. * *

For a function:

*
 * var mockedFunc = mockFunction();
 * var context1 = {};
 * var context2 = {};
 *
 * when(mockedFunc).call(equalTo(context2), anything()).thenReturn('hello');
 *
 * mockedFunc.call(context1, 'foo');
 * //the following returns 'hello'
 * mockedFunc.apply(context2, [ 'bar' ]);
 *
 * verify(mockedFunc).apply(context1, [ 'foo' ]);
 * verify(mockedFunc).call(context2, 'bar');
 * 
* *

For object method invocations, the context is usually the object itself. * But sometimes people do strange things, and you need to test it - so * the same approach can be used for an object:

*
 * var mockedArray = mock(Array);
 * var otherContext = {};
 *
 * when(mockedArray).slice.call(otherContext, 5).thenReturn('h');
 *
 * //the following returns 'h'
 * mockedArray.slice.apply(otherContext, [ 5 ]);
 *
 * verify(mockedArray).slice.call(equalTo(otherContext), 5);
 * 
* * * *

6. Making sure interactions never happened on a mock

* *
 * var mockOne = mock(Array);
 * var mockTwo = mock(Array);
 * var mockThree = mockFunction();
 *
 * //only mockOne is interacted with
 * mockOne.push(5);
 *
 * //verify a method was never called
 * verify(mockOne, never()).unshift('a');
 *
 * //verify that other mocks were not interacted with
 * verifyZeroInteractions(mockTwo, mockThree);
 * 
* *

7. Finding redundant invocations

* *
 * var mockArray = mock(Array);
 *
 * mockArray.push(5);
 * mockArray.push(8);
 *
 * verify(mockArray).push(5);
 *
 * // following verification will fail
 * verifyNoMoreInteractions(mockArray);
 * 
*/ declare namespace JsMockito { /** * Library version. */ export var version: string; /** * Builder for a textual description. */ export class JsMockitoStubBuilder { /** * Provide functions to be run in place of the mocked method. * * @param func Functions to be run in order of execution. * @return {JsMockitoStubBuilder} Itself for method chaining */ then(...func: ((obj: any) => any)[]): JsMockitoStubBuilder; /** * Provide values to be returned by the mocked function. * * @param obj Values to be returned in order of execution. * @return {JsMockitoStubBuilder} Itself for method chaining */ thenReturn(...obj: any[]): JsMockitoStubBuilder; /** * Provide exceptions to be thrown by the mocked function. * * @param obj Exceptions to be thrown in order of execution. * @return {JsMockitoStubBuilder} Itself for method chaining */ thenThrow(...obj: Error[]): JsMockitoStubBuilder; } /** * Used to verify how many times a function of method is called. */ export interface Verifier { } /** * Test if a given variable is a mock * * @param maybeMock An object * @return {boolean} true if the variable is a mock */ export function isMock(maybeMock: any): boolean; /** * Add a stub for a mock object method or mock function * * @param mock A mock object or mock anonymous function * @return {T} A stub builder on which the method or function to be stubbed can be invoked */ export function when(mock: T): T; /** * Verify that a mock object method or mock function was invoked * * @param mock A mock object or mock anonymous function * @param verifier Optional JsMockito.Verifier instance (default: JsMockito.Verifiers.once()) * @return {T} A verifier on which the method or function to be verified can be invoked */ export function verify(mock: T): T; export function verify(mock: T, verifier: Verifier): T; /** * Verify that no mock object methods or the mock function were ever invoked * * @param mock A mock object or mock anonymous function (multiple accepted) */ export function verifyZeroInteractions(...mock: any[]): void; /** * Verify that no mock object method or mock function invocations remain * unverified * * @param mock A mock object or mock anonymous function (multiple accepted) */ export function verifyNoMoreInteractions(...mock: any[]): void; /** * Create a mock that proxies a real function or object. All un-stubbed * invocations will be passed through to the real implementation, but can * still be verified. * * @param delegate A 'real' (concrete) object or function that the mock will delegate unstubbed invocations to * @return {T} A mock object (as per mock) or mock function (as per mockFunction) */ export function spy(delegate: T): T; /** * Create a mockable and stubbable anonymous function. * *

Once created, the function can be invoked and will return undefined for * any interactions that do not match stub declarations.

* *
     * var mockFunc = JsMockito.mockFunction();
     * JsMockito.when(mockFunc).call(anything(), 1, 5).thenReturn(6);
     * mockFunc(1, 5); // result is 6
     * JsMockito.verify(mockFunc)(1, greaterThan(2));
     * 
* * @param funcName The name of the mock function to use in messages (defaults to 'func') * @param delegate The function to delegate unstubbed calls to (optional) * @return {T} an anonymous function */ export function mockFunction(): Function; export function mockFunction(funcName: string): Function; export function mockFunction(funcName: string, delegate: Function): Function; /** * Create a mockable and stubbable objects. * *

A mock is created with the constructor for an object as an argument. * Once created, the mock object will have all the same methods as the source * object which, when invoked, will return undefined by default.

* *

Stub declarations may then be made for these methods to have them return * useful values or perform actions when invoked.

* *
     * MyObject = function() {
     *   this.add = function(a, b) { return a + b }
     * };
     *
     * var mockObj = JsMockito.mock(MyObject);
     * mockObj.add(5, 4); // result is undefined
     *
     * JsMockito.when(mockFunc).add(1, 2).thenReturn(6);
     * mockObj.add(1, 2); // result is 6
     *
     * JsMockito.verify(mockObj).add(1, greaterThan(2)); // ok
     * JsMockito.verify(mockObj).add(1, equalTo(2)); // ok
     * JsMockito.verify(mockObj).add(1, 4); // will throw an exception
     * 
* * @param Obj {function} the constructor for the object to be mocked * @return {object} a mock object */ export function mock(Obj: { new(): T ;}): T; namespace Verifiers { /** * Test that a invocation never occurred. For example: *
         * verify(mock, never()).method();
         * 
* @see JsMockito.Verifiers.times(0) */ export function never(): Verifier; /** Test that no interaction were made on the mock. For example: *
         * verify(mock, zeroInteractions());
         * 
* @see JsMockito.verifyZeroInteractions() */ export function zeroInteractions(): Verifier; /** Test that no further interactions remain unverified on the mock. For * example: *
         * verify(mock, noMoreInteractions());
         * 
* @see JsMockito.verifyNoMoreInteractions() */ export function noMoreInteractions(): Verifier; /** * Test that an invocation occurred a specific number of times. For example: *
         * verify(mock, times(2)).method();
         * 
* * @param wanted The number of desired invocations */ export function times(wanted: number): Verifier; /** * Test that an invocation occurred exactly once. For example: *
         * verify(mock, once()).method();
         * 
* This is the default verifier. * @see JsMockito.Verifiers.times(1) */ export function once(): Verifier; } namespace Integration { /** * Import the public JsMockito API into the specified object (namespace) * * @param {object} target An object (namespace) that will be populated with * the functions from the public JsMockito API */ export function importTo(target: any): void; /** * Make the public JsMockito API available in Screw.Unit * @see JsMockito.Integration.importTo(Screw.Matchers) */ export function screwunit(): void; /** * Make the public JsMockito API available to JsTestDriver * @see JsMockito.Integration.importTo(window) */ export function JsTestDriver(): void; /** * Make the public JsMockito API available to JsUnitTest * @see JsMockito.Integration.importTo(JsUnitTest.Unit.Testcase.prototype) */ export function JsUnitTest(): void; /** * Make the public JsMockito API available to YUITest * @see JsMockito.Integration.importTo(window) */ export function YUITest(): void; /** * Make the public JsMockito API available to QUnit * @see JsMockito.Integration.importTo(window) */ export function QUnit(): void; /** * Make the public JsMockito API available to jsUnity * @see JsMockito.Integration.importTo(jsUnity.env.defaultScope) */ export function jsUnity(): void; /** * Make the public JsMockito API available to jSpec * @see JsMockito.Integration.importTo(jSpec.defaultContext) */ export function jSpec(): void; } } // // Functions that are copied by JsMockito.Integration.importTo() to the global scope are repeated here. // /** * Test if a given variable is a mock * * @param maybeMock An object * @return {boolean} true if the variable is a mock */ declare function isMock(maybeMock: any): boolean; /** * Add a stub for a mock object method or mock function * * @param mock A mock object or mock anonymous function * @return {T} A stub builder on which the method or function to be stubbed can be invoked */ declare function when(mock: T): T; /** * Verify that a mock object method or mock function was invoked * * @param mock A mock object or mock anonymous function * @param verifier Optional JsMockito.Verifier instance (default: JsMockito.Verifiers.once()) * @return {T} A verifier on which the method or function to be verified can be invoked */ declare function verify(mock: T): T; declare function verify(mock: T, verifier: JsMockito.Verifier): T; /** * Verify that no mock object methods or the mock function were ever invoked * * @param mock A mock object or mock anonymous function (multiple accepted) */ declare function verifyZeroInteractions(...mock: any[]): void; /** * Verify that no mock object method or mock function invocations remain * unverified * * @param mock A mock object or mock anonymous function (multiple accepted) */ declare function verifyNoMoreInteractions(...mock: any[]): void; /** * Create a mock that proxies a real function or object. All un-stubbed * invocations will be passed through to the real implementation, but can * still be verified. * * @param delegate A 'real' (concrete) object or function that the mock will delegate unstubbed invocations to * @return {T} A mock object (as per mock) or mock function (as per mockFunction) */ declare function spy(delegate: T): T; /** * Create a mockable and stubbable anonymous function. * *

Once created, the function can be invoked and will return undefined for * any interactions that do not match stub declarations.

* *
 * var mockFunc = JsMockito.mockFunction();
 * JsMockito.when(mockFunc).call(anything(), 1, 5).thenReturn(6);
 * mockFunc(1, 5); // result is 6
 * JsMockito.verify(mockFunc)(1, greaterThan(2));
 * 
* * @param funcName The name of the mock function to use in messages (defaults to 'func') * @param delegate The function to delegate unstubbed calls to (optional) * @return {T} an anonymous function */ declare function mockFunction(): Function; declare function mockFunction(funcName: string): Function; declare function mockFunction(funcName: string, delegate: Function): Function; /** * Create a mockable and stubbable objects. * *

A mock is created with the constructor for an object as an argument. * Once created, the mock object will have all the same methods as the source * object which, when invoked, will return undefined by default.

* *

Stub declarations may then be made for these methods to have them return * useful values or perform actions when invoked.

* *
 * MyObject = function() {
     *   this.add = function(a, b) { return a + b }
     * };
 *
 * var mockObj = JsMockito.mock(MyObject);
 * mockObj.add(5, 4); // result is undefined
 *
 * JsMockito.when(mockFunc).add(1, 2).thenReturn(6);
 * mockObj.add(1, 2); // result is 6
 *
 * JsMockito.verify(mockObj).add(1, greaterThan(2)); // ok
 * JsMockito.verify(mockObj).add(1, equalTo(2)); // ok
 * JsMockito.verify(mockObj).add(1, 4); // will throw an exception
 * 
* * @param Obj {function} the constructor for the object to be mocked * @return {object} a mock object */ declare function mock(Obj: { new(): T ;}): T; /** * Test that a invocation never occurred. For example: *
 * verify(mock, never()).method();
 * 
* @see JsMockito.Verifiers.times(0) */ declare function never(): JsMockito.Verifier; /** Test that no interaction were made on the mock. For example: *
 * verify(mock, zeroInteractions());
 * 
* @see JsMockito.verifyZeroInteractions() */ declare function zeroInteractions(): JsMockito.Verifier; /** Test that no further interactions remain unverified on the mock. For * example: *
 * verify(mock, noMoreInteractions());
 * 
* @see JsMockito.verifyNoMoreInteractions() */ declare function noMoreInteractions(): JsMockito.Verifier; /** * Test that an invocation occurred a specific number of times. For example: *
 * verify(mock, times(2)).method();
 * 
* * @param wanted The number of desired invocations */ declare function times(wanted: number): JsMockito.Verifier; /** * Test that an invocation occurred exactly once. For example: *
 * verify(mock, once()).method();
 * 
* This is the default verifier. * @see JsMockito.Verifiers.times(1) */ declare function once(): JsMockito.Verifier;