本系列目录:Abp介绍和经验分享-目录

Abp的模块系统支持插件机制,可以在指定目录中放置模块程序集,然后应用程序启动时会搜索该目录,加载其中所有程序集中的模块。

如何使用这套机制进行功能插件化开发?

首先,插件程序集和应用程序是毫无关系的,应用程序不依赖这个程序集,所以我们要解决几个常见问题:

  1. 插件中提供的功能需要权限认证,如何自动注册权限到使用了该插件的应用程序?
  2. 插件中提供的功能在展现层中需要菜单导航,如何自动注册菜单项目?
  3. 插件中提供的功能需要配置,如何让插件自已能进行简单的配置管理,而不用去改宿主的配置文件?
  4. 插件中提供了新的Mvc Controller,当然也需要注册路由。

以下代码从本系列QuickStartA中的Personball.Demo解决方案开始

开始我们的第一个插件程序集的开发

首先启动QuickStartA中一切就绪的HelloWorld,并且登陆进去看看首页

一切正常!

Step1 新建插件程序集

  1. 在解决方案中新建目录PlugIns,新建程序集项目Personball.PlugIns.PlugInZero
  2. 修改默认命名空间,由于这是一个插件,我选择默认命名空间为Personball.PlugIns,将PlugInZero作为这个示例插件的名称;
  3. 在程序包管理其控制台,选择默认项目PlugIns\Personball.PlugIns.PlugInZero,执行Install-Package Abp.Web.Mvc -Version 2.3(保证和宿主使用的Abp框架版本一致,可以减少很多不必要的麻烦,这里安装Abp.Web.Mvc是因为我们将在插件中实现一个MvcController);
  4. 在插件程序集中添加一个目录PlugInZero,移除默认的Class1.cs。

Step2 注册权限

Abp中权限构建基本都是通过继承AuthorizationProvider,实现SetPermissions方法,并添加到IAuthorizationConfiguration.Providers

插件本身也是一个模块,只要实现自己的AuthorizationProvider,并注册进Providers即可。

PlugInZero目录中定义我们的插件模块PlugInZeroModule,代码如下:

因本博客样式原因,源代码排版容易乱,下文大段代码全部贴图,文末附最终代码压缩包供下载。

PlugInZero目录下新建常量定义文件PlugInZeroConsts,代码如下:

namespace Personball.PlugIns.PlugInZero
{
public static class PlugInZeroConsts
{
public static class PermissionNames
{
public const string PlugIns = "Personball.PlugIns";
public const string PlugInZero = "Personball.PlugIns.PlugInZero";
}
}
}

PlugInZero目录下新建目录Authorization,新建类PlugInZeroAuthorizationProvider,代码如下:

再到插件模块PlugInZeroModule中注册上述权限:

public override void PreInitialize()
{
Configuration.Authorization.Providers
.Add<PlugInZeroAuthorizationProvider>();
//TODO 等会要用
}

Step3 等不及要先看看效果了,让我们启用Personball.Demo.Web的插件加载!

让我们打开Personball.Demo.Web项目的Global.asax文件,添加代码后如下:

  1. Personball.Demo.Web项目添加一个目录PlugIns;
  2. 生成插件程序集,将插件项目bin\debug目录中的Personball.PlugIns.PlugInZero.dllPersonball.PlugIns.PlugInZero.pdb复制到Personball.Demo.WebPlugIns目录下,启动Personball.Demo.Web!
  3. 启动后,登陆,点开Roles菜单,点开角色Admin的编辑弹窗,看权限的选项,确实增加了我们刚才在插件中定义的新权限!

如果你有Asp.Net Zero的代码(收费的),那么权限编辑功能是可以直接使用的,这里官网免费生成的项目仅包含了module-zero基础功能,UI部分并未实现权限编辑功能。

Step4 注册菜单

和权限类似,菜单通过继承NavigationProvider,实现SetNavigation方法,并添加到INavigationConfiguration.Providers

同上,插件可以实现自己的NavigationProvider,并注册。

PlugInZero目录下新建目录Navigation,新建类PlugInZeroNavigationProvider,代码如下:

再到插件模块PlugInZeroModule中注册菜单:

