A universal mocking class for Apex, built using the Apex Stub API, subject to all its limitations. The api design choices for this class have been driven by a desire to make mocking as simple as possible for developers to understand and implement. It favors fluency and readability above everything else. Consequently, trade-offs have been made such as the limitation noted towards the end of this Readme.
- Simply copy the
UniversalMocker.cls
to your org. Theexamples
folder merely serves as a reference.
-
Create an instance of
UniversalMocker
for each class you want to mock.UniversalMocker mockInstance = UniversalMocker.mock(AccountDBService.class);
-
Set the mock values you want to return for each method.
mockInstance.when('getOneAccount').thenReturn(mockAccount);
-
Use
withParamTypes
for overloaded methods.
mockInstance.when('getOneAccount').withParamTypes(new List<Type>{Id.class})
.thenReturn(mockAccount);
-
You can also set up a method to throw an exception
mockInstance.when('getOneAccount').thenThrow(new MyCustomException());
-
Create an instance of the class you want to mock.
AccountDBService mockDBService = (AccountDBService)mockInstance.createStub();
There might be instances where you may need the same method to mock different return values within the same test when testing utility methods or selector classes and such. You can specify different return values based on the call count in such cases
- Basic example
mockInstance.when('getOneAccount').thenReturnUntil(3,mockAccountOne).thenReturn(mockAccountTwo);
Here, mockAccountOne
is returned the first 3 times getOneAccount
is called. All subsequent calls to getOneAccount
will return mockAccountTwo
- You can also pair it with param types or to mock exceptions
mockInstance.when('getOneAccount').withParamTypes(new List<Type>{Id.class})
.thenReturnUntil(1,mockAccountOne)
.thenThrowUntil(3,mockException)
.thenReturn(mockAccountTwo);
Refer to the relevant unit tests for further clarity
Note: It is recommended that you end all setup method call chains with thenReturn
or thenThrow
There might be instances where you need to modify the original arguments passed into the function. A typical example
would be to set the Id
field of records passed into a method responsible for inserting them.
- Create a class that implements the
UniversalMocker.Mutator
interface. The interface has a single methodmutate
with the following signature.
void mutate(
Object stubbedObject, String stubbedMethodName,
List<Type> listOfParamTypes, List<Object> listOfArgs
);
Here's the method for setting fake ids on inserted records, in our example.
public void mutate(
Object stubbedObject, String stubbedMethodName,
List<Type> listOfParamTypes, List<Object> listOfArgs
) {
Account record = (Account) listOfArgs[0];
record.Id = this.getFakeId(Account.SObjectType);
}
- Pass in an instance of your implementation of the
Mutator
class to mutate the method arguments.
mockInstance.when('doInsert').mutateWith(dmlMutatorInstance).thenReturnVoid();
Check out the AccountDomainTest class for the full example.
You can call the mutateWith
method any number of times in succession, with the same or different mutator instances,
to create a chain of methods to mutate method arguments.
You can also use specific mutators based on call count. Multiple mutators with the same value of call count will be accumulated and applied in succession for all calls since the previous established call count.
For example, lets say you have a DescriptionMutator
class as shown below. It appends a given string to the Account Description
field.
//Adds a given suffix to account description
public class DescriptionMutator implements UniversalMocker.Mutator {
private String stringToAdd = '';
public DescriptionMutator(String stringToAdd) {
this.stringToAdd = stringToAdd;
}
public void mutate(Object stubbedObject, String stubbedMethodName, List<Type> listOfParamTypes, List<Object> listOfArgs) {
Account record = (Account) listOfArgs[0];
if (record.get('Description') != null) {
record.Description += this.stringToAdd;
} else {
record.Description = this.stringToAdd;
}
}
}
If you wanted to append the string 12
to the Account Description for the first 2 calls and then the string 3
for all subsequent
calls, your setup would look something like:
mockService.when(mockedMethodName).mutateUntil(2, new DescriptionMutator('1')).mutateUntil(2, new DescriptionMutator('2'))
.mutateWith(new DescriptionMutator('3'));
or
mockService.when(mockedMethodName).mutateUntil(2, new DescriptionMutator('12')).mutateWith(new DescriptionMutator('3'));
If you wanted to append the string 1
to the Account Description for the first call, the string 2
for the second call,
and string 3
for all subsequent calls, your setup would look as follows:
mockService.when(mockedMethodName).mutateUntil(1, new DescriptionMutator('1')).mutateUntil(2, new DescriptionMutator('2'))
.mutateWith(new DescriptionMutator('3'));
Check out the AccountDomainTest class for the full example.
-
Assert the exact number of times a method was called.
mockInstance.assertThat().method('getOneAccount').wasCalled(1); mockInstance.assertThat().method('getOneAccount').wasCalled(2);
-
Assert if the number of times a method was called was more or less than a given integer.
mockInstance.assertThat().method('getOneAccount').wasCalled(1,UniversalMocker.Times.OR_MORE); mockInstance.assertThat().method('getOneAccount').wasCalled(1,UniversalMocker.Times.OR_LESS);
-
Assert that a method was not called. This works both for methods that had mock return values set up before the test and for ones that didn't.
mockInstance.assertThat().method('dummyMethod').wasNeverCalled();
Note that
mockInstance.assertThat().method('dummyMethod').wasCalled(0,UniversalMocker.Times.EXACTLY);
would only work if you had a mock return value set up fordummyMethod
before running the test. -
Get the value of an argument passed into a method. Use
withParamTypes
for overloaded methods.mockInstance.forMethod('doInsert').andInvocationNumber(0).getValueOf('acct'); mockInstance.forMethod('doInsert').withParamTypes(new List<Type>{Account.class}).andInvocationNumber(0).getValueOf('acct');
Note: If you use
mutateWith
to mutate the original method arguments, the values returned here are the mutated arguments and not the original method arguments.
- Method and argument names are case-insensitive.
- If you don't have overloaded methods, it is recommended to not use
withParamTypes
. Conversely, if you do have overloaded methods, it is recommended that you do usewithParamTypes
for mocking as well as verification. - If you use
withParamTypes
for setting up the mock, you need to use it for verification and fetching method arguments as well. - It is highly recommended that you always verify the mocked method call counts to insulate against typos in method names being mocked and any future refactoring.
- The glaring limitation in the current version is the inability to mock methods with exact arguments, so this may not work if that's what you're looking to do.
- Although it is not recommended to test async behavior in unit tests since that is a platform feature, the library does support it.
Many thanks to my fellow SFXD members @jamessimone @ThieveryShoe @jlyon11 @elements for their feedback and contribution.