单元测试具有子/子类的类的FluentValidation规则

时间:2012-12-17 13:50:30

标签: c# unit-testing fluentvalidation

当我们验证的对象具有也经过验证的子类时,是否可以为fluentvalidation类编写单元测试。

举个例子: 我的班级看起来像这样

public class TestModel
{

    public class SubData
    {
        public int SubDataVal { get; set; }
    }

    public int ParentVal { get; set; }
    public SubData Sub { get; set; }

}

我的验证逻辑如下所示:

public class TestModelValidator : AbstractValidator<TestModel>
{
    public TestModelValidator()
    {
        RuleFor(o => o.ParentVal).GreaterThan(0);
        RuleFor(o => o.Sub.SubDataVal).GreaterThan(0);
    }
}

当我写下面的单元测试时

    [Test]
    public void Should_have_error_when_val_is_zero()
    {
        validator = new TestModelValidator();
        validator.ShouldHaveValidationErrorFor(model => model.ParentVal, 0);
    }

我得到一个“System.NullReferenceException:对象引用未设置为对象的实例。”例外 的 FluentValidation.TestHelper.ValidatorTester`2.ValidateError(T instanceToValidate)

(如果我删除 RuleFor(o =&gt; o.Sub.SubDataVal).GreaterThan(0); 行,那么它可以工作!)

同样,如果我尝试使用以下单元测试实际的子类:

    [Test]
    public void Should_have_error_when_sub_dataVal_is_zero()
    {
        validator = new TestModelValidator();
        validator.ShouldHaveValidationErrorFor(model => model.Sub.SubDataVal, 0);
    }

我得到一个“System.Reflection.TargetException:Object与目标类型不匹配。”来自 FluentValidation.TestHelper.ValidatorTester`2.ValidateError(T instanceToValidate)

3 个答案:

答案 0 :(得分:7)

您可以对模型和子模型进行单元测试,但是您需要更改验证类以使用单独的验证器类来验证子模型:

public class TestModelValidator : AbstractValidator<TestModel>
{
    public TestModelValidator()
    {
        RuleFor(o => o.ParentVal).GreaterThan(0);
        RuleFor(o => o.Sub).SetValidator(new SubDataValidator());
    }
}

public class SubDataValidator : AbstractValidator<SubData>
{
    public SubDataValidator()
    {
        RuleFor(o => o.SubDataVal).GreaterThan(0);
    }
}

然后,您可以编写单元测试来测试每个验证器或两者。

答案 1 :(得分:3)

我得出结论,因为这个ShouldHaveValidationErrorFor不能处理子类,所以已经采取了手动操作。即。

    [Test]
    public void Should_have_error_when_val_is_zero()
    {
        validator = new TestModelValidator();
        TestModel testRequest = new TestModel();
        //populate with dummy data
        var result = validator.Validate(testRequest);
        Assert.That(result.Errors.Any(o => o.PropertyName== "ParentVal"));
    }

答案 2 :(得分:1)

借助MSTest和FluentAssertions,您可以编写

[TestMethod]
public void Should_have_error_when_val_is_zero()
{
    // Given
    var validator = new TestModelValidator();
    var testModel = TestModel
    {
        ParentVal = 0
    }; // You should create a invalid TestModel object here

    // When
    validator.Validate(testModel).IsValid.Should().BeFalse();
}