MiniMock 0.9.6

There is a newer version of this package available.
See the version list below for details.
dotnet add package MiniMock --version 0.9.6                
NuGet\Install-Package MiniMock -Version 0.9.6                
This command is intended to be used within the Package Manager Console in Visual Studio, as it uses the NuGet module's version of Install-Package.
<PackageReference Include="MiniMock" Version="0.9.6" />                
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add MiniMock --version 0.9.6                
#r "nuget: MiniMock, 0.9.6"                
#r directive can be used in F# Interactive and Polyglot Notebooks. Copy this into the interactive tool or source code of the script to reference the package.
// Install MiniMock as a Cake Addin
#addin nuget:?package=MiniMock&version=0.9.6

// Install MiniMock as a Cake Tool
#tool nuget:?package=MiniMock&version=0.9.6                

MiniMock

Mini mock offers a minimalistic approach to mocking in .Net. It is designed to be simple to use and easy to understand. It is not as feature rich as other mocking frameworks, but aims to solve 95% of the use cases. For the remaining 5% you should consider creating a custom mock.

Mini mock is extremely strict requiring you to specify all features you want to mock. This is by design to make sure you are aware of what you are mocking. Unmocked features will throw an exception if used.

Simple example

    public interface IVersionLibrary
    {
        bool DownloadExists(string version);
    }

    [Fact]
    [Mock<IVersionLibrary>]
    public void SimpleExample()
    {
        var library = Mock.IVersionLibrary(config =>
            config.DownloadExists(returns: true));

        var actual = library.DownloadExists("2.0.0.0");

        Assert.True(actual);
    }

Key feature summery

  • Minimalistic api with fluent method chaining, documentation and full intellisence
  • Mocking of interfaces, abstract classes and virtual methods (with limitations)
  • Mocking of methods, properties, indexers and events
  • Simple factory methods to initialize mocks
  • Mocking of Async methods, overloads and generic methods
  • Ref and out parameters in methods supported
  • Generic interfaces supported

