当 IValidatableObject Validate 使用服务进行验证时,如何对其进行单元测试?

Kel*_*vin 6 c# asp.net-mvc unit-testing moq xunit.net

我的Validate方法是访问各种服务进行验证。这些服务是依赖注入的,可以使用该validationContext.GetService方法进行访问。在运行时,这工作得很好。

public IEnumerable<ValidationResult> Validate(ValidationContext validationContext)
{
    var logic = (ShipRegistrationLogic)validationContext.GetService(typeof(ShipRegistrationLogic));
    var userManager = (UserManager<IdentityUser>)validationContext.GetService(typeof(UserManager<IdentityUser>));
    var spaceTransitAuthority = (SpaceTransitAuthority)validationContext.GetService(typeof(ISpaceTransitAuthority));
    var hull = logic.GetHull(HullId);
    var user = userManager.FindByIdAsync(UserId).Result;

    if (spaceTransitAuthority.CheckActualHullCapacity(hull) > logic.GetMaxAuthorizedMass(user))
    {
        yield return new ValidationResult("You do not have the required pilot's license for this hull.", new[] { nameof(HullId) });
    }
}
Run Code Online (Sandbox Code Playgroud)

我想对这个复杂的验证进行单元测试。但是,在测试执行过程中,我无法弄清楚如何访问测试评估所需的服务。测试运行期间返回GetServicenull

我正在使用Moqand xUnit,并且我尝试模拟 ValidationContext 的 GetService 方法,但没有成功,因为 ValidationContext 是一个密封类,无法模拟。

这是显示我尝试过的测试:

[Theory]
[InlineData(1)]
[InlineData(2)]
[InlineData(3)]
public void Validate_ShouldPassMassCheck(int hullId)
{
    // Arrange
    var userId = _users[0].Id;
    var viewModel = new ShipRegistrationViewModel()
    {
        UserId = userId,
        HullId = hullId
    };

    var hull = new Hull()
    {
        DefaultMaximumTakeOffMass = TakeOffMassEnum.Tank,
        Id = hullId,
        Name = "Test"
    };

    var shipRegistrationLogicMock = new Mock<ShipRegistrationLogic>();
    var spaceTransitAuthorityMock = new Mock<ISpaceTransitAuthority>();
    shipRegistrationLogicMock.Setup(x => x.GetHull(hullId)).Returns(hull);
    shipRegistrationLogicMock.Setup(x => x.GetMaxAuthorizedMass(_users[0])).Returns((int)TakeOffMassEnum.Tank);
    _userManager.Setup(x => x.FindByNameAsync(userId))
        .ReturnsAsync(new IdentityUser()
        {
            UserName = "user",
        });

    spaceTransitAuthorityMock.Setup(x => x.CheckActualHullCapacity(hull)).Returns((double)TakeOffMassEnum.Tank);

    var context = new Mock<ValidationContext>(viewModel);
    context.Setup(x => x.GetService(typeof(ShipRegistrationLogic))).Returns(shipRegistrationLogicMock);
    context.Setup(x => x.GetService(typeof(UserManager<IdentityUser>))).Returns(_userManager);
    context.Setup(x => x.GetService(typeof(ISpaceTransitAuthority))).Returns(spaceTransitAuthorityMock);

    // Act
    var results = viewModel.Validate(context.Object);

    // Assert
    Assert.Single(results);
    Assert.Equal(ValidationResult.Success, results.First());
}
Run Code Online (Sandbox Code Playgroud)

Orw*_*wel 7

您可以模拟 aIServiceProvider并将其传递到构造函数中:

var services = new Mock<IServiceProvider>();
services.Setup(x => x.GetService(typeof(ShipRegistrationLogic)))
    .Returns(shipRegistrationLogicMock.Object);
services.Setup(x => x.GetService(typeof(UserManager<IdentityUser>)))
    .Returns(_userManager.Object);
services.Setup(x => x.GetService(typeof(ISpaceTransitAuthority)))
    .Returns(spaceTransitAuthorityMock.Object);

var context = new ValidationContext(viewModel, services.Object, null);

// Act
var results = viewModel.Validate(context);
Run Code Online (Sandbox Code Playgroud)

你也可以直接构建一个IServiceProvider不带mock的