插件模式历史悠久,各种中大型软件基本上都会实现插件机制,以此支持功能扩展,从开发部署层面,插件机制也可实现功能解耦,对于并行开发、项目部署、功能定制等都有比较大的优势。

在.NET Core下,一般我们基于.NET Core扩展库进行开发,通常使用依赖注入、配置、设置(Options)等机制,如果将插件模式与依赖注入、配置、设置进行结合,将可以提供非常灵活的扩展机制。基于此,我们实现了一个开源的插件框架,本文将进行简单的介绍。

目录

PluginFactory插件库

  • 项目地址:

  • Nuget包:
  • 主要功能:
    • 插件的自动载入
    • 通过插件的初始化接口可让插件控制主应用的依赖注入
    • 插件的启动及停止,此机制可与.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

  1. ```dotnet
  2. dotnet add package Xfrogcn.PluginFactory
  3. ```

在插件项目中添加Xfrogcn.PluginFactory.Abstractions

  1. ```dotnet
  2. dotnet add package Xfrogcn.PluginFactory.Abstractions
  3. ```

在主程序中启用

可通过以下两种方式来启用插件库,一是通过在Host层级的Use机制以及在依赖注入IServiceCollection层级的Add机制,以下分别说明:

通过IHostBuilder的UsePluginFactory方法启用插件库

  1. ```c#
  2. var builder = Host.CreateDefaultBuilder(args);
  3. builder.UsePluginFactory();
  4. ```

UsePluginFactory具有多个重载版本,详细请查看API文档

默认配置下,将使用程序运行目录下的Plugins目录作为插件程序集目录, 使用宿主配置文件作为插件配置文件(通常为appsettings.json)

你也可以通过使用带有AssemblyIEnumerable<Assembly>参数的版本直接传入插件所在的程序集

通过IServiceCollection的AddPluginFactory方法启用插件库

  1. ```c#
  2. var builder = Host.CreateDefaultBuilder(args)
  3. .ConfigureServices((hostContext, services) =>
  4. {
  5. services.AddPluginFactory();
  6. });
  7. ```

AddPluginFactory具有多个重载版本,详细请查看API文档

默认配置下,将使用程序运行目录下的Plugins目录作为插件程序集目录

注意: AddPluginFactory方法不会使用默认的配置文件作为插件配置,你需要显式地传入IConfiguration, 如果是在 ASP.NET Core 环境中,你可以在Startup类中直接获取到

编写插件

插件是实现了IPlugin接口的类,在插件库中也提供了PluginBase基类,一般从此类继承即可。标准插件具有启动和停止方法,通过IPluginFactory进行控制。

要编写插件,一般遵循以下步骤:

  1. 创建插件项目(.NET Core 类库),如TestPluginA

  2. 添加Xfrogcn.PluginFactory.Abstractions

    1. ```nuget
    2. dotnet add package Xfrogcn.PluginFactory.Abstractions
    3. ```
  3. 创建插件类,如Plugin,从PluginBase继承

    1. ```c#
    2. public class Plugin : PluginBase
    3. {
    4. public override Task StartAsync(IPluginContext context)
    5. {
    6. Console.WriteLine("插件A已启动");
    7. return base.StartAsync(context);
    8. }
    9. public override Task StopAsync(IPluginContext context)
    10. {
    11. Console.WriteLine("插件A已停止");
    12. return base.StopAsync(context);
    13. }
    14. }
    15. ```

    启动或停止方法中可通过context中的ServiceProvider获取注入服务

  4. 通过PluginAttribute特性设置插件的元数据

    1. ```c#
    2. [Plugin(Alias = "PluginA", Description = "测试插件")]
    3. public class Plugin : PluginBase
    4. {
    5. }
    6. ```

    插件元数据以及插件载入的插件列表信息可以通过IPluginLoader.PluginList获取

插件启动

IPluginFactory本身实现了.NET Core扩展库的IHostedService机制,故如果你是在宿主环境下使用,如(ASP.NET Core),插件的启动及停止将自动跟随宿主进行

如果未使用宿主,可通过获取IPluginFactory实例调用相应方法来完成

  1. ```c#
  2. // 手动启动
  3. var pluginFactory = provider.GetRequiredService<IPluginFactory>();
  4. await pluginFactory.StartAsync(default);
  5. await pluginFactory.StopAsync(default);
  6. ```

编写支持初始化的插件

