asp.net-mvc-4 – MVC4 MEF插件和控制器命名空间

我正在尝试使用许多插件创建一个MVC4 Web应用程序,即基本上通过MEF导出的控制器以及解压缩到正确位置的内容文件.我找到了很多关于MVC插件的资料,主要涉及领域,但我不得不放弃MvcContrib,这是最明显的解决方案,因为它似乎没有更多的开发,显示了最新的MVC位的一些问题,而且我也喜欢这种架构的最简单复杂的实现.

因此我的要求是:

a)一个基于MEF的MVC插件解决方案,我只是在我的站点中删除一个包以供它使用,理想情况下甚至没有重启.这意味着将插件存储在与Bin不同的文件夹中,这也提供了更好的隔离.

b)符合IoC工具的解决方案比仅由MEF完成的解决方案更完整.我倾向于使用Autofac,因为它与MEF和MVC4(此时为RC)集成.

除了像视图这样的内容之外,其基本任务是让MVC在MEF插件中定位控制器并实例化它们,所以我需要一个控制器工厂.我在这里发现了一篇很好的文章:http://kennytordeur.blogspot.be/2012/08/mef-in-aspnet-mvc-4-and-webapi.html(我联系了Kenny,我感谢他指点我的路由问题).作者还将他的代码包装成一个方便的nuget包(MEF.MVC4).无论如何,我发现了一个似乎与路由和命名空间有关的问题:当遇到一个到插件控制器的路由时,MEF控制器工厂的GetControllerInstance方法得到一个null controllerType,最终导致404.我想我可能找到了罪魁祸首通过阅读这些帖子:

http://blog.davebouwman.com/2011/12/08/asp-net-mvc3-and-404s-for-area-controllers/

Custom Controller Factory,Dependency Injection / Structuremap problems with ASP.NET MVC

我想(但我可能错了)问题出在路由约定和插件区域控制器命名空间中:插件控制器的命名空间不在主机Web的同一“根”中.该帖子中提出的解决方案只是为Web应用程序添加了一条新路径,但这并不适用于区域作为插件工作的动态添加到主机应用程序的解决方案.我的主机网络应用程序必须保持不知道插件,这当然应该是一个相当普遍的要求,但我找不到明显的解决方案.

Repro解决方案

您可以快速创建repro解决方案,以便按照以下步骤查看我的方法的详细信息,或从here下载:

1)创建一个空白解决方案.

2)创建一个MVC4 Web应用程序(HostWeb),更新所有NuGet预装的包并添加Mef.MVC4,Autofac MVC 4(RC)和Autofac.Mef.在我的真实应用程序中,我想使用Autofac在构造函数中注入控制器的依赖项.

3)在HostWeb中创建一个Plugins文件夹,并在其中创建一个子文件夹Temp.这将包括使用Temp子文件夹作为卷影副本容器的插件,以便从它而不是直接从插件加载MEF目录.这与一些启动代码一起应该让我更新插件而不必重新启动Web应用程序(否则将锁定DLL).启动代码是一个名为PreApplicationInit的类,可以在Infrastructure文件夹中找到(稍微修改自http://shazwazza.com/post/Developing-a-plugin-framework-in-ASPNET-with-medium-trust.aspx).

4)将一个部件区域添加到主机Web并从视图根文件夹中复制_ViewStart文件(并更改布局视图中的现有链接,以便将空白区域添加到路径值,以便它们不会被破坏) .所有插件控制器都将命名空间到名为Parts的区域中.主机Web应用程序有一个没有控制器的区域,只是为插件内容文件准备文件夹结构和路由(视图,将从插件安装程序模块解压缩到正确的位置,而二进制文件将放在插件中).

5)在HostWeb的App_Start中定制MefConfig并添加处理Autofac的IocConfig.然后向全局asax添加对两者的调用:MefConfig.RegisterMef()和IocConfig.RegisterDependencies().

6)在其中创建另一个MVC4 Web应用程序(AlphaPlugin),更新所有NuGet预安装的软件包并添加Autofac MVC 4(RC)和Autofac.Mef.我选择了一个Web应用程序模板(而不是类库),因此我可以使用MVC的所有VS工具,并最终直接在那里进行一些测试.

