-
Notifications
You must be signed in to change notification settings - Fork 31
Developer Guide
The developer guide provides a comprehensive overview of the features of Mockatoo.
- Create a Mock
- Verifying Behaviour
- Basic Stubbing
- Argument Matchers
- Verifying exact number of invocations / at least once / never
- Advanced Stubbing with consecutive calls or callbacks
- Resetting a mock
- Spying on real objects (since 1.1.0)
- Partial Mocking (since 1.2.0)
- Mocking Properties that are Read or Write only (since 1.2.0)
- Known limitations
Import Mockatoo;
import mockatoo.Mockatoo;
Mocks can be generated from any Class, Interface or Typedef alias (not typedef structure)
var mockedClass = Mockatoo.mock(SomeClass);
var mockedInterface = Mockatoo.mock(SomeInterface);
A Mock class type will be generated that extends the Class (or Interface), stubbing all methods, and generate the code to instanciate the instance:
var mockedClass = new SomeClassMocked();
var mockedInterface = new SomeInterfaceMocked();
If a class requires Type paramaters then you can either use a typedef alias.
typedef FooBar = Foo<Bar>;
...
var mockFoo = Mockatoo.mock(FooBar);
Or pass through the types as a second paramater
var mockFoo = Mockatoo.mock(Foo, [Bar]);
Both these generates the equivalent expressions:
var mockFoo = new FooMocked<Bar>();
Note: These usages are required in order to circumvent limitation of compiler with generics. You cannot compile
Foo.doSomething(Array<String>)
You can also 'mock' a typedef structure, however the generated object is not technically a mock, so does not support verification or stubbing.
typedef SomeTypeDef = {name:String}
...
var mockedTypDef = Mockatoo.mock(SomeTypeDef);
Verification refers to validation of of if, and how often a method has been called (invoked) with particular argument values.
To verify that a method foo has been invoked:
Mockatoo.verify(mock).foo();
Mockatoo.verify(mock).foo("bar");
Mockatoo.verify(mock).foo("foo");
Mockatoo.verify(mock).foo("foo", true);
Once created, mock will remember all interactions. Then you can selectively verify whatever interaction you are interested in.
Mockatoo allows the behaviour of methods to be stubbed
Mockatoo.when(mock.foo("bar")).thenReturn("hello");
You can also specify an execption
Mockatoo.when(mock.foo("not bar")).thenThrow(new SomeApplicationException("not a bar"));
Mockatoo verifies argument values in natural syntax: by using an equals()
method. Sometimes, when extra flexibility is required then you might use argument matchers:
Matching against a type:
Mockatoo.verify(mock).foo(anyString);
Mockatoo.verify(mock).foo(anyInt);
Mockatoo.verify(mock).foo(anyFloat);
Mockatoo.verify(mock).foo(anyBool);
Mockatoo.verify(mock).foo(anyObject); //anonymous data structures only (not class instances)
Mockatoo.verify(mock).foo(anyIterator); // any Iterator or Iterable (e.g. Array, Hash, etc)
Mockatoo.verify(mock).foo(anyEnum); // any enum value of any enum;
Matching against a specific class or enum:
Mockatoo.verify(mock).foo(enumOf(Color)); // any enum value of Enum Colour
Mockatoo.verify(mock).foo(instanceOf(SomeClass)); // any instance of SomeClass (or it's subclasses)
Wildcard matches
Mockatoo.verify(mock).foo(any); //any value (including null)
Mockatoo.verify(mock).foo(isNotNull); // any non null value
Mockatoo.verify(mock).foo(isNull); // same as verifying 'null')
Custom matching function
var f = function(value:Dynamic):Bool
{
...
}
Mockatoo.verify(mock).foo(customMatcher(f));
Verifications use natural language to specify the minimum and maximum times a method was invoked with specific arguments
Mockatoo.verify(mock, times(2)).foo();
Mockatoo.verify(mock, atLeast(2)).foo();
Mockatoo.verify(mock, atLeastOnce).foo();
Mockatoo.verify(mock, never).foo();
Note: Default mode is times(1);
Stubbing is chainable, so you can stub with different behavior for consecutive method calls.
This can be achieved by providing multiple return (or thrown) values:
Mockatoo.when(mock.someMethod()).thenReturn("a", "b");
Mockatoo.when(mock.someMethod()).theThrow("one", "two");
Combinations can also be chained together
Mockatoo.when(mock.someMethod()).thenReturn("one", "two").thenThrow("empty");
The last stubbing (e.g: thenThrow("empty")) determines the behavior for any further consecutive calls.
Custom Callback Stub
You can also set a custom callback when a method is invoked
var f = function(args:Array<Dynamic>)
{
if(Std.is(args[0], String) return args[0].charAt(0) == "b";
}
Mockatoo.when(mock.foo("bar")).thenCall(f);
Mockatoo supports several other stubbing options defined later in this document
Mockatoo.when(mock.foo("bar")).thenStub();//sets default stubbing behaviour
Mockatoo.when(mock.foo("bar")).thenCallRealMethod(); //calls out to real method
You can reset a mock to remove any custom stubs and/or verifications
Mockatoo.reset(mock);
Since 1.1.0
You can create spies of real objects. When you use the spy then the real methods are called unless a method is explicitly stubbed.
Real spies should be used carefully and occasionally, for example when dealing with legacy code.
var mock = Mockatoo.spy(SomeClass);
mock.someMethod();//calls real method
Mockatoo.when(mock.someMethod()).thenReturn("one");
mock.someMethod();//returns stub value;
Mockatoo.verify(mock, times(2)).someMethod(); //both calls recorded
You can also quickly convert a real method to the default stub behaviour:
Mockatoo.when(mock.someMethod()).thenStub();
mock.someMethod();//returns default mock value (usually null)
There are several limitations to spying:
-
The real constructor cannot be accessed directly, and will always be called with the default values for each argument type - e.g:
super(null,null,null);
-
Interfaces cannot be spied (Mockatoo will ignore these and used normal mocking instead)
Since 1.2.0
You can explicitly set a mock to call out to the underlying real method
Mockatoo.when(mock.someMethod()).thenCallRealMethod();
Like spies, you can also reset them to their mocked behaviour
Mockatoo.when(mock.someMethod()).thenStub();
Since 1.2.0
A property that is read/write only, or calls out to getter/setter functions can be stubbed to a specific return value:
Mockatoo.when(mock.someReadOnlyProperty).thenReturn("foo");
If a property has a getter function, stubbing a return value will automatically stub the underlying getter method - the equivalent of:
Mockatoo.when(mock.get_someReadOnlyProperty()).thenReturn("foo");
Getters can also be stubbed to a custom callback:
Mockatoo.when(mock.someGetter).thenCall(f);
Both Getters and Setters can also be stubbed with an exception:
Mockatoo.when(mock.someGetter).thenThrow("foo");
Mockatoo.when(mock.someSetter).thenThrow("foo");
var result = mock.someGetter;//throws exception
mock.someSetter = "a";//throws exception
Note: If a property has both a getter and setter, both getting and setting the property with trigger the exception
There are some limitations to property stubbing:
- stubbed properties cannot be chained (thenReturn("foo", "bar", "etc"))
- only getters support stubbed callbacks (
thenCall(f)
) - only getter and setters support stubbed exceptions (
thenThrow(xxx)
)
In Haxe 2.10 inlined methods cannot be mocked. Mockatoo will print a compiler warning and skip affected fields.
inline public function someMethod()
{
..///
}
In Haxe 2.11 (svn) this has been resolved (http://code.google.com/p/haxe/issues/detail?id=1231) and requires the --no-inline
compiler flag
Mockatoo supports overriding @:final classes and methods in targets other than flash.
@:final public function someMethod()
{
..///
}
Mocking a @:final class in flash will throw a compiler error.
Mocking a @:final method in flash will generate a compiler warning and leave the real method untouched.
See http://code.google.com/p/haxe/issues/detail?id=1246 for more details.
Some classes may expect arguments, or return values typed to a private Class, Enum or Typedef. For example, mocking haxe.Http
will fail to compile on the neko target due to a reference to private typedef AbstractSocket
Mockatoo.mock(haxe.Http);
This is due to an edge case with tink_macros that is fixed in tink_macros 1.2.1;