在很多场景,我们需要在插件中控制宿主的依赖注入,如注入新的服务等,这时候我们可通过实现支持初始化的插件(ISupportInitPlugin)来实现,该接口的Init方法将在依赖注入构建之前调用,通过方法参数IPluginInitContext中的ServiceCollection可以控制宿主注入容器。

  1. ```c#
  2. [Plugin(Alias = "PluginA", Description = "测试插件")]
  3. public class Plugin : PluginBase, ISupportInitPlugin
  4. {
  5. public void Init(IPluginInitContext context)
  6. {
  7. // 注入服务
  8. //context.ServiceCollection.TryAddScoped<ICustomerService>();
  9. }
  10. }
  11. ```

使用插件配置

插件支持 .NET Core 扩展库中的Options及Configuration机制,你只需要从SupportConfigPluginBase<TOptions>类继承实现插件即可,其中TOptions泛型为插件的配置类型。插件配置自动从宿主配置或启用插件工厂时传入的配置中获取,插件配置位于配置下的Plugins节点,该节点下以插件类名称或插件别名(通过PluginAttribute特性指定)作为键名,此键之下为插件的配置,如以下配置文件:

  1. ```appsettings.json
  2. {
  3. "Plugins": {
  4. "PluginA": {
  5. "TestConfig": "Hello World"
  6. },
  7. }
  8. }
  9. ```

扩展PluginA实现配置:

  1. 定义配置类,如PluginOptions

    1. ```c#
    2. public class PluginOptions
    3. {
    4. public string TestConfig { get; set; }
    5. }
    6. ```
  2. 实现插件

    1. ```c#
    2. [Plugin(Alias = "PluginA", Description = "测试插件")]
    3. public class Plugin : SupportConfigPluginBase<PluginOptions>, ISupportInitPlugin
    4. {
    5. public Plugin(IOptionsMonitor<PluginOptions> options) : base(options)
    6. {
    7. }
    8. public void Init(IPluginInitContext context)
    9. {
    10. // 注入服务
    11. //context.ServiceCollection.TryAddScoped<ICustomerService>();
    12. Console.WriteLine($"Init 插件配置:{Options.TestConfig}");
    13. }
    14. public override Task StartAsync(IPluginContext context)
    15. {
    16. Console.WriteLine("插件A已启动");
    17. Console.WriteLine($"StartAsync 插件配置:{Options.TestConfig}");
    18. return base.StartAsync(context);
    19. }
    20. public override Task StopAsync(IPluginContext context)
    21. {
    22. Console.WriteLine("插件A已停止");
    23. return base.StopAsync(context);
    24. }
    25. ```

    注意:在插件初始化方法中也可使用注入的配置

  3. 跨插件配置

有些配置可能需要在多个插件中共享,此时你可通过Plugins下的_Share节点进行配置,此节点下配置将会被合并到插件配置中,可通过PluginOptions进行访问。

  1. ```appsettings.json
  2. {
  3. "Plugins": {
  4. "PluginA": {
  5. },
  6. "_Share": {
  7. "TestConfig": "Hello World"
  8. }
  9. }
  10. }
  11. ```

插件化 ASP.NET Core

要让 ASP.NET Core 获取得到插件中的控制器,你只需要在插件的初始化方法Init中,向MVC注入插件程序集:

  1. ```c#
  2. context.ServiceCollection.AddMvcCore()
  3. .AddApplicationPart(typeof(Plugin).Assembly);
  4. ```