public override void PreInitialize()
{
//注册权限
Configuration.Authorization.Providers
.Add<PlugInZeroAuthorizationProvider>();
//注册菜单
Configuration.Navigation.Providers
.Add<Navigation.PlugInZeroNavigationProvider>();
}

来看看效果(生成插件程序集,并复制替换到Web项目的PlugIns目录下):

权限和菜单相对简单,接下来是重点。

Step5 注册路由和寻找Controller

首先我们提供一个PlugInZeroController,简单起见,仅有一个返回字符串的Action。

PlugInZero目录下新建目录Controller,新建类PlugInZeroController,代码如下:

public class PlugInZeroController : AbpController
{
public PlugInZeroController()
{
LocalizationSourceName = "Abp";
} public Task<string> Hello()
{
return Task.FromResult($"hello at {DateTime.Now}");
}
}

接着,我们注册下路由(在PlugInZeroModule的PreInitialize方法中):

//注册权限
Configuration.Authorization.Providers
.Add<PlugInZeroAuthorizationProvider>();
//注册菜单
Configuration.Navigation.Providers
.Add<Navigation.PlugInZeroNavigationProvider>();
//注册路由
RouteTable.Routes.MapRoute(
"Plugins",
url: "PlugIns/{controller}/{action}",
defaults: new { controller = "Home", action = "Index" });

修改一下菜单项上的url,并指定链接的target(在PlugInZeroNavigationProvider中):

plugInRoot.AddItem(
new MenuItemDefinition(
"PlugInZero",
new FixedLocalizableString("PlugInZero(插件)"),
//指定Controller的url
url: "PlugIns/PlugInZero/Hello",
icon: "",
target: "_blank"//新开一个窗口
));

更新插件程序集,启动,点击之前的插件菜单(如果遇到404,请参考下方Tip01):

Step6 插件配置

OK,Last Question:如何提供插件单独的配置?

思路是,从程序集的App.config入手!

如果插件自身使用数据库,有个DbContext,怎么读取配置并让IoC构建DbContext时使用这个配置?

右键插件项目,新增项目,选择应用程序配置文件,文件名自动就是App.config,添加!

编辑App.config,如下:

<?xml version="1.0" encoding="utf-8" ?>
<configuration>
<connectionStrings>
<add name="PlugInZeroDB" connectionString="localhost" providerName="System.Data.SqlClient"/>
</connectionStrings>
<appSettings>
<add key="PlugInZeroSettingKey" value="Wahoo, wahoo"/>
</appSettings>
</configuration>

改下PlugInZeroController的代码尝试读取PlugInZeroSettingKey并输出:

public Task<string> Hello()
{
var config = ConfigurationManager.OpenExeConfiguration(
Assembly.GetExecutingAssembly().Location);
var value = config.AppSettings
.Settings["PlugInZeroSettingKey"].Value;
return Task.FromResult($"hello at {DateTime.Now} {value}");
}

重新生成插件程序集,并复制三个文件(含Personball.PlugIns.PlugInZero.dll.config)到PlugIns目录中,启动:

读取数据库连接字符串,并让指定的DbContext(插件自己的)使用该配置,这里仅给出示例代码,并不实际运行演示。

//插件配置(直接从当前执行的程序集的config文件读取数据库连接串)
var config = ConfigurationManager.OpenExeConfiguration(
Assembly.GetExecutingAssembly().Location);
string connectStr = config.ConnectionStrings.ConnectionStrings["PlugInZeroDB"].ConnectionString;
//注册DbContext,构建时使用指定参数
IocManager.IocContainer.Register(
Component.For<PlugInZeroDbContext>()
.DependsOn(
Dependency.OnValue(
"connectionString", connectStr)));

Tip01

注意!注意!注意!

BuildManager对于ControllerType有缓存,在服务器上仅仅加载插件,注册路由,可能还是会遇到404(找不到Controller)。

这种时候必须改动对于iis敏感的几个路径(bin目录)或Web.config文件,BuildManager才会更新ControllerType的缓存(这是个文件缓存!),将插件内的Controller类型也算进去。

这里虽然只有寥寥数语,却是各种心酸血泪之后的总结,期间甚至自己扩展过一个ControllerFactory,那也是一套可行的方案,不赘述了。

