一个.NET Core下的开源插件框架
插件模式历史悠久,各种中大型软件基本上都会实现插件机制,以此支持功能扩展,从开发部署层面,插件机制也可实现功能解耦,对于并行开发、项目部署、功能定制等都有比较大的优势。
在.NET Core下,一般我们基于.NET Core扩展库进行开发,通常使用依赖注入、配置、设置(Options)等机制,如果将插件模式与依赖注入、配置、设置进行结合,将可以提供非常灵活的扩展机制。基于此,我们实现了一个开源的插件框架,本文将进行简单的介绍。
目录
PluginFactory插件库
- 项目地址:
- Nuget包:
- Xfrogcn.PluginFactory 实现包,在主项目中引用
- Xfrogcn.PluginFactory.Abstractions 抽象包,在插件项目中引用,或者你可以完全重新实现自己的插件机制
- 主要功能:
- 插件的自动载入
- 通过插件的初始化接口可让插件控制主应用的依赖注入
- 插件的启动及停止,此机制可与.NET Core的Hosting扩展结合,在宿主启动时自动启动插件
- 抽象分离,你可以通过实现
Xfrogcn.PluginFactory.Abstractions
中的相关接口来完全实现自己的插件载入、启动等机制 - 与.NET Core配置、设置、宿主等完美融合
- 支持插件依赖程序集的多版本载入
主要概念
- 插件载入器:对应IPluginLoader接口,负责从指定位置加载插件程序集
- 插件工厂:对应IPluginFactory接口,负责插件的实例化及插件的启动和停止
- 插件:对应IPlugin接口,所有插件需要实现此接口,实现插件的启动及停止机制
- 可初始化插件:对应ISupportInitPlugin接口,通过此接口可以在依赖注入Provider构建之前向容器注入自定义的服务
- 可配置插件:对应ISupportConfigPlugin接口,通过此接口可将插件配置与.NET Core的配置(Configuration)及设置(Options)机制集合
使用向导
示例项目可参考:Xfrogcn.PluginFactory.Example
Gitee地址 Github地址
安装
在主程序项目中添加Xfrogcn.PluginFactory
包
```dotnet
dotnet add package Xfrogcn.PluginFactory
```
在插件项目中添加Xfrogcn.PluginFactory.Abstractions
包
```dotnet
dotnet add package Xfrogcn.PluginFactory.Abstractions
```
在主程序中启用
可通过以下两种方式来启用插件库,一是通过在Host
层级的Use机制以及在依赖注入IServiceCollection
层级的Add机制,以下分别说明:
通过IHostBuilder的UsePluginFactory方法启用插件库
```c#
var builder = Host.CreateDefaultBuilder(args);
builder.UsePluginFactory();
```
UsePluginFactory
具有多个重载版本,详细请查看API文档
默认配置下,将使用程序运行目录下的Plugins
目录作为插件程序集目录, 使用宿主配置文件作为插件配置文件(通常为appsettings.json)
你也可以通过使用带有Assembly
或IEnumerable<Assembly>
参数的版本直接传入插件所在的程序集
通过IServiceCollection的AddPluginFactory方法启用插件库
```c#
var builder = Host.CreateDefaultBuilder(args)
.ConfigureServices((hostContext, services) =>
{
services.AddPluginFactory();
});
```
AddPluginFactory
具有多个重载版本,详细请查看API文档
默认配置下,将使用程序运行目录下的Plugins
目录作为插件程序集目录
注意:
AddPluginFactory方法不会
使用默认的配置文件作为插件配置,你需要显式地传入IConfiguration
, 如果是在 ASP.NET Core 环境中,你可以在Startup类中直接获取到
编写插件
插件是实现了IPlugin接口的类,在插件库中也提供了PluginBase基类,一般从此类继承即可。标准插件具有启动和停止方法,通过IPluginFactory
进行控制。
要编写插件,一般遵循以下步骤:
创建插件项目(.NET Core 类库),如TestPluginA
添加
Xfrogcn.PluginFactory.Abstractions
包```nuget
dotnet add package Xfrogcn.PluginFactory.Abstractions
```
创建插件类,如Plugin,从PluginBase继承
```c#
public class Plugin : PluginBase
{
public override Task StartAsync(IPluginContext context)
{
Console.WriteLine("插件A已启动");
return base.StartAsync(context);
}
public override Task StopAsync(IPluginContext context)
{
Console.WriteLine("插件A已停止");
return base.StopAsync(context);
}
}
```
启动或停止方法中可通过context中的ServiceProvider获取注入服务
通过
PluginAttribute
特性设置插件的元数据```c#
[Plugin(Alias = "PluginA", Description = "测试插件")]
public class Plugin : PluginBase
{
}
```
插件元数据以及插件载入的插件列表信息可以通过
IPluginLoader.PluginList
获取
插件启动
IPluginFactory
本身实现了.NET Core扩展库的IHostedService
机制,故如果你是在宿主环境下使用,如(ASP.NET Core),插件的启动及停止将自动跟随宿主进行
如果未使用宿主,可通过获取IPluginFactory
实例调用相应方法来完成
```c#
// 手动启动
var pluginFactory = provider.GetRequiredService<IPluginFactory>();
await pluginFactory.StartAsync(default);
await pluginFactory.StopAsync(default);
```
编写支持初始化的插件
在很多场景,我们需要在插件中控制宿主的依赖注入,如注入新的服务等,这时候我们可通过实现支持初始化的插件(ISupportInitPlugin
)来实现,该接口的Init
方法将在依赖注入构建之前调用,通过方法参数IPluginInitContext
中的ServiceCollection
可以控制宿主注入容器。
```c#
[Plugin(Alias = "PluginA", Description = "测试插件")]
public class Plugin : PluginBase, ISupportInitPlugin
{
public void Init(IPluginInitContext context)
{
// 注入服务
//context.ServiceCollection.TryAddScoped<ICustomerService>();
}
}
```
使用插件配置
插件支持 .NET Core 扩展库中的Options及Configuration机制,你只需要从SupportConfigPluginBase<TOptions>
类继承实现插件即可,其中TOptions泛型为插件的配置类型。插件配置自动从宿主配置或启用插件工厂时传入的配置中获取,插件配置位于配置下的Plugins节点,该节点下以插件类名称或插件别名(通过PluginAttribute
特性指定)作为键名,此键之下为插件的配置,如以下配置文件:
```appsettings.json
{
"Plugins": {
"PluginA": {
"TestConfig": "Hello World"
},
}
}
```
扩展PluginA实现配置:
定义配置类,如PluginOptions
```c#
public class PluginOptions
{
public string TestConfig { get; set; }
}
```
实现插件
```c#
[Plugin(Alias = "PluginA", Description = "测试插件")]
public class Plugin : SupportConfigPluginBase<PluginOptions>, ISupportInitPlugin
{
public Plugin(IOptionsMonitor<PluginOptions> options) : base(options)
{
}
public void Init(IPluginInitContext context)
{
// 注入服务
//context.ServiceCollection.TryAddScoped<ICustomerService>();
Console.WriteLine($"Init 插件配置:{Options.TestConfig}");
}
public override Task StartAsync(IPluginContext context)
{
Console.WriteLine("插件A已启动");
Console.WriteLine($"StartAsync 插件配置:{Options.TestConfig}");
return base.StartAsync(context);
}
public override Task StopAsync(IPluginContext context)
{
Console.WriteLine("插件A已停止");
return base.StopAsync(context);
}
```
注意:在插件初始化方法中也可使用注入的配置
跨插件配置
有些配置可能需要在多个插件中共享,此时你可通过Plugins
下的_Share
节点进行配置,此节点下配置将会被合并到插件配置中,可通过PluginOptions进行访问。
```appsettings.json
{
"Plugins": {
"PluginA": {
},
"_Share": {
"TestConfig": "Hello World"
}
}
}
```
插件化 ASP.NET Core
要让 ASP.NET Core 获取得到插件中的控制器,你只需要在插件的初始化方法Init
中,向MVC注入插件程序集:
```c#
context.ServiceCollection.AddMvcCore()
.AddApplicationPart(typeof(Plugin).Assembly);
```
一个.NET Core下的开源插件框架的更多相关文章
- Asp.Net Core下的开源任务调度平台ScheduleMaster
从何说起 2017年初的时候,由于当时项目需要做了一个乞丐版定时调度系统,那时候只在单机上实现了核心的调度功能.做这个玩意之前也调研了社区中开源的解决方案,找了几个实地部署试跑了一下,其实都很不错.但 ...
- Asp.Net Core下的开源任务调度平台ScheduleMaster—快速上手
概述 ScheduleMaster是一个开源的分布式任务调度系统,它基于Asp.Net Core平台构建,支持跨平台多节点部署运行. 它的项目主页在这里: https://github.com/hey ...
- .NET Core下的开源分布式任务调度系统ScheduleMaster-v2.0低调发布
从1月份首次公开介绍这个项目到现在也快4个月了,期间做了一些修修补补整体没什么大的改动.2.0算是发布之后第一个大的版本更新,带来了许多新功能新特性,也修复了一些已知的bug,在此感谢在博客.Issu ...
- .net core 下调用.net framework框架的WCF方法写法
通过添加服务引用后生成的代码,可以得知首先要设置Basic连接写法的属性,并且设置WCF服务的地址: 我在这里建立工厂类如下: using System; using System.ServiceMo ...
- Quartz.NET开源作业调度框架系列(一):快速入门step by step
Quartz.NET是一个被广泛使用的开源作业调度框架 , 由于是用C#语言创建,可方便的用于winform和asp.net应用程序中.Quartz.NET提供了巨大的灵活性但又兼具简单性.开发人员可 ...
- Quartz.NET开源作业调度框架系列(一):快速入门step by step-转
Quartz.NET是一个被广泛使用的开源作业调度框架 , 由于是用C#语言创建,可方便的用于winform和asp.net应用程序中.Quartz.NET提供了巨大的灵活性但又兼具简单性.开发人员可 ...
- Quartz.NET开源作业调度框架系列
Quartz.NET是一个被广泛使用的开源作业调度框架 , 由于是用C#语言创建,可方便的用于winform和asp.net应用程序中.Quartz.NET提供了巨大的灵活性但又兼具简单性.开发人员可 ...
- [转]C/C++:构建你自己的插件框架
本文译自Gigi Sayfan在DDJ上的专栏文章.Gigi Sayfan是北加州的一个程序员,email:gigi@gmail.com. 本文是一系列讨论架构.开发和部署C/C++跨平台插件框架的文 ...
- 分享在Linux下使用OSGi.NET插件框架快速实现一个分布式服务集群的方法
在这篇文章我分享了如何使用分层与模块化的方法来设计一个分布式服务集群.这个分布式服务集群是基于DynamicProxy.WCF和OSGi.NET插件框架实现的.我将从设计思路.目标和实现三方面来描述. ...
随机推荐
- >>8) & 0xFF中的 >> 和 &0xFF 的作用
参考:https://blog.csdn.net/iamgamer/article/details/79354617 其中有两个位运算,一个是>>,一个是&. 0xff的作用一: ...
- 【题解】hdu4757 【TJOI2018】异或
题目链接 题目大意:有一颗树,有点权,每次询问:一条路径\(x->y\)中与\(z\)异或的最大值,或是以\(x\)为根的子树中与\(y\)异或的最大值. 树剖--还是算了. 观察到,子树的\( ...
- Metasploit简单使用——后渗透阶段
在上文中我们复现了永恒之蓝漏洞,这里我们学习一下利用msf简单的后渗透阶段的知识/ 一.meterperter常用命令 sysinfo #查看目标主机系统信息 run scraper #查看目标主机详 ...
- 【3】Java面试-Servlet
Servlet面试问题 Q1.什么是servlet? Java Servlet是服务器端技术,通过提供对动态响应和数据持久性的支持来扩展Web服务器的功能. javax.servlet和javax.s ...
- 利用 Python 插入 Oracle 数据
# coding=utf-8 ''''' Created on 2020-01-05 @author: Mr. Zheng ''' import json; import urllib2 import ...
- 多测师讲解jmeter _接口请求_(003)高级讲师肖sir
1.简单接口的请求 2. 3. 正则查看: 正则提取:在后置处理器中正则请求 设置:正则表达式 JSESSIONID提取器: Debug sampler 总结:
- JDBC的学习(一)
JDBC的学习(一) 概念 所谓英文简写的意思是:Java DataBase Connectivity ,即 Java数据库的连接,用Java语言来操作数据库 本质 简单的来说,就是写这个JDBC的公 ...
- harbor搭建与使用
前两天测试服务docker化并k8s布署时,出于方便,使用了docker hub.由于我们的代码是要放到镜像里的,通过运行容器,便能获取我们的全部代码,风险很大.所以我们决定进行私有化的镜像部 ...
- rabbitmq之后台管理和用户设置
前言 前面介绍了erlang环境的安装和rabbitmq环境安装,接下来介绍rabbitmq的web管理和用户设置. 启用后台管理插件 通过后台管理插件我们可以动态监控mq的流量,创建用户,队列等. ...
- centos7下面 es7.5 搭建
centos6 搭建 参考 https://www.cnblogs.com/php-linux/p/8758788.html 搭建linux虚拟机 https://www.cnblogs.com/ph ...