7)将一个部件区域添加到主机Web,并从views根文件夹中将_ViewStart文件复制到其中.

8)在“部件”区域中添加可导出的控制器.我的名字叫AlphaController,它只有一个名为Hail的动作方法,它在ViewBag中输入一个字符串并返回默认视图.

9)回到主机,只需在主视图中添加插件控制器动作的链接,即可通过MEF进行测试.

现在,如果我构建所有并将AlphaPlugin.dll二进制文件复制到HostWeb Plugins文件夹中,我希望MVC通过MEF找到它,但随后抛出一个未找到视图的错误,因为我还没有复制主机网络中的任何内容文件.相反,我得到以下内容:

Value cannot be null.
Parameter name: type
Description: An unhandled exception occurred during the execution of the current web request. Please review the stack trace for more information about the error and where it originated in the code.

Exception Details: System.ArgumentNullException: Value cannot be null.
Parameter name: type

Source Error:
An unhandled exception was generated during the execution of the current web request. Information regarding the origin and location of the exception can be identified using the exception stack trace below.

Stack Trace:
[ArgumentNullException: Value cannot be null.
Parameter name: type]
   System.ComponentModel.Composition.Hosting.ExportProvider.GetExportsCore(Type type,Type metadataViewType,String contractName,ImportCardinality cardinality) +263923
   System.ComponentModel.Composition.Hosting.ExportProvider.GetExports(Type type,String contractName) +41
   MEF.MVC4.MefControllerFactory.GetControllerInstance(RequestContext requestContext,Type controllerType) +84
   System.Web.Mvc.DefaultControllerFactory.CreateController(RequestContext requestContext,String controllerName) +226
   System.Web.Mvc.MvcHandler.ProcessRequestInit(HttpContextBase httpContext,IController& controller,IControllerFactory& factory) +326
   System.Web.Mvc.MvcHandler.BeginProcessRequest(HttpContextBase httpContext,AsyncCallback callback,Object state) +177
   System.Web.Mvc.MvcHandler.BeginProcessRequest(HttpContext httpContext,Object state) +88
   System.Web.Mvc.MvcHandler.System.Web.IHttpAsyncHandler.BeginProcessRequest(HttpContext context,AsyncCallback cb,Object extraData) +50
   System.Web.CallHandlerExecutionStep.System.Web.HttpApplication.IExecutionStep.Execute() +301
   System.Web.HttpApplication.ExecuteStep(IExecutionStep step,Boolean& completedSynchronously) +155

这是最相关的代码(您可以在repro解决方案中找到它):这将处理Plugins文件夹内容,以便Web应用程序从副本加载它们:

[assembly: PreApplicationStartMethod(typeof(PreApplicationInit),"Initialize")]
static public class PreApplicationInit
{
    /// 
    /// The source plugin folder from which to shadow copy from.
    /// 
    /// This folder can contain sub folders to organize plugin types.
    internal static DirectoryInfo PluginFolder { get; private set; }

    /// 
    /// The folder to shadow copy the plugin DLLs to use for running the app.
    /// 
    internal static DirectoryInfo ShadowCopyFolder { get; private set; }

    static PreApplicationInit()
    {
        PluginFolder = new DirectoryInfo(HostingEnvironment.MapPath("~/Plugins"));
        ShadowCopyFolder = new DirectoryInfo(HostingEnvironment.MapPath("~/Plugins/Temp"));
    }

    public static void Initialize()
    {
        if (!Directory.Exists(ShadowCopyFolder.FullName))
            Directory.CreateDirectory(ShadowCopyFolder.FullName);
        else
        {
            foreach (FileInfo fi in ShadowCopyFolder.GetFiles("*.dll",SearchOption.AllDirectories))
            {
                try
                {
                    fi.Delete();
                }
                catch (Exception ex)
                {
                    // TODO log
                    Debug.WriteLine(ex.ToString());
                }
            }
        }

        // shadow copy files
        foreach (FileInfo fi in PluginFolder.GetFiles("*.dll"))
        {
            try
            {
                File.Copy(fi.FullName,Path.Combine(ShadowCopyFolder.FullName,fi.Name),true);
            }
            catch (Exception ex)
            {
                // TODO log
                Debug.WriteLine(ex.ToString());
            }
        }
    }
}

