无法通过System.Reflection.TypeInfo

时间:2019-07-15 14:10:23

标签: c# unit-testing autofixture automoq

我们正在尝试为ASP.Net Core API控制器创建一个简单的单元测试。我们正在将autofix与autoMoq,XUnit2配合使用。我们如何模拟TypeInfo的创建?还是有更好的方法?

我们遵循this post来解决以下初始错误:

  

AutoFixture.ObjectCreationException:AutoFixture无法从Microsoft.AspNetCore.Mvc.ViewFeatures.ViewDataDictionary创建实例,因为创建意外地因异常而失败。请参考内部异常以调查失败的根本原因。

namespace Tests.Controllers
{
    using Api.Controllers;
    using AutoFixture;
    using AutoFixture.AutoMoq;
    using Shouldly;
    using System.Threading.Tasks;
    using Xunit;

    public class DiagnosticControllerTests
    {
        private readonly DiagnosticController _sut;

        public DiagnosticControllerTests()
        {
            var fixture = new Fixture();
            fixture.Customize(new AutoMoqCustomization())
                   .Customize(new ApiControllerCustomization()); // from Matt Uliasz see link above

            _sut = fixture.Create<DiagnosticController>();
        }

        [Fact]
        public async Task Ping_Returns_True()
        {
            var actual = await _sut.Ping();
            actual.Data.ShouldBe(true);
        }
    }
}

这将引发以下错误:

  

AutoFixture.ObjectCreationExceptionWithPath   AutoFixture无法从System.Reflection.TypeInfo创建实例,因为创建意外失败,并带有异常。请参考内部异常以调查失败的根本原因。   ...   内部异常消息:       Castle.DynamicProxy.InvalidProxyConstructorArgumentsException:无法实例化类System.Reflection.TypeInfo的代理。   找不到无参数的构造函数。

修改 经过进一步测试,当我们停止从类 Microsoft.AspNetCore.Mvc.Controller 派生时,错误消失了。

namespace Api.Controllers
{
    using Microsoft.AspNetCore.Authorization;
    using Microsoft.AspNetCore.Mvc;
    using Models;
    using System.Threading.Tasks;

    [Authorize]
    [Route("api/[controller]")]
    public class DiagnosticController: Controller
    {
        [AllowAnonymous]
        [HttpGet]
        [Route("ping")]
        public Task<PingResultDto> Ping()
        {
            var result = new PingResultDto
            {
                Data = true
            };

            return Task.FromResult(result);
        }
    }
}

编辑2 我们当前的解决方法是不使用AutoFixture / AutoMoq:

var sut = new Mock<DiagnosticController>().Object; //This works

2 个答案:

答案 0 :(得分:0)

尝试一下:

internal class ControllerBaseCustomization : ICustomization
{
    public void Customize(IFixture fixture)
    {
        fixture.Customizations.Add(
            new FilteringSpecimenBuilder(
                new Postprocessor(
                    new MethodInvoker(
                        new ModestConstructorQuery()),
                    new ControllerBaseFiller()),
                new ControllerBaseSpecification()));
    }

    private class ControllerBaseFiller : ISpecimenCommand
    {
        public void Execute(object specimen, ISpecimenContext context)
        {
            if (specimen == null) throw new ArgumentNullException(nameof(specimen));
            if (context == null) throw new ArgumentNullException(nameof(context));

            if (specimen is ControllerBase controller)
            {
                controller.ControllerContext = new ControllerContext
                {
                    HttpContext = (HttpContext)context.Resolve(typeof(HttpContext))
                };
            }
            else
            {
                throw new ArgumentException("The specimen must be an instance of ControllerBase", nameof(specimen));
            }
        }
    }

    private class ControllerBaseSpecification : IRequestSpecification
    {
        public bool IsSatisfiedBy(object request) => request is Type type && typeof(ControllerBase).IsAssignableFrom(type);
    }
}

答案 1 :(得分:-1)

请注意,我在下面的回答并没有真正解决您的问题,但这只是一个提示:

我认为您的测试过于复杂了。

请记住,单元测试是为核心业务逻辑保留的。

对于端到端测试,应该进行集成测试。

如果要进行集成测试,请使用以下命令: https://docs.microsoft.com/en-us/aspnet/core/test/integration-tests?view=aspnetcore-2.2

如果您真的想测试您的控制器: https://docs.microsoft.com/en-us/aspnet/core/mvc/controllers/testing?view=aspnetcore-2.2