一个.NET Core下的开源插件框架的更多相关文章

  1. Asp.Net Core下的开源任务调度平台ScheduleMaster

    从何说起 2017年初的时候,由于当时项目需要做了一个乞丐版定时调度系统,那时候只在单机上实现了核心的调度功能.做这个玩意之前也调研了社区中开源的解决方案,找了几个实地部署试跑了一下,其实都很不错.但 ...

  2. Asp.Net Core下的开源任务调度平台ScheduleMaster—快速上手

    概述 ScheduleMaster是一个开源的分布式任务调度系统,它基于Asp.Net Core平台构建,支持跨平台多节点部署运行. 它的项目主页在这里: https://github.com/hey ...

  3. .NET Core下的开源分布式任务调度系统ScheduleMaster-v2.0低调发布

    从1月份首次公开介绍这个项目到现在也快4个月了,期间做了一些修修补补整体没什么大的改动.2.0算是发布之后第一个大的版本更新,带来了许多新功能新特性,也修复了一些已知的bug,在此感谢在博客.Issu ...

  4. .net core 下调用.net framework框架的WCF方法写法

    通过添加服务引用后生成的代码,可以得知首先要设置Basic连接写法的属性,并且设置WCF服务的地址: 我在这里建立工厂类如下: using System; using System.ServiceMo ...

  5. Quartz.NET开源作业调度框架系列(一):快速入门step by step

    Quartz.NET是一个被广泛使用的开源作业调度框架 , 由于是用C#语言创建,可方便的用于winform和asp.net应用程序中.Quartz.NET提供了巨大的灵活性但又兼具简单性.开发人员可 ...

  6. Quartz.NET开源作业调度框架系列(一):快速入门step by step-转

    Quartz.NET是一个被广泛使用的开源作业调度框架 , 由于是用C#语言创建,可方便的用于winform和asp.net应用程序中.Quartz.NET提供了巨大的灵活性但又兼具简单性.开发人员可 ...

  7. Quartz.NET开源作业调度框架系列

    Quartz.NET是一个被广泛使用的开源作业调度框架 , 由于是用C#语言创建,可方便的用于winform和asp.net应用程序中.Quartz.NET提供了巨大的灵活性但又兼具简单性.开发人员可 ...

  8. [转]C/C++:构建你自己的插件框架

    本文译自Gigi Sayfan在DDJ上的专栏文章.Gigi Sayfan是北加州的一个程序员,email:gigi@gmail.com. 本文是一系列讨论架构.开发和部署C/C++跨平台插件框架的文 ...

  9. 分享在Linux下使用OSGi.NET插件框架快速实现一个分布式服务集群的方法

    在这篇文章我分享了如何使用分层与模块化的方法来设计一个分布式服务集群.这个分布式服务集群是基于DynamicProxy.WCF和OSGi.NET插件框架实现的.我将从设计思路.目标和实现三方面来描述. ...

随机推荐

  1. 获取豆瓣读书所有热门标签并保存到mongodb数据库

    目标url:https://book.douban.com/tag/?view=type&icn=index-sorttags-all 目的:抓取所有标签名称(tag_name),标签链接(t ...

  2. 第0天 | 12天搞定Pyhon,前言

    依稀记得,在2014年的某一天,一位运营电商平台的多年好朋友,找我帮忙:一个月内,实现抓取竞争对手在某电商平台上的所有产品信息并统计每个产品的点击率. 说出来有些不好意思,那些年,参与过的产品挺多的, ...

  3. 《流畅的Python》第三部分 把函数视作对象 【一等函数】【使用一等函数实现设计模式】【函数装饰器和闭包】

    第三部分 第5章 一等函数 一等对象 在运行时创建 能赋值给变量或数据结构中的元素 能作为参数传递给函数 能作为函数的返回结果 在Python中,所有函数都是一等对象 函数是对象 函数本身是 func ...

  4. CAD& CG 2020 胡事民教授—开源框架Jittor的创新与探索

    题目:深度学习框架"计图"的创新与探索 报告人:胡事民 报告人简介:胡事民,清华大学计算机系教授,主要研究方向为计算机图形学.虚拟现实.智能信息处理和系统软件等. 报告简介:深度学 ...

  5. 判断移动还是PC 以及微信环境

    //判断pc还是移动端 function IsPC() {   var userAgentInfo = navigator.userAgent;   var Agents = ["Andro ...

  6. Python数据类型--字典(dict)

    Python中的字典是键值对(key-value)的无序集合.每个元素包含"键"和"值"两部分,这两部分之间使用冒号分隔,表示一种对应关系.不同元素之间用逗号分 ...

  7. Docker-V 详解

      1. 作用 挂载宿主机的一个目录. 2. 案例 譬如我要启动一个centos容器,宿主机的/test目录挂载到容器的/soft目录,可通过以下方式指定:   # docker run -it -v ...

  8. centos8平台redis cluster集群搭建(redis5.0.7)

    一,规划 redis cluster 1,cluster采用六台redis,3主3从 redis1    : ip: 172.17.0.2 redis2    : ip: 172.17.0.3 red ...

  9. 爬虫学习之-scrapy交互式命令 scrapy查看页面

    scrapy shell https:///www.baidu.com  会启动爬虫请求网页 view(response) 会在浏览器打开请求到的临时文件 response.xpath("/ ...

  10. 闭包 - Js函数笔记

    闭包 当函数被保存到外部时,将会生成闭包 闭包会导致原有作用域链不释放,造成内存泄漏 类似的代码就叫闭包 闭包的运行作用域 代码 a被执行,b被定义并保存出来 a结束,b被执行时,a的执行期上下文指向 ...