一个超轻量级工作流引擎:Workflow-Core

640?wx_fmt=gif

近期工作上有一个工作流的开发需求,自己基于面向对象和职责链模式捣鼓了一套小框架,后来在github上发现一个轻量级的工作流引擎轮子:Workflow-Core,看完其wiki之后决定放弃之前自己造的轮子,使用这个开源项目来改造,也就有了这一篇博文。


01

关于Workflow-Core


        640?wx_fmt=png

Workflow-Core是一个基于.NET Standard的轻量级工作流引擎,其GitHub地址为:https://github.com/danielgerlag/workflow-core,目前有超过1200+个star。它提供了FluentAPI、多任务、持久化以及并行处理的功能,适合于小型工作流、责任链的需求开发。

  由于Workflow-Core支持工作流长期运行,因此Workflow-Core支持以下多种数据源格式的持久化,可以通过安装不同的Provider包来实现对应的持久化:

  • 内存(默认提供,用于测试和开发)

  • MongoDB

  • MS SQL Server

  • MySql

  • Sqlite

  • Redis

  • PostgreSQL

  立刻上手把,Nuget上安装一把,目前最新版本2.0.0:

PM> Install-Package WorkflowCore



02

Workflow-Core 基础使用

1、Hello World

这里创建了一个.NET Core控制台应用程序,快速演示第一个Workflow-Core的Hello World,展示如何开始一个Workflow:

  (1)定义一个实现IWorkflow接口的Workflow:

    public class HelloWorldWorkflow : IWorkflow	{	public string Id => "HelloWorld";	public int Version => 1;	public void Build(IWorkflowBuilder<object> builder)	{	builder	.StartWith<HelloWorld>()	.Then<ActiveWorld>()	.Then<GoodbyeWorld>();	}	}

  这里定义了一个HelloWorldWorkflow,其版本号为1,它有3个步骤:HelloWorld、ActiveWorld和GoodbyeWorld,会依次执行。

  (2)定义三个继承自StepBody类的步骤类:

    public class HelloWorld : StepBody	{	public override ExecutionResult Run(IStepExecutionContext context)	{	Console.WriteLine("Hello World!");	return ExecutionResult.Next();	}	}	public class ActiveWorld : StepBody	{	public override ExecutionResult Run(IStepExecutionContext context)	{	Console.WriteLine("I am activing in the World!");	return ExecutionResult.Next();	}	}	public class GoodbyeWorld : StepBody	{	public override ExecutionResult Run(IStepExecutionContext context)	{	Console.WriteLine("Goodbye World!");	return ExecutionResult.Next();	}	}	(3)ServiceCollection中注入Workflow-Core相关组件	private static IServiceProvider ConfigureServices()	{	IServiceCollection services = new ServiceCollection();	services.AddLogging(); // WorkflowCore需要用到logging service	services.AddWorkflow();	var serviceProvider = services.BuildServiceProvider();	return serviceProvider;	}

  (4)在Program.cs的Main方法中获取到注入的host并执行工作流

        public static void Main(string[] args)	{	var serviceProvider = ConfigureServices();	var host = serviceProvider.GetService<IWorkflowHost>();	host.RegisterWorkflow<HelloWorldWorkflow>();	host.Start();	// Demo1:Hello World	host.StartWorkflow("HelloWorld");	Console.ReadKey();	host.Stop();	}    

  这里传入的是Workflow的Id,Workflow-Core会根据Id去自动匹配最新版本的对应Workflow,运行结果如下:

  640?wx_fmt=png

2、无处不在的If

        在工作流处理中,往往会有很多的条件判断,那么在Workflow-Core中也提供了直接的If功能,如下面这个IfStatementWorkflow所示:

public class IfStatementWorkflow : IWorkflow<MyData>	{	public string Id => "if-sample";	public int Version => 1;	public void Build(IWorkflowBuilder<MyData> builder)	{	builder	.StartWith<SayHello>()	.If(data => data.Counter < 3).Do(then => then	.StartWith<PrintMessage>()	.Input(step => step.Message, data => "Outcome is less than 3")	)	.If(data => data.Counter < 5).Do(then => then	.StartWith<PrintMessage>()	.Input(step => step.Message, data => "Outcome is less than 5")	)	.Then<SayGoodbye>();	}	}

  这个传递进来的MyData的定义如下:

public class MyData	{	public int Counter { get; set; }	}

  当传递进来的MyData的Counter属性<3 或 <5时会有不同的分支进行逻辑的处理。

3、MySQL持久化支持

        想要将工作流配置持久化到MySQL,只需以下两步:

  (1)通过Nuget安装MySQL Provider包:

PM> Install-Package WorkflowCore.Persistence.MySQL

  (2)注入到ServiceCollection

services.AddWorkflow(x => x.UseMySQL(@"Server=127.0.0.1;Database=workflow;User=root;Password=password;", true, true));

  一旦启动,你就会发现Workflow-Core自动帮你创建了很多表用于持久化工作流配置和实例。

  640?wx_fmt=png

4、计划任务和循环任务

        Workflow-Core还继承了计划任务和循环任务的功能:

  (1)计划任务:比如在工作流步骤中设置一个延迟5分钟执行的计划任务

builder	.StartWith(context => Console.WriteLine("Hello"))	.Schedule(data => TimeSpan.FromSeconds(5)).Do(schedule => schedule	.StartWith(context => Console.WriteLine("Doing scheduled tasks"))	)	.Then(context => Console.WriteLine("Doing normal tasks"));

  (2)循环任务:比如在工作流步骤中设置一个延迟5分钟进行的循环任务,知道Counter > 5才结束

builder	.StartWith(context => Console.WriteLine("Hello"))	.Recur(data => TimeSpan.FromSeconds(5), data => data.Counter > 5).Do(recur => recur	.StartWith(context => Console.WriteLine("Doing recurring task"))	)	.Then(context => Console.WriteLine("Carry on"));

5、Saga支持

        了解分布式事务方案的童鞋应该都知道Saga,在Workflow-Core中也有支持,这是一个十分有用的功能:

  (1)比如:在创建一个客户信息之后,将其推送到Salesforce和ERP,如果推送过程中发生了错误,那么就通过重试进行补偿,并且重试有时间间隔。


builder	.StartWith<CreateCustomer>()	.Then<PushToSalesforce>()	.OnError(WorkflowErrorHandling.Retry, TimeSpan.FromMinutes(10))	.Then<PushToERP>()	.OnError(WorkflowErrorHandling.Retry, TimeSpan.FromMinutes(10));

  (2)又比如:当Task2发生异常时,Workflow-Core会帮助执行UndoTask2 和 UndoTask1 帮你回滚数据以恢复状态。


builder	.StartWith<LogStart>()	.Saga(saga => saga	.StartWith<Task1>()	.CompensateWith<UndoTask1>()	.Then<Task2>()	.CompensateWith<UndoTask2>()	.Then<Task3>()	.CompensateWith<UndoTask3>()	)	.OnError(Models.WorkflowErrorHandling.Retry, TimeSpan.FromMinutes(10))	.Then<LogEnd>();

  更多Saga示例,请参考:https://github.com/danielgerlag/workflow-core/tree/master/src/samples/WorkflowCore.Sample17


03

ASP.NET Core中使用Workflow-Core


1、注入与初始化

        (1)注入:使用AddWorkflow()扩展方法

      public void ConfigureServices(IServiceCollection services)	{	services.AddWorkflow();	services.AddMvc().SetCompatibilityVersion(CompatibilityVersion.Version_2_2);	}

  (2)初始化:

    public void Configure(IApplicationBuilder app, IHostingEnvironment env)	{	.......	app.UseWorkflow();	}

  扩展方法如下:

    public static class ConfigureExtensions	{	public static IApplicationBuilder UseWorkflow(this IApplicationBuilder app)	{	var host = app.ApplicationServices.GetService<IWorkflowHost>();	host.RegisterWorkflow<EdcWorkflow>();	host.RegisterWorkflow<EdcDataWorkflow, EdcData>();	host.Start();	var appLifetime = app.ApplicationServices.GetService<IApplicationLifetime>();	appLifetime.ApplicationStopping.Register(() =>	{	host.Stop();	});	return app;	}	}

  这里需要注意的就是:将你要用到的所有Workflow都事先进行Register注册。

        在你想要用到的地方,无论是Controller还是Service,通过依赖注入获取到Host,并使用它:

    [Route("api/[controller]")]	[ApiController]	public class ValuesController : ControllerBase	{	private IWorkflowController _workflowService;	public ValuesController(IWorkflowController workflowService)	{	_workflowService = workflowService;	}	// GET api/values	[HttpGet]	public async Task<IEnumerable<string>> Get()	{	await _workflowService.StartWorkflow("EdcWorkflow");	return new string[] { "EdcWorkflow v1" };	}	// GET api/values/5	[HttpGet("{id}")]	public async Task<string> Get(int id)	{	await _workflowService.StartWorkflow("EdcDataWorkflow", new EdcData() { Id = id });	return "EdcDataWorkflow v1";	}	}

  这两个Workflow的定义如下:

    public class EdcWorkflow : IWorkflow	{	public string Id => "EdcWorkflow";	public int Version => 1;	public void Build(IWorkflowBuilder<object> builder)	{	builder	.StartWith<HelloWorld>()	.Then<GoodbyeWorld>();	}	}	public class EdcDataWorkflow : IWorkflow<EdcData>	{	public string Id => "EdcDataWorkflow";	public int Version => 1;	public void Build(IWorkflowBuilder<EdcData> builder)	{	builder	.StartWith<HelloWorld>()	.If(data => data.Id < 3).Do(then => then	.StartWith<PrintMessage>()	.Input(step => step.Message, data => "Passed Id is less than 3")	)	.If(data => data.Id < 5).Do(then => then	.StartWith<PrintMessage>()	.Input(step => step.Message, data => "Passed Id is less than 5")	)	.Then<GoodbyeWorld>();	}	}

  示例结果很简单:

  (1)api/values

  640?wx_fmt=png

  (2)api/values/1

  640?wx_fmt=png


04

小结


640?wx_fmt=jpeg

Workflow-Core是一个适合.NET Core的优秀的轻量级工作流引擎,对于小型工作流和责任链类型的需求开发很适合,可以节约大量时间避免重复造轮子,将时间主要花在业务逻辑上面。当然,这里演示的示例只是众多功能特性中的一小部分,我只是选取了我用到的部分而已,大家有兴趣的话可以去GitHub上先给个star再仔细研究其wiki文档,应用到自己的项目中去。



恰童鞋骚年,风华不再正茂,仍想挥斥方遒

640?wx_fmt=jpeg



点个在看少个bug ?



本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/314867.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Codeforces Round #590 (Div. 3) E. Special Permutations 差分 + 思维

传送门 文章目录题意&#xff1a;思路&#xff1a;题意&#xff1a; 思路&#xff1a; 直接考虑比较难想&#xff0c;这种公式题基本都是将部分答案看成一个整体&#xff0c;考虑xi,xi1x_i,x_{i1}xi​,xi1​的贡献的。 假设当前的xix,xi1y,x<yx_ix,x_{i1}y,x<yxi​x,xi1…

ASP.NET Core 3.0中支持AI的生物识别安全

本文共两个部分&#xff0c;这是第一部分&#xff0c;其中介绍了 ASP.NET Core 3 中旨在将授权逻辑与基本的用户角色相分离的基于策略的授权模型。此部分提供了此授权进程的基于生物识别信息&#xff08;如人脸识别或语音识别&#xff09;的具体示例。在此示例中&#xff0c;检…

架构杂谈《五》

保证最终一致性的模式在大规模、高并发服务化系统中&#xff0c;一个功能被拆分成多个具有功能单一的子功能&#xff0c;一个流程会有多个系统的多个单一功能的服务组合实现&#xff0c;如果使用两阶段提交协议和三阶段提交协议&#xff0c;确实能解决系统间的一致性问题。其实…

在Linux的Windows子系统上(WSL)使用Docker(Ubuntu)

背景平时开发大部人都是在提供了高效GUI的window下工作&#xff0c;但是真正部署环境普遍都是在Linux中&#xff0c;所以为了让开发环境和部署环境统一&#xff0c;我们需要在windows模拟LInux环境&#xff0c;以前我们可能通过虚拟机的方式实现&#xff0c;不过自从微软拥抱开…

.net持续集成cake篇之cake介绍及简单示例

cake介绍Cake 是.net平台下的一款自动化构建工具,可以完成对.net项目的编译,打包,运行单元测试,集成测试甚至发布项目等等.如果有些特征Cake没有实现,我们还可以很容易地通过扩展Cake来实现我们想要的功能.Cake有以下特点1) 使用c#语言编写,可以在Cake脚本里使用C#语言来实现我…

Educational Codeforces Round 73 (Rated for Div. 2) Make The Fence Great Again dp + 结论

传送门 文章目录题意&#xff1a;思路&#xff1a;题意&#xff1a; 思路&#xff1a; 首先证明一个结论&#xff1a;一个数最多被加两次。 首先假设a[i]a[i−1]a[i]a[i-1]a[i]a[i−1]或a[i]a[i1]a[i]a[i1]a[i]a[i1]&#xff0c;那么此时可以将a[i]a[i]1a[i]a[i]1a[i]a[i]1&am…

.NET Core 3.0 发布小尺寸 self-contained 单体可执行程序

.NET Core 提供的发布应用程序选项 self-contained 是共享应用程序的好方法&#xff0c;因为应用程序的发布目录包含所有组件、运行时和框架。您只需要告诉使用者应用程序的入口 exe 文件&#xff0c;就可以使程序运行起来&#xff0c;而不必担心目标计算机上是否存在.NET Core…

Educational Codeforces Round 73 (Rated for Div. 2) E. Game With String 思维博弈 好题(2500)

传送门 文章目录题意&#xff1a;思路&#xff1a;题意&#xff1a; 思路&#xff1a; 我们将每一段...拿出来看成若干段&#xff0c;将其分成以下四种情况&#xff1a; (1)len<b(1)len<b(1)len<b (2)b≤len<a(2)b\le len<a(2)b≤len<a (3)a≤len<2∗b(3…

「Sqlserver」数据分析师有理由爱Sqlserver之三-最值得使有低投入高产出的Sqlserver功能...

数据分析师群体&#xff0c;有别于一般的传统开发群体和数据库运维群体&#xff0c;对Sqlserver的功能需求上也各不相同&#xff0c;本篇以笔者的亲身经历&#xff0c;用一种有别于一般性教程的角度来讲解Sqlserver值得我们学习&#xff0c;投入产出比高的一些功能模块。当然&a…

.NET开发框架(八)-服务器集群之网络负载平衡(视频)

【视频教程在文章底部】&#xff0c;本文讲解Windows服务器集群的网络负载平衡NLB的作用&#xff0c;以及在.NET开发框架的架构设计中&#xff0c;如何应用NLB与ARR&#xff0c;使用它们各有什么优点。视频教程目录&#xff1a;1、讲解NLB概念与演示其作用 2、安装配置负载均衡…

人生如戏,别太入戏

这里是Z哥的个人公众号每周五早8点 按时送达当然了&#xff0c;也会时不时加个餐&#xff5e;我的第「81」篇原创敬上大家好&#xff0c;我是Z哥。对&#xff0c;就是脑袋很大的那个。大到什么程度呢&#xff1f;我给新来的小伙伴们说说。我昨天还打算配副新眼镜来着&#xff0…

C#只用属性来解决兔子,不用方法和循环

属性在调用者看来就像一个普通的变量&#xff0c;但作为类的设计者&#xff0c;你可以利用属性来隐藏你类中的一些字段&#xff0c;使外界只能通过属性来访问你的字段&#xff0c;你可以通过属性来限制外界对你的字段的存取&#xff0c;就利用get、set。如果想让用户随意存取你…

.netcore项目docker化,以及docker之间通信

简言&#xff1a;最近刚完成公司的新系统&#xff0c;系统使用的是微服务架构&#xff0c;由于领导说要将服务docker化。下面将我的研究结果分享出来&#xff0c;如若有错误的地方&#xff0c;还请各位大佬多多指点。目录&#xff1a;  什么是docker&#xff1f;使用docker有…

Codeforces Round #592 (Div. 2) F. Chips 构造 + 细节

传送门 文章目录题意&#xff1a;思路&#xff1a;题意&#xff1a; 思路&#xff1a; 恶心的构造题&#xff0c;思路很简单但是代码细节很多&#xff0c;搞了半天。 根据题目的性质不难发现&#xff0c;如果有两个相同颜色的球相邻&#xff0c;那么他们的颜色永远不会改变。 …

.NET CORE 多语言实现方案

根据市场需求&#xff0c;基于.NET CORE平台开发的RoadFlow工作流平台需要实现多语言版本。经过方案讨论和比对&#xff0c;决定采用.NET自带的本地化功能来实现多语言。话不多说&#xff0c;直接上实现方式。首先修改Startup.cs在public void ConfigureServices(IServiceColle…

中国程序员,请挺起你的腰杆!

这两天发生一个事&#xff0c;登月50周年之际&#xff0c;阿波罗11号制导计算机&#xff08;AGC&#xff09;指令和登月模块的源代码在Github上发布公开了&#xff0c;大量中国人前往围观&#xff0c;把issues区当成了论坛版块灌水留言。猎奇起哄本为消遣作乐&#xff0c;有不妥…

Codeforces Round #592 (Div. 2) G. Running in Pairs 构造(水)

传送门 文章目录题意&#xff1a;思路&#xff1a;题意&#xff1a; 思路&#xff1a; 史上最水GGG题&#xff0c;没有之一。 考虑最小的情况如何构造&#xff0c;显然就是让a,ba,ba,b都1−n1-n1−n依次排列即可&#xff0c;这样的最小值为n∗(n1)2\frac{n*(n1)}{2}2n∗(n1)​…

Async,Await和ConfigureAwait的关系

在.NET Framework 4.5中&#xff0c;async / await关键字已添加到该版本中&#xff0c;简化多线程操作&#xff0c;以使异步编程更易于使用。为了最大化利用资源而不挂起UI&#xff0c;你应该尽可能地尝试使用异步编程。虽然async / await让异步编程更简单&#xff0c;但是有一…

Educational Codeforces Round 75 (Rated for Div. 2) E2. Voting (Hard Version) 贪心

传送门 文章目录题意&#xff1a;思路&#xff1a;题意&#xff1a; n≤2e5,m≤n,p≤1e9n\le2e5,m\le n,p\le 1e9n≤2e5,m≤n,p≤1e9 思路&#xff1a; 首先需要发现一些性质&#xff0c;假设preipre_iprei​代表所有mj<im_j< imj​<i的pjp_jpj​和。可以发现&#…

DevOps案例研究:知人善任——Google敏捷核心文化

内容来源&#xff1a;DevOps案例深度研究-Google敏捷实践战队&#xff0c;本文只展示部分PPT及研究成果&#xff0c;更多细节请关注案例分享会。本文内容贡献者&#xff1a;陈霁、刘翀、谈佳婧、张霖。阅读干货前先感受一下热烈的氛围~一、Google如何快速交付原型1.1 Savioke公…