Skip to content

Latest commit

 

History

History
444 lines (308 loc) · 12.9 KB

README.md

File metadata and controls

444 lines (308 loc) · 12.9 KB

Moqqer

Intro

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 Mocks 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.

Usage

// 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!

Example

using MoqqerNamespace
[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);
    }
}

Features

Constructor Injection

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);

Concrete Class Injection

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);

The default constructor requirement can be overriding the autogenerate option when calling Moq.Creat<T>:

Moqqer.Create<T>(bool autogenerate = false)

E.g.

var mock = _moq
	.Create<ClassWithCtorContainingClassWithoutParameterlessCtor>(autogenerate: true);

mock.CtorParam.A.Should().NotBeNull();

Default Object Injection

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);
}

Default IQueryable<T> Implementation

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);

IQueryable<T> Null Guarding

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;

AsMoqqerQueryable()

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);

Recursive Mocking

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);

Lazy Mocking

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));

Quicker Verification:

// 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));

Concrete Implementation

// 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 for reference types
_moq.Use("GitHub");
_moq.Create<StringCtor>().Text.Should().Be("GitHub");

// ### Shortcuts
// Shortcut for creating a concrete type, and using this instance for all implemented interfaces.
// e.g. syntactic sugar for _moq.Use(_moq.Create<Leaf>).ForAllImplementedInterfaces()
var leaf = _moq.Use<Leaf>(); 
_moq.Create<Branch>().Leaf.Should().BeSameAs(leaf);

List

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 items to List<T>
_moq.Add(item);
_moq.Add(item, item, item);
_moq.AddItems(new List<Leaf>{ item, item, item });

// Confirm List has 8 Items
_moq.Of<IContext>()
    .Object.Leaves.Should().HaveCount(8);

Factory Method

In case you want to inject different instances of mocks, or different implementations, you can setup Moqqer with a factory method for a given type and return any specific instance according to the context such as the following:

// Create custom leaf
var customLeaf = new Leaf(25);

// Register ILeaf factory function
_moq.Factory<ILeaf>(context =>
	context.CallType == CallType.Constructor
        ? customLeaf // For Ctor injection return custom leaf
        : context.Default // Otherwise return default Mock
);

// Get instance of Branch (has ILeaf in Ctor)
var branch = _moq.Create<Branch>();

// Its ILeaf should be same as custom leaf
branch.Leaf.Should().BeSameAs(customLeaf);
branch.Leaf.Age.Should().Be(25);

// Get an instance of tree
var tree = _moq.Create<Tree>();

// Indirect (non Ctor injection) should be default mock
tree.Branch.Leaf.Should().NotBeSameAs(customLeaf);

tree.Branch.Leaf.Should().BeSameAs(_moq.Of<ILeaf>().Object);

Mock Concrete Return Types

When an interface has a Member that returns a Concrete type, it by default will not be mocked. You can override this by specifying _moq.ConcreteReturnTypes = true such as the following:

// Tell moqqer to Mock members with concrete return types
_moq.MockConcreteReturnTypes = true;

// Get instance of Tree (has IBranch injected)
var tree = _moq.Create<Tree>(true);

// IBranch has a property that returns and concrete type Leaf
tree.Branch.ConcreteLeaf.Should()
    .NotBeNull();

Known Issues

Currently auto mocking of methods with out or ref parameters doesn't work, until the following issues are fixed:

Moq Extensions

Default Mocks

Type Defaulted To
string string.Empty
List<T> moq.List<T>()
IList<T> moq.List<T>()
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>

Func Resolution

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>>

Installing

Find it on NuGet under Moqqer

https://www.nuget.org/packages/Moqqer/

Roadmap

Method Name Selection

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();

Contributors

Special thanks to the following contributors:

Zoltan Rajnai (zrajnai)