搜了一个MVC-ControllerTypeCache.xml,内容如下(这里并未包含插件中的PlugInZeroController):

<?xml version="1.0" encoding="utf-8"?>
<!--This file is automatically generated. Please do not modify the contents of this file.-->
<typeCache lastModified="8/22/2017 12:00:44 AM" mvcVersionId="cc73190b-ab9d-435c-8315-10ff295c572a">
<assembly name="Personball.Demo.Web, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null">
<module versionId="9c27c37d-a073-42aa-b339-b5887549b123">
<type>Personball.Demo.Web.Controllers.AboutController</type>
<type>Personball.Demo.Web.Controllers.AccountController</type>
<type>Personball.Demo.Web.Controllers.HomeController</type>
<type>Personball.Demo.Web.Controllers.RolesController</type>
<type>Personball.Demo.Web.Controllers.TenantsController</type>
<type>Personball.Demo.Web.Controllers.UsersController</type>
<type>Personball.Demo.Web.Controllers.LayoutController</type>
</module>
</assembly>
<assembly name="Abp.Web.Mvc, Version=2.3.0.0, Culture=neutral, PublicKeyToken=null">
<module versionId="6bcf306f-dc82-4ad6-99be-7efe88288f89">
<type>Abp.Web.Mvc.Controllers.AbpAppViewController</type>
<type>Abp.Web.Mvc.Controllers.AbpScriptsController</type>
<type>Abp.Web.Mvc.Controllers.AbpUserConfigurationController</type>
<type>Abp.Web.Mvc.Controllers.Localization.AbpLocalizationController</type>
</module>
</assembly>
</typeCache>

Tip02

插件的dll文件替换时遇到进程锁定问题

请先停止iis站点或者应用程序池,替换插件dll后再启动

如果配置了CI,比如tfs使用webdeploy发布

务必请在发布时指定webdeploy的选项,忽略插件目录-skip:Directory="PlugIns"

插件一般考虑手动更新(大多是非核心的功能,变更极少),如果CI每次都要考虑重新build插件并更新,就得不偿失了。

假如非要CI每次更新插件程序集,那就需要先用webdeploy停止目标站点的应用程序池,发完后再启动应用程序池。

详情请参见下方参考条目:Operations on application pools as admin and non-admin

参考

ABP模块系统插件机制

Taming the BuildManager, ASP.Net Temp files and AppDomain restarts

Operations on application pools as admin and non-admin

本文源码下载

Personball.Demo.PlugIn.7z

[2017-08-21]Abp系列——如何使用Abp插件机制(注册权限、菜单、路由)的更多相关文章

  1. 基于DDD的现代ASP.NET开发框架--ABP系列文章总目录

    ABP相关岗位招聘:给热爱.NET新技术和ABP框架的朋友带来一个高薪的工作机会 ABP交流会录像视频:ABP架构设计交流群-7月18日上海线下交流会的内容分享(有高清录像视频的链接) 代码自动生成: ...

  2. 基于DDD的现代ASP.NET开发框架--ABP系列文章总目录(转)

    出处:http://www.cnblogs.com/mienreal/p/4528470.html ABP相关岗位招聘:给热爱.NET新技术和ABP框架的朋友带来一个高薪的工作机会 ABP交流会录像视 ...

  3. 浅入 ABP 系列(4):事件总线

    浅入 ABP 系列(4):事件总线 版权护体作者:痴者工良,微信公众号转载文章需要 <NCC开源社区>同意. 目录 浅入 ABP 系列(4):事件总线 事件总线 关于事件总线 为什么需要这 ...

  4. 基于DDD的现代ASP.NET开发框架--ABP系列之3、ABP分层架构

    基于DDD的现代ASP.NET开发框架--ABP系列之3.ABP分层架构 ABP是“ASP.NET Boilerplate Project (ASP.NET样板项目)”的简称. ABP的官方网站:ht ...

  5. 基于DDD的现代ASP.NET开发框架--ABP系列之2、ABP入门教程

    基于DDD的现代ASP.NET开发框架--ABP系列之2.ABP入门教程 ABP是“ASP.NET Boilerplate Project (ASP.NET样板项目)”的简称. ASP.NET Boi ...

  6. 点这里进入ABP系列文章总目录

    基于DDD的现代ASP.NET开发框架--ABP系列之1.ABP总体介绍 ABP是“ASP.NET Boilerplate Project (ASP.NET样板项目)”的简称. ASP.NET Boi ...

  7. [2017-09-04]Abp系列——为什么值对象必须设计成不可变的

    本系列目录:Abp介绍和经验分享-目录 这篇是之前翻备忘录发现漏了的,前阵子刚好同事又提及过这个问题,这里补上. 本文重点在于理解什么是值对象的不可变性. Abp的ValueObject以及EF的Co ...

  8. [2017-08-28]Abp系列——业务异常与错误码设计及提示语的本地化

    本系列目录:Abp介绍和经验分享-目录 前言 ABP中有个异常UserFriendlyException经常被使用,但是它所在的命名空间是Abp.UI,总觉得和展现层联系过于紧密,在AppServic ...

  9. [2017-08-16]ABP系列——QuickStartB:正确理解Abp解决方案的代码组织方式、分层和命名空间

    本系列目录:Abp介绍和经验分享-目录 介绍ABP的文章,大多会提到ABP框架吸收了很多最佳实践,比如: 1.N层 (复用一下上篇的图) 展现层(Personball.Demo.Web):asp.ne ...

