DEV Community

Cover image for 3 ways to check the object passed to mocks with Moq in C#
Davide Bellone
Davide Bellone

Posted on • Originally published at code4it.dev

3 ways to check the object passed to mocks with Moq in C#

When writing unit tests, you can use Mocks to simulate the usage of class dependencies.

Even though some developers are harshly against the usage of mocks, they can be useful, especially when the mocked operation does not return any value, but still, you want to check that you've called a specific method with the correct values.

In this article, we will learn 3 ways to check the values passed to the mocks when using Moq in our C# Unit Tests.

To better explain those 3 ways, I created this method:

public void UpdateUser(User user, Preference preference)
{
    var userDto = new UserDto
    {
        Id = user.id,
        UserName = user.username,
        LikesBeer = preference.likesBeer,
        LikesCoke = preference.likesCoke,
        LikesPizza = preference.likesPizza,
    };

    _userRepository.Update(userDto);
}
Enter fullscreen mode Exit fullscreen mode

UpdateUser simply accepts two objects, user and preference, combines them into a single UserDto object, and then calls the Update method of _userRepository, which is an interface injected in the class constructor.

As you can see, we are not interested in the return value from _userRepository.Update. Rather, we are interested in checking that we are calling it with the right values.

We can do it in 3 ways.

Verify each property with It.Is

The simplest, most common way is by using It.Is<T> within the Verify method.

[Test]
public void VerifyEachProperty()
{
    // Arrange
    var user = new User(1, "Davide");
    var preferences = new Preference(true, true, false);

    UserDto expected = new UserDto
    {
        Id = 1,
        UserName = "Davide",
        LikesBeer = true,
        LikesCoke = false,
        LikesPizza = true,
    };

    //Act

    userUpdater.UpdateUser(user, preferences);

    //Assert
    userRepo.Verify(_ => _.Update(It.Is<UserDto>(u =>
        u.Id == expected.Id
        && u.UserName == expected.UserName
        && u.LikesPizza == expected.LikesPizza
        && u.LikesBeer == expected.LikesBeer
        && u.LikesCoke == expected.LikesCoke
    )));
}
Enter fullscreen mode Exit fullscreen mode

In the example above, we used It.Is<UserDto> to check the exact item that was passed to the Update method of userRepo.

Notice that it accepts a parameter. That parameter is of type Func<UserDto, bool>, and you can use it to define when your expectations are met.

In this particular case, we've checked each and every property within that function:

u =>
    u.Id == expected.Id
    && u.UserName == expected.UserName
    && u.LikesPizza == expected.LikesPizza
    && u.LikesBeer == expected.LikesBeer
    && u.LikesCoke == expected.LikesCoke
Enter fullscreen mode Exit fullscreen mode

This approach works well when you have to perform checks on only a few fields. But the more fields you add, the longer and messier that code becomes.

Also, a problem with this approach is that if it fails, it becomes hard to understand which is the cause of the failure, because there is no indication of the specific field that did not match the expectations.

Here's an example of an error message:

Expected invocation on the mock at least once, but was never performed: _ => _.Update(It.Is<UserDto>(u => (((u.Id == 1 && u.UserName == "Davidde") && u.LikesPizza == True) && u.LikesBeer == True) && u.LikesCoke == False))

Performed invocations:

Mock<IUserRepository:1> (_):
    IUserRepository.Update(UserDto { UserName = Davide, Id = 1, LikesPizza = True, LikesCoke = False, LikesBeer = True })

Enter fullscreen mode Exit fullscreen mode

Can you spot the error? And what if you were checking 15 fields instead of 5?

Verify with external function

Another approach is by externalizing the function.

[Test]
public void WithExternalFunction()
{
    //Arrange
    var user = new User(1, "Davide");
    var preferences = new Preference(true, true, false);

    UserDto expected = new UserDto
    {
        Id = 1,
        UserName = "Davide",
        LikesBeer = true,
        LikesCoke = false,
        LikesPizza = true,
    };

    //Act
    userUpdater.UpdateUser(user, preferences);

    //Assert
    userRepo.Verify(_ => _.Update(It.Is<UserDto>(u => AreEqual(u, expected))));
}

private bool AreEqual(UserDto u, UserDto expected)
{
    Assert.AreEqual(expected.UserName, u.UserName);
    Assert.AreEqual(expected.Id, u.Id);
    Assert.AreEqual(expected.LikesBeer, u.LikesBeer);
    Assert.AreEqual(expected.LikesCoke, u.LikesCoke);
    Assert.AreEqual(expected.LikesPizza, u.LikesPizza);

    return true;
}
Enter fullscreen mode Exit fullscreen mode

Here, we are passing an external function to the It.Is<T> method.

This approach allows us to define more explicit and comprehensive checks.

The good parts of it are that you will gain more control over the assertions, and you will also have better error messages in case a test fails:

Expected string length 6 but was 7. Strings differ at index 5.
Expected: "Davide"
But was:  "Davidde"
Enter fullscreen mode Exit fullscreen mode

The bad part is that you will stuff your test class with lots of different methods, and the class can easily become hard to maintain. Unluckily, we cannot use local functions.

On the other hand, having external functions allows us to combine them when we need to do some tests that can be reused across test cases.

Intercepting the function parameters with Callback

Lastly, we can use a hidden gem of Moq: Callbacks.

With Callbacks, you can store in a local variable the reference to the item that was called by the method.

[Test]
public void CompareWithCallback()
{
    // Arrange

    var user = new User(1, "Davide");
    var preferences = new Preference(true, true, false);

    UserDto actual = null;
    userRepo.Setup(_ => _.Update(It.IsAny<UserDto>()))
        .Callback(new InvocationAction(i => actual = (UserDto)i.Arguments[0]));

    UserDto expected = new UserDto
    {
        Id = 1,
        UserName = "Davide",
        LikesBeer = true,
        LikesCoke = false,
        LikesPizza = true,
    };

    //Act
    userUpdater.UpdateUser(user, preferences);

    //Assert
    userRepo.Verify(_ => _.Update(expected));
}
Enter fullscreen mode Exit fullscreen mode

In this way, you can use it locally and run assertions directly to that object without relying on the Verify method.

Or, if you use records, you can use the auto-equality checks to simplify the Verify method as I did in the previous example.

Wrapping up

In this article, we've explored 3 ways to perform checks on the objects passed to dependencies mocked with Moq.

Each way has its pros and cons, and it's up to you to choose the approach that fits you the best.

I personally prefer the second and third approaches, as they allow me to perform better checks on the passed values.

What about you?

For now, happy coding!

๐Ÿง

Top comments (1)

Collapse
 
pbouillon profile image
Pierre Bouillon

I also prefer the third approach but it's always nice to see other ways of doing things

Mocking may be very verbose in some case and knowing that you can extract this logic into a dedicated method will certainly be helpful out there!