与Moq一起苦苦挣扎:以下设置不匹配

Cha*_*ell 11 c# unit-testing moq

我是第一次使用Moq,而我正在努力让测试运行正常.

我正在尝试moq Save()我的服务层的方法.

public void Save(UserViewModel viewModel)
{
    // todo: this still doesn't address updating a password. The UserViewModel doesn't contain any Password data.
    if (viewModel.Id != Guid.Empty)
    {
        // The UserId is not empty, we're either updating an existing user
        // or we're inserting a new user via sync
        var user = _userRepository.GetById(viewModel.Id);
        if (user != null)
        {
            // Looks like we're updating a user because they're already in the database.
            _userRepository.Update(_userViewModelToModelMapper.BuildFrom(viewModel));
            return;
        }
    }

    // The user is being created, either via a Sync (Guid Exists), or via an Insert (Guid doesn't Exist)
    _userRepository.Create(_userViewModelToModelMapper.BuildFrom(viewModel));
}
Run Code Online (Sandbox Code Playgroud)

我有三个测试,其中,我不确定它们是否正确.前两个是通过,但第三个失败了

Moq.MockVerificationException:以下设置不匹配:
IUserRepository r => r.Update(It.Is(um => um.Equals()))

这是测试.

// PASSES but could be suspect
[Test]
public void ShouldSaveANewUserFromExistingId()
{
    // emulating a "sync"

    // Setup
    var userId = new Guid("81C7FE19-2DB5-4083-BD6A-5433687561F7");
    var userModel = new UserModel();
    var userViewModel = new UserViewModel {Id = userId};
    var userRepository = new Mock<IUserRepository>();
    var viewModelToModelMapper = new Mock<IAutoMapper<UserViewModel, UserModel>>();
    var modelToViewModelMapper = new Mock<IAutoMapper<UserModel, UserViewModel>>();

    // Setup the Mock UserRepository
    userRepository.Setup(r => r.Create(It.Is<UserModel>(um => um.Equals(userModel))));

    viewModelToModelMapper.Setup(vmm => vmm.BuildFrom(It.Is<UserViewModel>(u => u.Equals(userViewModel))))
           .Returns(userModel);

    var userService = new UserService(userRepository.Object, viewModelToModelMapper.Object, modelToViewModelMapper.Object);

    // Execute
    userService.Save(userViewModel);

    // Assert
    userRepository.VerifyAll();
    viewModelToModelMapper.VerifyAll();
}
Run Code Online (Sandbox Code Playgroud)
// PASSES but could be suspect.
[Test]
public void ShouldSaveANewUser()
{
    // emulating a standard create

    // Setup
    var userId = Guid.Empty;
    var userModel = new UserModel();
    var userViewModel = new UserViewModel { Id = userId };
    var userRepository = new Mock<IUserRepository>();
    var viewModelToModelMapper = new Mock<IAutoMapper<UserViewModel, UserModel>>();
    var modelToViewModelMapper = new Mock<IAutoMapper<UserModel, UserViewModel>>();

    // Setup the Mock UserRepository
    userRepository.Setup(r => r.Create(It.Is<UserModel>(um => um.Equals(userModel))));

    viewModelToModelMapper.Setup(vmm => vmm.BuildFrom(It.Is<UserViewModel>(u => u.Equals(userViewModel))))
           .Returns(userModel);

    var userService = new UserService(userRepository.Object, viewModelToModelMapper.Object, modelToViewModelMapper.Object);

    // Execute
    userService.Save(userViewModel);

    // Assert
    userRepository.VerifyAll();
    viewModelToModelMapper.VerifyAll();

}
Run Code Online (Sandbox Code Playgroud)
// FAILS MISERABLY
[Test]
public void ShouldSaveAnExistingUser()
{
    // emulating an "Update"

    // Setup
    var userId = new Guid("0A88AEC2-9F8D-44DE-BD01-3EB9A23C78E3");
    var userModel = new UserModel { Id = userId };
    var userViewModel = new UserViewModel { Id = userId };
    var userRepository = new Mock<IUserRepository>();
    var viewModelToModelMapper = new Mock<IAutoMapper<UserViewModel, UserModel>>();
    var modelToViewModelMapper = new Mock<IAutoMapper<UserModel, UserViewModel>>();

    // Setup the Mock UserRepository
    userRepository.Setup(r => r.Update(It.Is<UserModel>(um => um.Equals(userModel))));

    viewModelToModelMapper.Setup(vmm => vmm.BuildFrom(It.Is<UserViewModel>(u => u.Equals(userViewModel))))
           .Returns(userModel);

    var userService = new UserService(userRepository.Object, viewModelToModelMapper.Object, modelToViewModelMapper.Object);

    // Execute
    userService.Save(userViewModel);

    // Assert
    userRepository.VerifyAll();
    viewModelToModelMapper.VerifyAll();
}
Run Code Online (Sandbox Code Playgroud)

我的莫清哪里出错了?

Jes*_*and 12

看起来像你需要的最后一种情况:

userRepository.Setup(r => r.GetById(userId)).Returns(userModel);
Run Code Online (Sandbox Code Playgroud)

在userService.Save调用之前.如果没有它,userRepository.GetById将始终为null,并且永远不会采用更新分支.

  • 是的,但只有在GetById方法返回非空值时才会调用更新.由于您尚未在存储库模型上设置GetById方法,因此它将返回一个默认值,对于引用类型,该值为null.然后,用户服务中的if语句的计算结果为false,并且不会调用更新. (3认同)