如何在单元测试期间跳过 Fluent 验证静态函数?

How to skip Fluent validation static function during unit testing?

我正在使用带有 Moq 的 xunit 并在一个函数中测试其他功能,但我总是从无法绕过的 Fluent Validation 扩展方法中得到异常。

这是原始代码

    public class AdService : IAdService
{
    private readonly IValidator<AdToSearchDto> _adToSearchDtoValidator;
    public AdService(IValidator<AdToSearchDto> addToSearchDtoValidator)
    {
        _adToSearchDtoValidator = addToSearchDtoValidator
    }

    public async Task<AdsToReturnDto> GetAdsAsync(AdToSearchDto searchDto)
    {
         _adToSearchDtoValidator.ValidateAndThrow(searchDto); // This line always throw exception

          return await _adRepository.GetUserAdsAsync(searchDto, userId, user?.Country);            
    }

}

单元测试class

 [Fact]
public async Task Ads_Can_Searched_By_UserId()
{
    var validator = new Mock<IValidator<AdToSearchDto>>();

    var adService = new AdService(validator.Object);

    var adService = new AdService(validator.Object );

    var ads = DataGenerator.CreateAds(2);
    ads[0].UserId = searchingUserId;
    ads[1].UserId = anotherUserId;

    await adRepository.CreateAdAsync(ads[0]);
    await adRepository.CreateAdAsync(ads[1]);

    var result = await adService.GetAdsAsync( adToSearchDto, Guid.Empty ) ;

    result.Ads.Count.Should().Be(1);
    result.Ads.FirstOrDefault()?.UserId.Should().Be(searchingUserId.ToString());

}

绕不过去

ValidateAndThrow

因为这是扩展方法。

有什么方法可以绕过单元测试中的 ValidateAndThrow 方法 ?

已编辑:

我收到错误

System.NullReferenceException : Object reference not set to an instance of an object.
   at FluentValidation.DefaultValidatorExtensions.ValidateAndThrow[T](IValidator`1 validator, T instance, String ruleSet) in C:\Projects\FluentValidation\src\FluentValidation\DefaultValidatorExtensions.cs:line 943
   at Market.Business.Ad.AdService.GetAdsAsync(AdToSearchDto searchDto, Guid userId) in C:\Users\Shahid Abdullah\Projects\MyStuff.IdentityServer\src\Market\Market.Business\Ad\AdService.cs:line 129
   at Market.Test.Ad.AdTests.Ads_Can_Searched_By_UserId() in C:\Users\Shahid Abdullah\Projects\MyStuff.IdentityServer\src\Market\Market.Test\Ad\AdTests.cs:line 100
--- End of stack trace from previous location where exception was thrown ---

源代码异常

public static void ValidateAndThrow<T>(
      this IValidator<T> validator,
      T instance,
      string ruleSet = null)
    {
      ValidationResult validationResult = validator.Validate<T>(instance, (IValidatorSelector) null, ruleSet);
      if (!validationResult.IsValid) // this line throws exception as validationResult is null
        throw new ValidationException((IEnumerable<ValidationFailure>) validationResult.Errors);
    }
    /// <summary>
    /// Performs validation and then throws an exception if validation fails.
    /// </summary>
    /// <param name="validator">The validator this method is extending.</param>
    /// <param name="instance">The instance of the type we are validating.</param>
    /// <param name="ruleSet">Optional: a ruleset when need to validate against.</param>
    public static void ValidateAndThrow<T>(this IValidator<T> validator, T instance, string ruleSet = null) {
        var result = validator.Validate(instance, ruleSet: ruleSet);

        if (!result.IsValid) {
            throw new ValidationException(result.Errors);
        }
    }

Source

ValidateAndThrow 期望调用时来自验证器的结果是 return。

因为模拟没有结果

if (!result.IsValid) {...

抛出空引用异常。

将模拟设置为在执行测试时调用扩展方法时的预期行为

[Fact]
public async Task Ads_Can_Searched_By_UserId() {
    //Arrange
    var validator = new Mock<IValidator<AdToSearchDto>>();

    var validResult = new ValidationResult();
    validator
        .Setup(_ => _.Validate(It.IsAny<ValidationContext>())
        .Returns(validResult);

    var adService = new AdService(validator.Object);

    //...
}