Limitations

  • No validation of calls
  • Only support C# (workarounds exist for VB.Net and F#)
  • No support for Generic methods (issue #8)
  • Ref return values as ref properties are not supported (issue #5)
  • Partially mocking of classes
    • Base classes with constructors with parameters are not currently supported (Issue #4)
  • No support for static classes or methods

Installation & Initialization

Reference nuget package in your test project

dotnet add package MiniMock

Specify which interface to mock by using the [Mock] attribute before your test or test class.

[Fact]
[Mock<IMyRepository>]
public void MyTest {
}

Create a mock by using the mock factory

var mockRepository = Mock.IMyRepository();

Configure your mock to your needs

var mockRepo = Mock.IMyRepository(config => config.CreateCustormerAsync(return: Guid.NewGuid());

Use the mocked object

var sut = new CustomerMaitinance(mockRepo);
sut.Create(customerDTO, cancelationToken);

Quality of life features

Fluent interface with full intellisence and documentation.

All mockable members are available through a fluent interface with intellisence, type safety and documentation.

Since the mock code is generated at development time you can inspect, stepped into and debug the code. This also allows for security and vulnerability scanning of the code.

All code required to run MiniMock is generated and has no runtime dependencies.

Simple return values

Simply specify what you expect returned from methods or properties. All parameters are ignored.

    var mockLibrary = Mock.IVersionLibrary(config => config
                .DownloadExists(returns: true) // Returns true for any parameter
                .DownloadLinkAsync(returns: new Uri("http://downloads/2.0.0")) // Returns a task with a download link
                .CurrentVersion(value: new Version(2, 0, 0, 0)) // Sets the initial version to 2.0.0.0
                .Indexer(values: versions) // Provides a dictionary to retrieve and store versions
    );

Multiple return values

Specify multiple return values for a method or property. The first value is returned for the first call, the second for the second call and so on.

    var mockLibrary = Mock.IVersionLibrary(config => config
                .DownloadExists(returns: true, false, true) // Returns true, false, true for the first, second and third call
                .DownloadLinkAsync(returns: [Task.FromResult(new Uri("http://downloads/2.0.0")), Task.FromResult(new Uri("http://downloads/2.0.1"))]) // Returns a task with a download link for the first and second call
                .DownloadLinkAsync(returns: new Uri("http://downloads/2.0.0"), new Uri("http://downloads/2.0.1")) // Returns a task with a download link for the first and second call
    );

Intercept method calls

    [Fact]
    [Mock<IVersionLibrary>]
    public async Task InterceptMethodCalls()
    {
        var currentVersionMock = new Version(2, 0, 0);
        
        var versionLibrary = Mock.IVersionLibrary(config => config
                .DownloadExists(call: (string s) => s.StartsWith("2.0.0") ? true : false ) // Returns true for version 2.0.0.x base on a string parameter
                .DownloadExists(call: (Version v) => v is { Major: 2, Minor: 0, Revision: 0 })// Returns true for version 2.0.0.x based on a version parameter
                //or
                .DownloadExists(call: LocalIntercept) // calls a local function
                .DownloadExists(call: version => this.ExternalIntercept(version, true)) // calls function in class

                .DownloadLinkAsync(call: s => Task.FromResult(new Uri($"http://downloads/{s}"))) // Returns a task containing a download link for version 2.0.0.x otherwise a error link
                .DownloadLinkAsync(call: s => new Uri($"http://downloads/{s}")) // Returns a task containing a download link for version 2.0.0.x otherwise a error link

                .CurrentVersion(get: () => currentVersionMock, set: version => currentVersionMock = version) // Overwrites the property getter and setter
                .Indexer(get: s => new Version(2,0,0,0), set: (s, version) => {}) // Overwrites the indexer getter and setter
        );

        return;

        bool LocalIntercept(Version version)
        {
            return version is { Major: 2, Minor: 0, Revision: 0 };
        }
    }

    private bool ExternalIntercept(string version, bool startsWith) => startsWith ? version.StartsWith("2.0.0") : version == "2.0.0";

Async methods

Simply return what you expect from async methods either as a Task object or a simple value.

    var versionLibrary = Mock.IVersionLibrary(config => config
                .DownloadLinkAsync(returns: Task.FromResult(new Uri("http://downloads/2.0.0"))) // Returns a task containing a download link for all versions
                .DownloadLinkAsync(call: s => Task.FromResult(new Uri($"http://downloads/{s}"))) // Returns a task containing a download link for version 2.0.0.x otherwise a error link
            // or
                .DownloadLinkAsync(returns: new Uri("http://downloads/2.0.0")) // Returns a task containing a download link for all versions
                .DownloadLinkAsync(call: s => new Uri($"http://downloads/{s}")) // Returns a task containing a download link for version 2.0.0.x otherwise a error link
        );

Strict mocking

Unmocked features will always throw InvalidOperationException.

    [Fact]
    [Mock<IVersionLibrary>]
    public void UnmockedFeaturesAlwaysThrowInvalidOperationException()
    {
        var versionLibrary = Mock.IVersionLibrary();

        var propertyException = Assert.Throws<InvalidOperationException>(() => versionLibrary.CurrentVersion);
        var methodException = Assert.Throws<InvalidOperationException>(() => versionLibrary.DownloadExists("2.0.0"));
        var asyncException = Assert.ThrowsAsync<InvalidOperationException>(() => versionLibrary.DownloadLinkAsync("2.0.0"));
        var indexerException = Assert.Throws<InvalidOperationException>(() => versionLibrary["2.0.0"]);
    }

Adding indexers

Mocking indexers is supported either by overloading the get and set methods or by providing a dictionary with expected values.

    [Fact]
    [Mock<IVersionLibrary>]
    public void Indexers()
    {
        var versions = new Dictionary<string, Version>() {{"current", new Version(2,0,0,0)}};
    
        var versionLibrary = Mock.IVersionLibrary(config => config
                .Indexer(get: s => new Version(2,0,0,0), set: (s, version) => {}) // Overwrites the indexer getter and setter
                .Indexer(values: versions) // Provides a dictionary to retrieve and store versions
        );
    
        var preCurrent = versionLibrary["current"];
        versionLibrary["current"] = new Version(3, 0, 0, 0);
        var postCurrent = versionLibrary["current"];
        Assert.NotEqual(preCurrent, postCurrent);
    }

Raising events

Raise events using an event trigger.

    Action<Version>? triggerNewVersionAdded = null;
    
    var versionLibrary = Mock.IVersionLibrary(config => config
            .NewVersionAdded(trigger: out triggerNewVersionAdded) // Provides a trigger for when a new version is added
        );
    
    triggerNewVersionAdded?.Invoke(new Version(2, 0, 0, 0));

Argument matching

MiniMock does not support argument matching using matchers like other mocking frameworks. Instead, you can use the call parameter to match arguments using predicates or internal functions.

    var versionLibrary = Mock.IVersionLibrary(config => config
            .DownloadExists(call: version => version is { Major: 2, Minor: 0 }) // Returns true for version 2.0.x based on a version parameter
        );

using internal functions

    var versionLibrary = Mock.IVersionLibrary(config =>
    {
       bool downloadExists(Version version) => version switch {
               { Major: 1, Minor: 0 } => true, // Returns true for version 1.0.x based on a version parameter
               { Major: 2, Minor: 0, Revision: 0 } => true, // Returns true for version 2.0.0.0 based on a version parameter
               { Major: 3, } => false, // Returns false for version 3.x based on a version parameter
               _ => throw new ArgumentException() // Throws an exception for all other versions
           };
    
       config.DownloadExists(downloadExists);
    });
Product Compatible and additional computed target framework versions.
.NET net5.0 was computed.  net5.0-windows was computed.  net6.0 was computed.  net6.0-android was computed.  net6.0-ios was computed.  net6.0-maccatalyst was computed.  net6.0-macos was computed.  net6.0-tvos was computed.  net6.0-windows was computed.  net7.0 was computed.  net7.0-android was computed.  net7.0-ios was computed.  net7.0-maccatalyst was computed.  net7.0-macos was computed.  net7.0-tvos was computed.  net7.0-windows was computed.  net8.0 was computed.  net8.0-android was computed.  net8.0-browser was computed.  net8.0-ios was computed.  net8.0-maccatalyst was computed.  net8.0-macos was computed.  net8.0-tvos was computed.  net8.0-windows was computed. 
.NET Core netcoreapp2.0 was computed.  netcoreapp2.1 was computed.  netcoreapp2.2 was computed.  netcoreapp3.0 was computed.  netcoreapp3.1 was computed. 
.NET Standard netstandard2.0 is compatible.  netstandard2.1 was computed. 
.NET Framework net461 was computed.  net462 was computed.  net463 was computed.  net47 was computed.  net471 was computed.  net472 was computed.  net48 was computed.  net481 was computed. 
MonoAndroid monoandroid was computed. 
MonoMac monomac was computed. 
MonoTouch monotouch was computed. 
Tizen tizen40 was computed.  tizen60 was computed. 
Xamarin.iOS xamarinios was computed. 
Xamarin.Mac xamarinmac was computed. 
Xamarin.TVOS xamarintvos was computed. 
Xamarin.WatchOS xamarinwatchos was computed. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.

NuGet packages

This package is not used by any NuGet packages.

GitHub repositories

This package is not used by any popular GitHub repositories.

Version Downloads Last updated
0.9.11 99 10/13/2024
0.9.10 95 10/9/2024
0.9.9 95 10/9/2024
0.9.8 88 9/24/2024
0.9.7 108 9/12/2024
0.9.6 107 9/12/2024
0.9.5 106 9/11/2024
0.9.4 103 9/10/2024
0.9.3 98 9/3/2024
0.9.2 94 9/2/2024
0.9.1 102 9/2/2024

Initial version of MiniMock supporting Interfaces and a subset of class mocking.