这是我的工厂,无论如何都得到一个null controllerType,所以它的代码永远不会超出它的第一行:

public class MefControllerFactory : DefaultControllerFactory
{
    private readonly CompositionContainer _compositionContainer;

    public MefControllerFactory(CompositionContainer compositionContainer)
    {
        _compositionContainer = compositionContainer;
    }

    protected override IController GetControllerInstance(System.Web.Routing.RequestContext requestContext,Type controllerType)
    {
        // https://stackoverflow.com/questions/719678/custom-controller-factory-dependency-injection-structuremap-problems-with-asp
        if (controllerType == null) return base.GetControllerInstance(requestContext,null);

        var export = _compositionContainer.GetExports(controllerType,null,null).SingleOrDefault();

        IController result;

        if (export != null) result = export.Value as IController;
        else
        {
            result = base.GetControllerInstance(requestContext,controllerType);
            _compositionContainer.ComposeParts(result);
        }

        return result;
    }

解决方法

控制器工厂需要找到适合外部MEF组件的正确控制器类型.像这样覆盖MefControllerFactory类的GetControllerType方法.
public class MefControllerFactory : DefaultControllerFactory
{
    protected override Type GetControllerType(RequestContext requestContext,string controllerName)
    {
        var controllerType = base.GetControllerType(requestContext,controllerName);

        if (controllerType == null)
        {
            var controller = _compositionContainer.GetExports<IController,IControllerMetaData>().SingleOrDefault(x => x.Metadata.ControllerName == controllerName).Value;

            if (controller != null)
            {
                return controller.GetType();
            }
       }
       return controllerType;
    }
}

其中IControllerMetaData是指定控制器名称的接口

public interface IControllerMetaData
{
    string ControllerName { get;}
}

并且您的控制器在元数据中指定controllerName.例如.

[Export (typeof(IController))]
[ExportMetadata("ControllerName","Home")]
[PartCreationPolicy(CreationPolicy.NonShared)]
public class HomeController : Controller,IController
{
    public ActionResult Index()
    {
        return new EmptyResult();
    }
}

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


引言 本文从Linux小白的视角, 在CentOS 7.x服务器上搭建一个Nginx-Powered AspNet Core Web准生产应用。 在开始之前,我们还是重温一下部署原理,正如你所常见的.Net Core 部署图: 在Linux上部署.Net Core App最好的方式是在Linux机器
引言: 多线程编程/异步编程非常复杂,有很多概念和工具需要去学习,贴心的.NET提供Task线程包装类和await/async异步编程语法糖简化了异步编程方式。 相信很多开发者都看到如下异步编程实践原则: 遵守以上冷冰冰的②③条的原则,可保证异步程序按照预期状态正常运作;我们在各大编程论坛常看到违背
一. 宏观概念 ASP.NET Core Middleware是在应用程序处理管道pipeline中用于处理请求和操作响应的组件。 每个组件是pipeline 中的一环。 自行决定是否将请求传递给下一个组件 在处理管道的下个组件执行之前和之后执行业务逻辑 二. 特性和行为 ASP.NET Core处
背景 在.Net和C#中运行异步代码相当简单,因为我们有时候需要取消正在进行的异步操作,通过本文,可以掌握 通过CancellationToken取消任务(包括non-cancellable任务)。 Task&#160;表示无返回值的异步操作, 泛型版本Task&lt;TResult&gt;表示有返
HTTP基本认证 在HTTP中,HTTP基本认证(Basic Authentication)是一种允许网页浏览器或其他客户端程序以(用户名:口令) 请求资源的身份验证方式,不要求cookie,session identifier、login page等标记或载体。 - 所有浏览器据支持HTTP基本认
1.Linq 执行多列排序 OrderBy的意义是按照指定顺序排序,连续两次OrderBy,后面一个有可能会打乱前面一个的排序顺序,可能与预期不符。 要实现sql中的order by word,name类似效果; LINQ 有ThenBy可以紧接使用, ThenBy记住原本排序的值,然后再排其他值,
ASP.NET Core 核心特性:开源、跨平台、高性能是其决战JAVA的必胜法宝,最引人关注的跨平台特性 到底是怎么实现? &#xA; 本文分Unix、Windows剖析跨平台内幕,读完让你大呼过瘾。
前导 Asynchronous programming Model(APM)异步编程模型以BeginMethod(...) 和 EndMethod(...)结对出现。 IAsyncResult BeginGetResponse(AsyncCallback callback, object state
引言 最近在公司开发了一个项目,项目部署架构图如下: 思路 如图中文本所述,公司大数据集群不允许直接访问外网,需要一个网关服务器代理请求,本处服务器A就是边缘代理服务器的作用。 通常技术人员最快捷的思路是在服务器A上部署IISʺpplication Request Routing Module组件
作为一枚后端程序狗,项目实践常遇到定时任务的工作,最容易想到的的思路就是利用Windows计划任务/wndows service程序/Crontab程序等主机方法在主机上部署定时任务程序/脚本。 但是很多时候,若使用的是共享主机或者受控主机,这些主机不允许你私自安装exe程序、Windows服务程序
引言 熟悉TPL Dataflow博文的朋友可能记得这是个单体程序,使用TPL Dataflow 处理工作流任务, 在使用Docker部署的过程中, 有一个问题一直无法回避: 在单体程序部署的瞬间(服务不可用)会有少量流量无法处理;更糟糕的情况下,迭代部署的这个版本有问题,上线后无法运作, 更多的流
合格的web后端程序员,除搬砖技能,还必须会给各种web服务器配置Https,本文结合ASP.NET Core部署模型聊一聊启用Https的方式。 温故知新 目前常见的Http请求明文传输,请求可能被篡改,访问的站点可能被伪造。 HTTPS是HTTP加上TLS/SSL协议构建的可进行加密传输、身份认
长话短说 前文《解剖HttpClientFactory,自由扩展HttpMessageHandler》主要讲如何为HttpClientFactory自定义HttpMessageHandler组件, 现在来完成课后的小作业: 将重点日志字段显示到Nlog的LayoutRenderer上。 本文实现一个
引言问题 作为资深老鸟,有事没事,出去面试;找准差距、定位价值。 面试必谈哈希, Q1:什么是哈希? Q2:哈希为什么快? Q3:你是怎么理解哈希算法利用空间换取时间的? Q4:你是怎么解决哈希冲突的? Q5:你有实际用写过哈希算法吗? 知识储备 哈希(也叫散列)是一种查找算法(可用于插入),哈希算
前言 如题,有感于博客园最近多次翻车,感觉像胡子眉毛一把抓, 定位不了生产环境的问题。 抛开流程问题,思考在生产环境中如何做故障排除,&#160;发现博客园里面这方面的文章比较少。 .Net 本身是提供了sos.dll工具帮助我们在生产中故障排除,通过提供有关内部公共语言运行时(CLR)环境的信息,
.NET程序是基于.NET Framework、.NET Core、Mono、【.NET实现】开发和运行的 ,定义以上【.NET实现】的标准规范称为.NET Standard .NET Standard .NET标准是一组API集合,由上层三种【.NET实现】的Basic Class Library
长话短说 上个月公司上线了一个物联网数据科学项目,我主要负责前端接受物联网事件,并提供 参数下载。 webapp 部署在Azure云上,参数使用Azure SQL Server存储。 最近从灰度测试转向全量部署之后,日志时常收到: SQL Session超限报错。 排查 我在Azure上使用的是 S
临近年关,搜狗,360浏览器出现页面无法成功跳转,同域Cookie丢失? 也许是服务端 SameSite惹的祸。&#xA;本文揭示由于Chrome低版本内核不识别 SameSite= None, 引发的单点登录故障。
本文聊一聊TraceID的作用和一般组成,衍生出ASP. NETCore 单体和分布式程序中 TraceId 的使用方式
通过给 HttpClint请求的日志增加 TraceId,解锁自定义扩展 HttpClientFacroty 的姿势