c# - StructureMap 在 Web API 帮助页面 ModelDescriptionLink.cshtml 中导致 Stack Empty 异常

标签 c# asp.net-web-api2 structuremap structuremap3

我有一个 Web API 项目,它使用 StructureMap 作为其 DI。它已经工作了一段时间,但我在使用 Web API 帮助页面 (Microsoft.AspNet.WebApi.HelpPage) 时遇到了一些问题,其中由于空堆栈而抛出 InvalidOperationExceptions。

我使用帮助页面创建了一个新的 Web API 项目,在我添加 StructureMap.WebApi2 包之前它工作正常,而前面提到的异常在此处抛出,在 ModelDescriptionLink.cshtml 中

else if (modelDescription is CollectionModelDescription)
{
    var collectionDescription = modelDescription as CollectionModelDescription;
    var elementDescription = collectionDescription.ElementDescription;
    @:Collection of @Html.DisplayFor(m => elementDescription.ModelType, "ModelDescriptionLink", new { modelDescription = elementDescription })
}
else
{
    @Html.DisplayFor(m => modelDescription)
}

当它试图显示指向他的这个模型的资源描述。

这是一个精简的路线,但仍然会导致异常:

[Route("Test")]
public IHttpActionResult Post([FromBody] IEnumerable<MySimpleModel> models)
{
    return null;
}

尝试在 http://localhost:21966/Help/Api/POST-Test 访问此路由的文档会导致异常:

Exception image

我只能找到一个 example of someone having the same problem他们的解决方案是从 StructureMap 切换到 Ninject 或通过空检查避免异常。

这是堆栈跟踪的顶部:

[InvalidOperationException: Stack empty.]
System.ThrowHelper.ThrowInvalidOperationException(ExceptionResource resource) +52
System.Collections.Generic.Stack`1.Peek() +6693321
System.Web.WebPages.WebPageBase.get_Output() +51
System.Web.WebPages.WebPageBase.GetOutputWriter() +35
System.Web.WebPages.WebPageExecutingBase.BeginContext(String virtualPath, Int32 startPosition, Int32 length, Boolean isLiteral) +50
ASP._Page_Areas_HelpPage_Views_Help_DisplayTemplates_ModelDescriptionLink_cshtml.Execute() in c:...ModelDescriptionLink.cshtml:28
System.Web.WebPages.WebPageBase.ExecutePageHierarchy() +271
System.Web.Mvc.WebViewPage.ExecutePageHierarchy() +122
System.Web.WebPages.WebPageBase.ExecutePageHierarchy(WebPageContext pageContext, TextWriter writer, WebPageRenderingBase startPage) +145
System.Web.Mvc.RazorView.RenderView(ViewContext viewContext, TextWriter writer, Object instance) +695
System.Web.Mvc.BuildManagerCompiledView.Render(ViewContext viewContext, TextWriter writer) +382
System.Web.Mvc.Html.ActionCacheViewItem.Execute(HtmlHelper html, ViewDataDictionary viewData) +278

通过在这个地方捕获异常,它稍后会在 HelpPageApiModel.cshtml 中的几乎相同的行上弹出:@Html.DisplayFor(m => m.ResourceDescription.ModelType, “ModelDescriptionLink”,新的 { modelDescription = Model.ResourceDescription })。这是该堆栈跟踪的顶部:

[InvalidOperationException: Stack empty.]
System.ThrowHelper.ThrowInvalidOperationException(ExceptionResource resource) +52
System.Collections.Generic.Stack`1.Pop() +6667365
System.Web.WebPages.WebPageBase.PopContext() +66
System.Web.WebPages.WebPageBase.ExecutePageHierarchy(WebPageContext pageContext, TextWriter writer, WebPageRenderingBase startPage) +154

最佳答案

仍然不确定为什么 StructureMap.WebApi2 会遇到这个问题,但我的解决方案基本上是在没有该库的情况下重写依赖项解析。 This blog post对于在不使用 IDependencyScope 的情况下实现它的方法非常有帮助

编辑:我被要求展示一个例子。这基本上是该博客上发布的解决方案,但我希望它能有所帮助。

为结构图添加自定义 Controller 激活器:

public class StructureMapWebApiControllerActivator : IHttpControllerActivator
{
    private readonly IContainer _container;

    public StructureMapWebApiControllerActivator(IContainer container)
    {
        _container = container;
    }

    public IHttpController Create(HttpRequestMessage request, HttpControllerDescriptor controllerDescriptor, Type controllerType)
    {
        var nested = _container.GetNestedContainer();
        var instance = nested.GetInstance(controllerType) as IHttpController;
        request.RegisterForDispose(nested);
        return instance;
    }
}

这是我的容器初始化的样子:

public class IoC
{
    public static IContainer InitializeContainer()
    {
        IContainer container = new Container
        (
            c => c.Scan
            (
                scan =>
                {
                    scan.Assembly("assembly1");
                    scan.Assembly("assembly2");
                    scan.LookForRegistries();
                    scan.WithDefaultConventions();
                }
            )
        );
        return container;
    }
}

然后无论您在何处设置 HttpConfiguration(例如调用 MapHttpAttributeRoutes()),将 Controller 激活器替换为您的新结构映射之一:

HttpConfig.MapHttpAttributeRoutes();
var container = IoC.InitializeContainer();
HttpConfig.Services.Replace(typeof(IHttpControllerActivator), new StructureMapWebApiControllerActivator(container));

在我的例子中,我使用的是 OWIN,所以我没有使用 GlobalConfiguration.Configuration HttpConfiguration,但想法是一样的。我相信通常有一个 WebApiConfig 类,它有一个传递给 HttpConfiguration 的方法,因此您可以在那里进行替换。

再次编辑 HelpController.cs 的更改部分:

最初 StructureMap 试图使用第二个构造函数(其参数为 HttpConfiguration),导致它无法创建 HelpController 的实例。我通过将第二个构造函数标记为 protected 来修复它。

我还在第一个构造函数的调用中将引用从 GlobalConfiguration 的 HttpConfiguration 更改为我自己的(从我的 Startup 类),因为我使用的是 Owin。

public HelpController(): this(Startup.HttpConfig)
{
    logger.Trace("Help controller created");
}

protected HelpController(HttpConfiguration config)
{
    Configuration = config;
}

关于c# - StructureMap 在 Web API 帮助页面 ModelDescriptionLink.cshtml 中导致 Stack Empty 异常,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31318512/

相关文章:

fluent-nhibernate - 在 StructureMap 中使用依赖注入(inject)时,如何在多个构造函数中进行选择?

c# - 在 StructureMap 中有条件地获取实例

c# - Azure 函数的模拟 BlobClient

c# - 导致索引和长度错误的子串

c# - Odata如何授权用户角色的$expand功能?

C# WebAPI 2反序列化产生空值但没有异常

c# - 如何在 C# 的循环内返回结果?

c# - 排除开头但包含结尾

c# - 获取 OAuth session 的过期时间

dependency-injection - 为您的 IoC 提供包装器是个好主意吗?