Moqqer is an AutoMocker for Moq.
It automatically injects mocks into the constructor of a class and lets you retrieve the injected mocks later for verification.
This helps to ease the pain of creating mocks each time, as well as time saving when you change the constructors or refactor code.
Moqqer also automatically stubs methods of an interface to return Mocks of the return type or some defaults (where applicable). See the section on default Mock value below for more info.
Moqqer provides a custom IEnumerable.AsQueryable
implemention which null
guards each expression to make sure empty object graphs do not cause NullReferenceException
as when using EF/Linq2Sql this would be handled gracefully.
// 1 - First create an instance of Moqqer which will
// act as a container for Mocks and Objects.
_moq = new Moqqer();
// 2 - Get Moqqer to create an instance of some class you want to test.
// It will auto inject mocks in the its constructor.
_subject = _moq.Create<SomeClass>();
// 3 - Call the mothod you want to test on SomeClass
_subject.CallA(); // Calls private field IDependencyA.Call()
// 4 - Verify a mock that was auto injected was called.
_moq.Of<IDepencyA>()
.Verify(x => x.Call(), Times.Once);
// Alternatively use the Verify extension method
_moq.Verify<IDepencyA>(x => x.Call())
.Once();
// 5 - Test and Refactor to your hearts content
// without worrying about changing the constructor calls!
using Moqqer.Namespace
[TestFixture]
public class IntroExampleTests
{
private Moqqer _moq;
private SomeClass _subject;****
// Setup run per test.
[SetUp]
public void A_TestInit()
{
// 1. Create instance of Moqqer
_moq = new Moqqer();
// Create an instance of SomeClass
_subject = _moq.Create<SomeClass>();
}
[Test]
public void DoWork_CallsRepositoryGetData()
{
_subject.CallA(); // Calls private field IDependencyA.Call()
_moq.Of<IDepencyA>()
.Verify(x => x.Call(), Times.Once);
}
}
Use the Moqqer.Create<T>()
method in order to get Moqqer to create an instance of <T>
and inject Mocks in the constructor.
var subject = _moq.Create<SomeClass>();
var depencyA = _moq.Of<IDepencyA>().Object;
var depencyB = _moq.Of<IDepencyB>().Object;
var expected = _moq.Of<IMockSetup>().Object;
subject.A.Should().BeSameAs(depencyA);
subject.B.Should().BeSameAs(depencyB);
subject.Mock.Should().BeSameAs(expected);
When Moqqer is creating a Mock<T>
which in it's constructor has a concrete class, as long as that concrete class has a default constructor, it will create an instance of that class and keep a reference to it in the Objecets
dictionary. You can access these using the _moqqer.Object<T>()
method.
var subject = _moq.Create<ClassHavingParameterlessConcreteClass>();
var classObject = subject.Class;
var moqedObject = _moq.Object<ParameterlessClass>();
classObject.Should().BeSameAs(moqedObject);
By default Moqqer will inject Mocks into types which are Mockable. For types which aren't Mockable, Moqqer will resolve those to the type or object as per the Non Mockable Defaults below.
public class StringConstructorClass
{
public string Text { get; set; }
public StringConstructorClass(string text)
{
Text = text;
}
}
[Test]
public void DefaultObjectInjection()
{
var subject = _moq.Create<StringConstructorClass>();
subject.Text.Should().Be(string.Empty);
}
By default Moqqer will inject Mocks into types which are Mockable. For types which aren't Mockable, Moqqer will resolve those to the type or object as per the Non Mockable Defaults below.
e.g. This helps with testing Entity Framework repository/expression logic allowing you to use Lists
as if they were the backing Tables.
var item = new Leaf(25);
_moq.List<Leaf>()
.Add(item);
// Contains `IQueryable<Leaf> Leaves { get; set; }`
var ctx = _moq.Of<IContext>().Object;
ctx.Leaves.Should().HaveCount(1);
ctx.Leaves.First().Should().BeSameAs(item);
ctx.Leaves.Should().BeSameAs(_moq.List<Leaf>());
ctx.Leaves.Where(x => x.Age == 25)
.Should().HaveCount(1);
By default, when using IQueryable and building expressions to query child objects, you do not have to null guard when using something like EF/Linq2Sql as SQL Server will gracefully handle these and set property to null. Where as in Linq2Objects (SomeList.AsQueryable()
) this will cause a NullReferenceException
.
To overcome this, by default Moqqer will provide a custom IQueryable
where it modifies expressions that contain reference type accessors and to gracefully handle null reference types.
Example:
_moq.Add(new Parent());
var ctx = _moq.Of<IContext>().Object; // Get a mocked context
ctx.Parents.Select(x => x.Name) // Standard accessor
.Single().Should().Be(null);
ctx.Parents.Select(x => (int?) x.Child.Age) // Linq2Objects will throw NullReferenceException
.Single().Should().Be(null);
This can be turned off by setting the following property to false
:
_moq.UseMoqqerEnumerableQuery = false;
The above IQueryable<T>
can also be "Null Guarded" when testing, by calling IEnumerable<T>.AsMoqqerQueryable()
rather than IEnumerable<T>.AsQueryable()
.
Example
var list = new List<Parent>
{
new Parent()
};
var queryable = list.AsMoqqerQueryable();
// list.AsQueryable() would throw a NullReferenceException as `Child` is `null`
var name = queryable.FirstOrDefault(x => x.Child.Name == "Test");
name.Should().Be(null);
As Moqqer creates a Mock, it goes through it's members and sets all overridable members to return Mock<T>.Object
's
var root = _moq.Create<Root>();
root.Tree.Branch.Leaf.Grow();
_moq.Of<ILeaf>().Verify(x => x.Grow(), Times.Once);
When Moqqer does recursive overriding, it only overrides the current class, and overrides all of its overridable properties to Func<T>
's which return another created Mock<T>
. This prevents Moqqer from crawling the whole object graph straight away, as more often then not, this is unnecesary.
var root = _moq.Create<Root>();
_moq.Mocks.Should().NotContainKey(typeof(IBranch));
root.Tree.Branch.Leaf.Grow();
_moq.Mocks.Should().ContainKey(typeof(IBranch));
// Quickly Verify that a mock member was never called
_moq.Verify<ILeaf>(x => x.Grow()).Never();
// Or that it was called once
_moq.Of<ILeaf>().Object.Grow();
_moq.Verify<ILeaf>(x => x.Grow()).Once();
// Or called X number of times
_moq.Of<ILeaf>().Object.Grow();
_moq.Verify<ILeaf>(x => x.Grow()).Times(2);
// Or fallback to using the Moq.Times class
_moq.Of<ILeaf>().Object.Grow();
_moq.Verify<ILeaf>(x => x.Grow()).Times(Times.AtLeast(3));
_moq.Verify<ILeaf>(x => x.Grow()).Times(Times.Between(3,7, Range.Inclusive));
// Create your concrete implementation
var fizz = new Fizz(3);
var buzz = new Buzz(5);
// Set it as the implementation of an interface
_moq.Use<IFizz>(fizz);
_moq.Create<FizzBuzzGame>().Fizz.Should().BeSameAs(fizz);
// Or use for all interfaces
_moq.Use(buzz).ForAllImplementedInterfaces();
_moq.Create<FizzBuzzGame>().Buzz.Should().BeSameAs(buzz);
// Allow you to set a default value for value types
_moq.Use(25);
_moq.Create<Fizz>().Divisor.Should().Be(25);
// Allow you to set a default value as well as reference types
_moq.Use("GitHub");
_moq.Create<StringCtor>().Text.Should().Be("GitHub");
Moqqer provides auto-mocking of collections to List<T>
. This allows for easier setup of methods that return IList<T>
and IQueryable<T>
, and potentially other Collection types in the future (raise GitHubIssues or Pull Requests for suggestions).
Example
var item = new Leaf(25);
// Get instance of List<T>
_moq.List<Leaf>()
.Add(item);
// Extension method to add T item to List<T>
_moq.Add(item);
// Confirm List has 2 Items
_moq.Of<IContext>()
.Object.Leaves.Should().HaveCount(2);
Type | Defaulted To |
---|---|
string |
string.Empty |
List<T> |
moq.List<T>() |
IList<T> |
moq.List<T>() |
ObservableCollection<T> |
ObservableCollection<T> that is backed by moq.List<T>() |
IQueryable<T> |
moq.List<T>() |
Task |
Completed Task e.g. Task.FromResult(true) |
Task<T> |
Completed task with Mocked/Default Task.Result of type T e.g. `Task.FromResult(moq.Of |
Mockable Object | Mock<T> |
Moqqer by default will resolve any Func<T>
to Func<Mock<T>.Object>
if T
is Mockable, otherwise it will try resolve to Func<_moq.Object<T>>
Find it on NuGet under MoqInjectionContainer
https://www.nuget.org/packages/MoqInjectionContainer/
When you don't want to provide It.IsAny<T>
for methods, you can provide a method name and let Moqqer use reflection.
// Setup
_moq.Setup(nameof(_root.Tree)).Returns(new Tree());
// Verify
_moq.Verify(nameof(_root.Tree)).Once();