随机推荐

  1. IDEA+Java:Selenium+Maven+TestNG基本WebUI自动化测试环境搭建

    IDEA+java:Selenium+Maven+TestNG 本文介绍的测试环境,应该是最基本的测试环境了,也是很多文章都有写,这里做一个完整的图文配置整理,方便阅读理解! 使用maven的好处,由 ...

  2. [学习笔记] CDQ分治 从感性理解到彻底晕菜

    最近学了一种叫做CDQ分治的东西...用于离线处理一系列操作与查询似乎跑得很快233 CDQ的名称似乎源于金牌选手陈丹琦 概述: 对于一坨操作和询问,分成两半,单独处理左半边和处理左半边对于右半边的影 ...

  3. HTML细节

    link   设置body中超链接默认颜色 : alink  设置body中超链接点击时候的颜色: vlink  设置body中超链接访问过后的颜色 字体的大小  size       大小 取值范围 ...

  4. 权限管理学习 一、ASP.NET Forms身份认证

    说明:本文示例使用的VS2017和MVC5. 系统无论大小.牛逼或屌丝,一般都离不开注册.登录.那么接下来我们就来分析下用户身份认证. 简单实现登录.注销 以前在学习.net的时候不知道什么Forms ...

  5. Azure 认知服务 (4) 计算机视觉API - 读取图片中的文字 (OCR)

    <Windows Azure Platform 系列文章目录> 微软Azure认知服务的计算机视觉API,还提供读取图片中的文字功能 在海外的Windows Azure认知服务的读取图片功 ...

  6. teamviewer试用期到期解决

    1.命令行输入:%appdata%删除teamviewer文件夹2.命令行输入:regedit删除teamviewer文件夹3.卸载tv4.控制面板->设备管理器->网络适配器->f ...

  7. Python用Pillow(PIL)进行简单的图像操作

    Python用Pillow(PIL)进行简单的图像操作 颜色与RGBA值 计算机通常将图像表示为RGB值,或者再加上alpha值(通透度,透明度),称为RGBA值.在Pillow中,RGBA的值表示为 ...

  8. cve-2017-0199&metasploit复现过程

    CVE-2017-0199 WORD/RTF嵌入OLE调用远程文件执行的一个漏洞.不需要用户交互.打开文档即中招 首先更新msf到最新,据说最新版简化了利用过程,不需要开启hta这一步.但没测成功 还 ...

  9. 一个普通的 Zepto 源码分析(三) - event 模块

    一个普通的 Zepto 源码分析(三) - event 模块 普通的路人,普通地瞧.分析时使用的是目前最新 1.2.0 版本. Zepto 可以由许多模块组成,默认包含的模块有 zepto 核心模块, ...

  10. Vue.js 计算属性的秘密

    计算属性是一个很邪门的东西,只要在它的函数里引用了 data 中的某个属性,当这个属性发生变化时,函数仿佛可以嗅探到这个变化,并自动重新执行. 上述代码会源源不断的打印出 b 的值.如果希望 a 依赖 ...