AutoFixture 自动设置返回类型 Task<IEnumerable<>> 自定义

And*_*Gis 3 c# unit-testing autofixture

我在我的测试中使用 AutoFixture 和 AutoMoqCustomization。

我有一个服务,它是被测系统的依赖项:

ISomeService
{
    Task<IEnumerable<int>> Get();
}
Run Code Online (Sandbox Code Playgroud)

我在被测系统内部调用它:

var collection = await _someService.Get(); // collection is empty
Run Code Online (Sandbox Code Playgroud)

我不在乎集合里面有什么,但我需要集合不是空的。我这样做:

_fixture.Freeze<Mock<ISomeService>>()
            .Setup(service => service.Get())
            .Returns(Task.FromResult(_fixture.CreateMany<int>()));
Run Code Online (Sandbox Code Playgroud)

看起来应该通过自定义来完成。我创建并注册了一个:

public class TaskCollectionCustomization : ICustomization
{
    public void Customize(IFixture fixture)
    {
        fixture.Customizations.Add(
            new FilteringSpecimenBuilder(
                new TaskCollectionBuilder(),
                new GenericTypeSpecification(typeof(Task<>))));
    }

    private class TaskCollectionBuilder : ISpecimenBuilder
    {
        public object Create(object request, ISpecimenContext context)
        {
            // never enters here
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

问题是它的 Create 方法从未被输入。任何想法或即用型解决方案?

编辑

添加 GenericTypeSpecification 源

public class GenericTypeSpecification : IRequestSpecification
{
    private readonly Type _type;

    public GenericTypeSpecification(Type type)
    {
        _type = type;
    }

    public bool IsSatisfiedBy(object request)
    {
        var requestedType = request as Type;

        return requestedType != null &&
               requestedType.IsGenericType &&
               requestedType.GetGenericTypeDefinition() == _type;
    }
}
Run Code Online (Sandbox Code Playgroud)

Mar*_*ann 6

AutoFixture 已经支持开箱即用的任务,正如本特性测试所证明的那样:

[Fact]
public void AutoFixtureAlreadySupportsTasks()
{
    var fixture = new Fixture();
    var t = fixture.Create<Task<IEnumerable<int>>>();
    Assert.NotEmpty(t.Result);
}
Run Code Online (Sandbox Code Playgroud)

因此,您需要配置服务的测试替身是这样的:

[Fact]
public void ConfigureMock()
{
    var fixture = new Fixture().Customize(new AutoMoqCustomization());
    fixture.Freeze<Mock<ISomeService>>()
        .Setup(s => s.Get())
        .Returns(fixture.Create<Task<IEnumerable<int>>>());

    var svc = fixture.Create<ISomeService>();

    Assert.NotEmpty(svc.Get().Result);
}
Run Code Online (Sandbox Code Playgroud)

如果您认为这工作量太大,您也可以要求AutoConfiguredMoqCustomization为您代劳,如下所示:

[Fact]
public void SimplestCustomization()
{
    var fixture = 
        new Fixture().Customize(new AutoConfiguredMoqCustomization());
    var svc = fixture.Create<ISomeService>();
    Assert.NotEmpty(svc.Get().Result);
}
Run Code Online (Sandbox Code Playgroud)

不过,就个人而言,我不是自动配置测试双打的忠实粉丝,因为我相信明确优于隐式和测试双配置应该是一个单元测试的一个明确的部分,因为它描述了间接输入的测试用例。

  • @gisek 只会触发两个自定义中的一个,这取决于您首先应用哪一个。由于 `AutoConfiguredMoqCustomization` 是 `AutoMoqCustomization` 的超集,因此无需同时使用两者。作为旁注,你也可以做`.ReturnsUsingFixture(fixture)`而不是`.Returns(fixture.Create&lt;Task&lt;IEnumerable&lt;int&gt;&gt;&gt;())` (2认同)