7

我有一个使用 StructureMap 作为其 DI 的 Web API 项目。它已经工作了一段时间,但我在 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)
}

@:Collection of @Html.DisplayFor(m => elementDescription.ModelType, "ModelDescriptionLink", new { modelDescription = elementDescription })当它试图为他的这个模型显示资源描述的链接时,它被抛出了。

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

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

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

异常图像

我只能找到一个遇到同样问题的人的例子,他们的解决方案是从 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", new { 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
4

1 回答 1

1

仍然不确定为什么 StructureMap.WebApi2 会出现此问题,但我的解决方案基本上是在没有该库的情况下重写依赖解析。这篇博客文章对于在不使用 IDependencyScope 的情况下实现它的方法非常有帮助

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

为结构映射添加自定义控制器激活器:

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()),将控制器激活器替换为您的新结构映射之一:

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 的实例。我通过将第二个构造函数标记为受保护来修复它。

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

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

protected HelpController(HttpConfiguration config)
{
    Configuration = config;
}
于 2015-07-09T17:30:46.793 回答