一个超轻量级工作流引擎: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…

【THUSC 2017】如果奇迹有颜色【polya引理】【矩阵】【计数dp】【BM打表+线性递推】

题意&#xff1a;长度为 nnn 的环染 mmm 种颜色&#xff0c;要求任意相邻 mmm 个元素不能包含全部的颜色。求方案数 模 109710^971097&#xff0c;循环同构。 n≤109,m≤7n\leq 10^9,m\leq7n≤109,m≤7 为啥我现在天天都在打表啊 先上 polya&#xff0c;对于移动 iii 位的置换…

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

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

Codeforces Round #588 (Div. 2) D. Marcin and Training Camp 思维

传送门 文章目录题意&#xff1a;思路&#xff1a;题意&#xff1a; 有nnn个人&#xff0c;每个人都有一个能力值bib_ibi​以及他会的技能aia_iai​&#xff0c;当他会第xxx个技能的时候&#xff0c;aia_iai​的第xxx位是111。定义当xxx不会某个技能但是yyy会的时候&#xff0…

架构杂谈《五》

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

【ROI 2019 Day2】课桌【贪心】【决策单调性】【分治】

题意&#xff1a;有 mmm 个班&#xff0c;每个班有 2n2n2n 个人&#xff0c;他们的身高给定。有 kkk 种双人桌&#xff0c;每张桌子有两个属性值 Li,RiL_i,R_iLi​,Ri​,一个身高为 hhh 的人坐第 iii 种桌子的不舒适度为 hhh 到区间 [Li,Ri][L_i,R_i][Li​,Ri​] 的最小距离。你…

Codeforces Round #588 (Div. 2) E. Kamil and Making a Stream 数学 + 暴力

传送门 文章目录题意&#xff1a;思路&#xff1a;题意&#xff1a; 给你一颗树&#xff0c;其中根是111&#xff0c;每个点有一个点权&#xff0c;求每个点到根的所有路径的gcdgcdgcd之和。 n≤1e5n\le1e5n≤1e5 思路&#xff1a; 一看到以为是个点分治&#xff0c;让后发现…

【NOIP模拟】彩色树【树形dp】【树链剖分性质】【复杂度分析】

题意&#xff1a;一棵初始时为空的树&#xff0c;依次加入 nnn 个叶结点&#xff0c;每次加入后询问 用若干不同颜色的路径将树边染色后 每个点到根经过的颜色数 的最大值 的最小值。 n≤106n\leq 10^6n≤106 首先发现这个路径没啥用&#xff0c;其实就是个剖分方案。 然后我…

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

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

NOIP2020 赛前总结

没有一眼秒的题都必须对拍&#xff0c;这里对拍包括疯狂造小数据人工检查。所以不会的话放心打暴力吧&#xff0c;反正写出来了也要对拍。码农题最后码。预估可能会失误&#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…

NOIP 2020 游记

Day -1 校内考试出阴间题&#xff0c;体验极差。 晚上写了发压位高精&#xff0c;一发 AC&#xff0c;感觉很飘&#xff08;flag&#xff09; Day 1 8:00 进考场 8:10 分发压缩包&#xff0c;随便点开看看&#xff0c;看题目名字感觉很友好。然后就在一个叫 ball 的文件夹…

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…

【WC2019】数树【子集反演】【结论】【树形dp】【生成函数】【函数求导】【多项式全家桶】

题意&#xff1a;有两棵基于同一点集的树&#xff0c;点集大小为 nnn &#xff0c;两棵树中有 opopop 棵未确定&#xff0c;可以取所有 nn−2n^{n-2}nn−2 种可能。给每个点染上 [1,y][1,y][1,y] 中的一个颜色&#xff0c;要求若 uuu 到 vvv 在两棵树上的路径完全相同&#xff…

Educational Codeforces Round 73 (Rated for Div. 2) F. Choose a Square 线段树 + 二维转一维

传送门 文章目录题意&#xff1a;思路&#xff1a;题意&#xff1a; 给你nnn个点(xi,yi)(x_i,y_i)(xi​,yi​)&#xff0c;每个点有个价值cic_ici​&#xff0c;现在你可以框一个正方形&#xff0c;要求左下角和右上角的坐标(x,y)(x,y)(x,y)必须xyxyxy&#xff0c;也就是说必须…

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

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

【CF113D】Museum【概率期望】【高斯消元】

题意&#xff1a;一张 nnn 个点的无向连通图&#xff0c;两个人开始时分别在 a,ba,ba,b。每次在 uuu 时会以 ppp 的概率原地不动&#xff0c;1−p1-p1−p 的概率等概率随机选择到一个相邻的点&#xff0c;当两人在同一点时停止。分别求在每个点相遇的概率。 n≤22n\leq 22n≤22…