2024-09-12 20:32:54 +08:00
|
|
|
|
using Serein.Library.Api;
|
|
|
|
|
|
using Serein.Library.Core.NodeFlow;
|
2024-09-15 12:15:32 +08:00
|
|
|
|
using Serein.Library.Entity;
|
|
|
|
|
|
using Serein.Library.Enums;
|
|
|
|
|
|
using Serein.Library.Utils;
|
2024-09-17 15:58:37 +08:00
|
|
|
|
using Serein.Library.Web;
|
2024-09-15 12:15:32 +08:00
|
|
|
|
using Serein.NodeFlow.Base;
|
2024-08-06 16:09:46 +08:00
|
|
|
|
using Serein.NodeFlow.Model;
|
2024-09-18 16:45:41 +08:00
|
|
|
|
using System.ComponentModel.Design;
|
2024-09-21 10:06:44 +08:00
|
|
|
|
using System.Runtime.CompilerServices;
|
2024-09-22 14:10:13 +08:00
|
|
|
|
using System.Xml.Linq;
|
2024-09-20 10:50:32 +08:00
|
|
|
|
using static Serein.Library.Utils.ChannelFlowInterrupt;
|
2024-08-06 16:09:46 +08:00
|
|
|
|
|
|
|
|
|
|
namespace Serein.NodeFlow
|
|
|
|
|
|
{
|
|
|
|
|
|
|
2024-09-15 12:15:32 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 流程启动器
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="serviceContainer"></param>
|
|
|
|
|
|
/// <param name="methodDetails"></param>
|
2024-09-15 19:48:27 +08:00
|
|
|
|
public class FlowStarter
|
2024-08-06 16:09:46 +08:00
|
|
|
|
{
|
2024-09-16 21:38:34 +08:00
|
|
|
|
public FlowStarter()
|
2024-09-15 19:48:27 +08:00
|
|
|
|
{
|
2024-09-16 21:38:34 +08:00
|
|
|
|
SereinIOC = new SereinIOC();
|
2024-09-15 19:48:27 +08:00
|
|
|
|
}
|
2024-09-20 10:50:32 +08:00
|
|
|
|
|
2024-09-16 21:38:34 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 流程运行状态
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public enum RunState
|
|
|
|
|
|
{
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 等待开始
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
NoStart,
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 正在运行
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
Running,
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 运行完成
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
Completion,
|
|
|
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
2024-09-21 10:06:44 +08:00
|
|
|
|
/// 控制触发器
|
2024-09-16 21:38:34 +08:00
|
|
|
|
/// </summary>
|
2024-09-22 14:10:13 +08:00
|
|
|
|
private CancellationTokenSource _flipFlopCts = null;
|
2024-09-21 10:06:44 +08:00
|
|
|
|
public const string FlipFlopCtsName = "<>.FlowFlipFlopCts";
|
2024-09-16 21:38:34 +08:00
|
|
|
|
|
2024-09-21 10:06:44 +08:00
|
|
|
|
public bool IsStopStart = false;
|
2024-09-16 21:38:34 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 运行状态
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public RunState FlowState { get; private set; } = RunState.NoStart;
|
|
|
|
|
|
public RunState FlipFlopState { get; private set; } = RunState.NoStart;
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 运行时的IOC容器
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
private ISereinIOC SereinIOC { get; } = null;
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 结束运行时需要执行的方法
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
private Action ExitAction { get; set; } = null;
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 运行的上下文
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
private IDynamicContext Context { get; set; } = null;
|
2024-09-21 10:06:44 +08:00
|
|
|
|
private void CheckStartState()
|
|
|
|
|
|
{
|
|
|
|
|
|
if (IsStopStart)
|
|
|
|
|
|
{
|
|
|
|
|
|
throw new Exception("停止启动");
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2024-09-22 14:10:13 +08:00
|
|
|
|
// <summary>
|
|
|
|
|
|
// 开始运行
|
|
|
|
|
|
// </summary>
|
|
|
|
|
|
// <param name="startNode">起始节点</param>
|
|
|
|
|
|
// <param name="env">运行环境</param>
|
|
|
|
|
|
// <param name="runNodeMd">环境中已加载的所有节点方法</param>
|
|
|
|
|
|
// <param name="flipflopNodes">触发器节点</param>
|
|
|
|
|
|
// <returns></returns>
|
|
|
|
|
|
|
2024-08-06 16:09:46 +08:00
|
|
|
|
/// <summary>
|
2024-09-15 12:15:32 +08:00
|
|
|
|
/// 开始运行
|
2024-08-06 16:09:46 +08:00
|
|
|
|
/// </summary>
|
2024-09-16 21:38:34 +08:00
|
|
|
|
/// <param name="env">运行环境</param>
|
2024-09-22 14:10:13 +08:00
|
|
|
|
/// <param name="nodes">环境中已加载的所有节点</param>
|
|
|
|
|
|
/// <param name="initMethods">初始化方法</param>
|
|
|
|
|
|
/// <param name="loadingMethods">加载时方法</param>
|
|
|
|
|
|
/// <param name="exitMethods">结束时方法</param>
|
2024-09-15 12:15:32 +08:00
|
|
|
|
/// <returns></returns>
|
2024-09-22 14:10:13 +08:00
|
|
|
|
public async Task RunAsync(IFlowEnvironment env,
|
|
|
|
|
|
List<NodeModelBase> nodes,
|
2024-09-16 21:38:34 +08:00
|
|
|
|
List<MethodDetails> initMethods,
|
|
|
|
|
|
List<MethodDetails> loadingMethods,
|
2024-09-22 14:10:13 +08:00
|
|
|
|
List<MethodDetails> exitMethods)
|
2024-08-06 16:09:46 +08:00
|
|
|
|
{
|
2024-09-16 21:38:34 +08:00
|
|
|
|
|
|
|
|
|
|
FlowState = RunState.Running; // 开始运行
|
2024-09-22 14:10:13 +08:00
|
|
|
|
NodeModelBase? startNode = nodes.FirstOrDefault(node => node.IsStart);
|
|
|
|
|
|
if (startNode is null) {
|
2024-09-16 21:38:34 +08:00
|
|
|
|
FlowState = RunState.Completion; // 不存在起点,退出流程
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
2024-09-15 12:15:32 +08:00
|
|
|
|
|
2024-09-22 14:10:13 +08:00
|
|
|
|
#region 获取所有触发器,以及已加载节点的方法信息
|
|
|
|
|
|
List<MethodDetails> runNodeMd;
|
|
|
|
|
|
List<SingleFlipflopNode> flipflopNodes;
|
|
|
|
|
|
|
|
|
|
|
|
flipflopNodes = nodes.Where(it => it.MethodDetails?.MethodDynamicType == NodeType.Flipflop && it.IsStart == false)
|
|
|
|
|
|
.Select(it => (SingleFlipflopNode)it)
|
|
|
|
|
|
.Where(node => node is SingleFlipflopNode flipflopNode && flipflopNode.NotExitPreviousNode())
|
|
|
|
|
|
.ToList();// 获取需要再运行开始之前启动的触发器节点
|
|
|
|
|
|
runNodeMd = nodes.Select(item => item.MethodDetails).ToList(); // 获取环境中所有节点的方法信息
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
|
|
|
|
|
2024-09-17 14:20:27 +08:00
|
|
|
|
#region 选择运行环境的上下文
|
|
|
|
|
|
|
2024-09-16 21:38:34 +08:00
|
|
|
|
// 判断使用哪一种流程上下文
|
2024-09-21 10:06:44 +08:00
|
|
|
|
var isNetFramework = false;
|
2024-09-15 12:15:32 +08:00
|
|
|
|
if (isNetFramework)
|
2024-09-12 20:32:54 +08:00
|
|
|
|
{
|
2024-09-16 21:38:34 +08:00
|
|
|
|
Context = new Serein.Library.Framework.NodeFlow.DynamicContext(SereinIOC, env);
|
2024-09-12 20:32:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
2024-09-21 10:06:44 +08:00
|
|
|
|
Context = new Serein.Library.Core.NodeFlow.DynamicContext(SereinIOC, env); // 从起始节点启动流程时创建上下文
|
2024-09-16 21:38:34 +08:00
|
|
|
|
}
|
2024-09-17 14:20:27 +08:00
|
|
|
|
#endregion
|
2024-09-16 21:38:34 +08:00
|
|
|
|
|
|
|
|
|
|
#region 初始化运行环境的Ioc容器
|
2024-09-20 10:50:32 +08:00
|
|
|
|
// 清除节点使用的对象,筛选出需要初始化的方法描述
|
2024-09-17 14:20:27 +08:00
|
|
|
|
var thisRuningMds = new List<MethodDetails>();
|
2024-09-18 16:45:41 +08:00
|
|
|
|
thisRuningMds.AddRange(runNodeMd.Where(md => md is not null));
|
|
|
|
|
|
thisRuningMds.AddRange(initMethods.Where(md => md is not null));
|
|
|
|
|
|
thisRuningMds.AddRange(loadingMethods.Where(md => md is not null));
|
|
|
|
|
|
thisRuningMds.AddRange(exitMethods.Where(md => md is not null));
|
2024-09-17 14:20:27 +08:00
|
|
|
|
|
|
|
|
|
|
// .AddRange(initMethods).AddRange(loadingMethods).a
|
|
|
|
|
|
foreach (var nodeMd in thisRuningMds)
|
2024-09-16 21:38:34 +08:00
|
|
|
|
{
|
|
|
|
|
|
nodeMd.ActingInstance = null;
|
|
|
|
|
|
}
|
|
|
|
|
|
SereinIOC.Reset(); // 开始运行时清空ioc中注册的实例
|
|
|
|
|
|
// 初始化ioc容器中的类型对象
|
2024-09-17 14:20:27 +08:00
|
|
|
|
foreach (var md in thisRuningMds)
|
2024-09-16 21:38:34 +08:00
|
|
|
|
{
|
2024-09-21 10:06:44 +08:00
|
|
|
|
if (md.ActingInstanceType != null)
|
2024-09-17 14:20:27 +08:00
|
|
|
|
{
|
|
|
|
|
|
SereinIOC.Register(md.ActingInstanceType);
|
|
|
|
|
|
}
|
2024-09-21 10:06:44 +08:00
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
await Console.Out.WriteLineAsync($"{md.MethodName} - 没有类型声明");
|
|
|
|
|
|
IsStopStart = true;
|
|
|
|
|
|
}
|
2024-09-16 21:38:34 +08:00
|
|
|
|
}
|
2024-09-22 14:10:13 +08:00
|
|
|
|
CheckStartState(); // 初始化IOC后检查状态
|
2024-09-21 10:06:44 +08:00
|
|
|
|
|
|
|
|
|
|
|
2024-09-17 14:20:27 +08:00
|
|
|
|
SereinIOC.Build(); // 流程启动前的初始化
|
2024-09-21 10:06:44 +08:00
|
|
|
|
|
2024-09-17 14:20:27 +08:00
|
|
|
|
foreach (var md in thisRuningMds)
|
2024-09-16 21:38:34 +08:00
|
|
|
|
{
|
2024-09-21 10:06:44 +08:00
|
|
|
|
md.ActingInstance = SereinIOC.GetOrRegisterInstantiate(md.ActingInstanceType);
|
|
|
|
|
|
if(md.ActingInstance is null)
|
2024-09-17 14:20:27 +08:00
|
|
|
|
{
|
2024-09-21 10:06:44 +08:00
|
|
|
|
await Console.Out.WriteLineAsync($"{md.MethodName} - 无法获取类型[{md.ActingInstanceType}]的实例");
|
|
|
|
|
|
IsStopStart = true;
|
2024-09-17 14:20:27 +08:00
|
|
|
|
}
|
2024-09-12 20:32:54 +08:00
|
|
|
|
}
|
2024-08-06 16:09:46 +08:00
|
|
|
|
|
2024-09-22 14:10:13 +08:00
|
|
|
|
CheckStartState();// 调用节点初始化后检查状态
|
2024-09-21 10:06:44 +08:00
|
|
|
|
|
2024-09-16 21:38:34 +08:00
|
|
|
|
#endregion
|
|
|
|
|
|
|
2024-09-21 10:06:44 +08:00
|
|
|
|
#region 检查并修正初始化、加载时、退出时方法作用的对象,保证后续不会报错(已注释)
|
|
|
|
|
|
//foreach (var md in initMethods) // 初始化
|
|
|
|
|
|
//{
|
|
|
|
|
|
// md.ActingInstance ??= Context.SereinIoc.GetOrRegisterInstantiate(md.ActingInstanceType);
|
|
|
|
|
|
//}
|
|
|
|
|
|
//foreach (var md in loadingMethods) // 加载
|
|
|
|
|
|
//{
|
|
|
|
|
|
// md.ActingInstance ??= Context.SereinIoc.GetOrRegisterInstantiate(md.ActingInstanceType);
|
|
|
|
|
|
//}
|
|
|
|
|
|
//foreach (var md in exitMethods) // 初始化
|
|
|
|
|
|
//{
|
|
|
|
|
|
// md.ActingInstance ??= Context.SereinIoc.GetOrRegisterInstantiate(md.ActingInstanceType);
|
|
|
|
|
|
//}
|
2024-09-17 14:20:27 +08:00
|
|
|
|
#endregion
|
|
|
|
|
|
|
2024-09-17 21:43:49 +08:00
|
|
|
|
#region 执行初始化,绑定IOC容器,再执行加载时
|
2024-08-06 16:09:46 +08:00
|
|
|
|
|
2024-09-21 10:06:44 +08:00
|
|
|
|
//object?[]? args = [Context];
|
2024-09-17 14:20:27 +08:00
|
|
|
|
foreach (var md in initMethods) // 初始化
|
|
|
|
|
|
{
|
2024-09-21 10:06:44 +08:00
|
|
|
|
((Action<object, object?[]?>)md.MethodDelegate).Invoke(md.ActingInstance, [Context]);
|
2024-09-17 14:20:27 +08:00
|
|
|
|
}
|
|
|
|
|
|
Context.SereinIoc.Build(); // 绑定初始化时注册的类型
|
|
|
|
|
|
foreach (var md in loadingMethods) // 加载
|
|
|
|
|
|
{
|
2024-09-21 10:06:44 +08:00
|
|
|
|
//object?[]? data = [md.ActingInstance, args];
|
|
|
|
|
|
//md.MethodDelegate.DynamicInvoke(data);
|
|
|
|
|
|
((Action<object, object?[]?>)md.MethodDelegate).Invoke(md.ActingInstance, [Context]);
|
2024-09-17 14:20:27 +08:00
|
|
|
|
}
|
|
|
|
|
|
Context.SereinIoc.Build(); // 预防有人在加载时才注册类型,再绑定一次
|
2024-09-17 21:43:49 +08:00
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
|
|
|
|
#region 设置流程退出时的回调函数
|
2024-08-06 16:09:46 +08:00
|
|
|
|
ExitAction = () =>
|
|
|
|
|
|
{
|
2024-09-17 15:58:37 +08:00
|
|
|
|
SereinIOC.Run<WebServer>(web => {
|
|
|
|
|
|
web?.Stop();
|
|
|
|
|
|
});
|
|
|
|
|
|
|
2024-08-06 16:09:46 +08:00
|
|
|
|
foreach (MethodDetails? md in exitMethods)
|
|
|
|
|
|
{
|
2024-09-21 10:06:44 +08:00
|
|
|
|
((Action<object, object?[]?>)md.MethodDelegate).Invoke(md.ActingInstance, [Context]);
|
2024-09-16 21:38:34 +08:00
|
|
|
|
}
|
2024-09-21 10:06:44 +08:00
|
|
|
|
|
2024-09-22 14:10:13 +08:00
|
|
|
|
if (_flipFlopCts != null && !_flipFlopCts.IsCancellationRequested)
|
|
|
|
|
|
{
|
|
|
|
|
|
_flipFlopCts?.Cancel();
|
|
|
|
|
|
_flipFlopCts?.Dispose();
|
|
|
|
|
|
}
|
2024-09-16 21:38:34 +08:00
|
|
|
|
FlowState = RunState.Completion;
|
|
|
|
|
|
FlipFlopState = RunState.Completion;
|
2024-08-06 16:09:46 +08:00
|
|
|
|
};
|
2024-09-16 21:38:34 +08:00
|
|
|
|
#endregion
|
|
|
|
|
|
|
2024-09-17 14:20:27 +08:00
|
|
|
|
#region 开始启动流程
|
2024-09-22 14:10:13 +08:00
|
|
|
|
|
2024-08-06 16:09:46 +08:00
|
|
|
|
try
|
|
|
|
|
|
{
|
2024-09-21 10:06:44 +08:00
|
|
|
|
|
2024-09-16 21:38:34 +08:00
|
|
|
|
if (flipflopNodes.Count > 0)
|
2024-09-09 16:42:01 +08:00
|
|
|
|
{
|
2024-09-16 21:38:34 +08:00
|
|
|
|
FlipFlopState = RunState.Running;
|
|
|
|
|
|
// 如果存在需要启动的触发器,则开始启动
|
2024-09-22 14:10:13 +08:00
|
|
|
|
_flipFlopCts = new CancellationTokenSource();
|
|
|
|
|
|
SereinIOC.CustomRegisterInstance(FlipFlopCtsName, _flipFlopCts,false);
|
2024-09-21 10:06:44 +08:00
|
|
|
|
|
2024-09-16 21:38:34 +08:00
|
|
|
|
// 使用 TaskCompletionSource 创建未启动的触发器任务
|
|
|
|
|
|
var tasks = flipflopNodes.Select(async node =>
|
|
|
|
|
|
{
|
2024-09-20 10:50:32 +08:00
|
|
|
|
await FlipflopExecute(env,node);
|
2024-09-16 21:38:34 +08:00
|
|
|
|
}).ToArray();
|
|
|
|
|
|
_ = Task.WhenAll(tasks);
|
|
|
|
|
|
}
|
2024-09-20 10:50:32 +08:00
|
|
|
|
await startNode.StartExecute(Context); // 开始运行时从起始节点开始运行
|
2024-09-15 19:48:27 +08:00
|
|
|
|
// 等待结束
|
2024-09-22 14:10:13 +08:00
|
|
|
|
if(FlipFlopState == RunState.Running && _flipFlopCts is not null)
|
2024-09-15 19:48:27 +08:00
|
|
|
|
{
|
2024-09-22 14:10:13 +08:00
|
|
|
|
while (!_flipFlopCts.IsCancellationRequested)
|
2024-09-16 21:38:34 +08:00
|
|
|
|
{
|
|
|
|
|
|
await Task.Delay(100);
|
|
|
|
|
|
}
|
2024-09-15 19:48:27 +08:00
|
|
|
|
}
|
2024-08-06 16:09:46 +08:00
|
|
|
|
}
|
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
|
{
|
|
|
|
|
|
await Console.Out.WriteLineAsync(ex.ToString());
|
2024-09-22 17:37:32 +08:00
|
|
|
|
// await Console.Out.WriteLineAsync(ex.Message);
|
2024-09-21 10:06:44 +08:00
|
|
|
|
}
|
|
|
|
|
|
finally
|
|
|
|
|
|
{
|
|
|
|
|
|
FlowState = RunState.Completion;
|
|
|
|
|
|
}
|
2024-09-17 14:20:27 +08:00
|
|
|
|
#endregion
|
2024-08-06 16:09:46 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2024-09-17 21:43:49 +08:00
|
|
|
|
public void AddFlipflopInRuning(SingleFlipflopNode singleFlipFlopNode, IFlowEnvironment flowEnvironment)
|
|
|
|
|
|
{
|
|
|
|
|
|
_ = Task.Run(async () =>
|
|
|
|
|
|
{
|
|
|
|
|
|
// 设置对象
|
|
|
|
|
|
singleFlipFlopNode.MethodDetails.ActingInstance = SereinIOC.GetOrRegisterInstantiate(singleFlipFlopNode.MethodDetails.ActingInstanceType);
|
2024-09-20 10:50:32 +08:00
|
|
|
|
await FlipflopExecute(flowEnvironment,singleFlipFlopNode); // 启动触发器
|
2024-09-17 21:43:49 +08:00
|
|
|
|
});
|
|
|
|
|
|
}
|
2024-09-18 16:45:41 +08:00
|
|
|
|
|
2024-09-15 12:15:32 +08:00
|
|
|
|
/// <summary>
|
2024-09-18 16:45:41 +08:00
|
|
|
|
/// 启动全局触发器
|
2024-09-15 12:15:32 +08:00
|
|
|
|
/// </summary>
|
2024-09-20 10:50:32 +08:00
|
|
|
|
/// <param name="flowEnvironment">流程运行全局环境</param>
|
|
|
|
|
|
/// <param name="singleFlipFlopNode">需要全局监听信号的触发器</param>
|
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
|
private async Task FlipflopExecute(IFlowEnvironment flowEnvironment,SingleFlipflopNode singleFlipFlopNode)
|
2024-08-06 16:09:46 +08:00
|
|
|
|
{
|
2024-09-21 10:06:44 +08:00
|
|
|
|
|
|
|
|
|
|
var context = new DynamicContext(SereinIOC, flowEnvironment); // 启动全局触发器时新建上下文
|
2024-08-06 16:09:46 +08:00
|
|
|
|
try
|
|
|
|
|
|
{
|
2024-09-22 14:10:13 +08:00
|
|
|
|
|
|
|
|
|
|
while (!_flipFlopCts.IsCancellationRequested)
|
2024-08-06 16:09:46 +08:00
|
|
|
|
{
|
2024-09-22 17:37:32 +08:00
|
|
|
|
var newFlowData = await singleFlipFlopNode.ExecutingAsync(context); // 获取触发器等待Task
|
|
|
|
|
|
await NodeModelBase.RefreshFlowDataAndExpInterrupt(context, singleFlipFlopNode, newFlowData); // 全局触发器触发后刷新该触发器的节点数据
|
2024-09-21 10:06:44 +08:00
|
|
|
|
if (singleFlipFlopNode.NextOrientation != ConnectionType.None)
|
2024-09-20 10:50:32 +08:00
|
|
|
|
{
|
2024-09-21 10:06:44 +08:00
|
|
|
|
var nextNodes = singleFlipFlopNode.SuccessorNodes[singleFlipFlopNode.NextOrientation];
|
2024-09-22 14:10:13 +08:00
|
|
|
|
for (int i = nextNodes.Count - 1; i >= 0 && !_flipFlopCts.IsCancellationRequested; i--)
|
2024-09-21 10:06:44 +08:00
|
|
|
|
{
|
2024-09-22 17:37:32 +08:00
|
|
|
|
// 筛选出启用的节点
|
|
|
|
|
|
if (nextNodes[i].DebugSetting.IsEnable)
|
2024-09-21 10:06:44 +08:00
|
|
|
|
{
|
|
|
|
|
|
nextNodes[i].PreviousNode = singleFlipFlopNode;
|
2024-09-22 17:37:32 +08:00
|
|
|
|
if (nextNodes[i].DebugSetting.InterruptClass != InterruptClass.None) // 执行触发前
|
|
|
|
|
|
{
|
|
|
|
|
|
var cancelType = await nextNodes[i].DebugSetting.GetInterruptTask();
|
|
|
|
|
|
await Console.Out.WriteLineAsync($"[{nextNodes[i].MethodDetails.MethodName}]中断已{cancelType},开始执行后继分支");
|
|
|
|
|
|
}
|
2024-09-22 14:10:13 +08:00
|
|
|
|
await nextNodes[i].StartExecute(context); // 启动执行触发器后继分支的节点
|
2024-09-21 10:06:44 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-09-20 10:50:32 +08:00
|
|
|
|
}
|
2024-09-22 17:37:32 +08:00
|
|
|
|
|
2024-09-18 16:45:41 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
|
{
|
|
|
|
|
|
await Console.Out.WriteLineAsync(ex.ToString());
|
|
|
|
|
|
}
|
2024-09-22 14:10:13 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//MethodDetails md = singleFlipFlopNode.MethodDetails;
|
|
|
|
|
|
//var del = md.MethodDelegate;
|
|
|
|
|
|
//object?[]? parameters = singleFlipFlopNode.GetParameters(context, singleFlipFlopNode.MethodDetails); // 启动全局触发器时获取入参参数
|
|
|
|
|
|
//// 设置委托对象
|
|
|
|
|
|
//var func = md.ExplicitDatas.Length == 0 ?
|
|
|
|
|
|
// (Func<object, object, Task<IFlipflopContext>>)del :
|
|
|
|
|
|
// (Func<object, object[], Task<IFlipflopContext>>)del;
|
|
|
|
|
|
|
|
|
|
|
|
//if(t)
|
|
|
|
|
|
//{
|
|
|
|
|
|
// IFlipflopContext flipflopContext = await ((Func<object, Task<IFlipflopContext>>)del.Clone()).Invoke(md.ActingInstance);// 开始等待全局触发器的触发
|
|
|
|
|
|
// var connectionType = flipflopContext.State.ToContentType();
|
|
|
|
|
|
// if (connectionType != ConnectionType.None)
|
|
|
|
|
|
// {
|
|
|
|
|
|
// await GlobalFlipflopExecute(context, singleFlipFlopNode, connectionType, cts);
|
|
|
|
|
|
// }
|
|
|
|
|
|
//}
|
|
|
|
|
|
//else
|
|
|
|
|
|
//{
|
|
|
|
|
|
// IFlipflopContext flipflopContext = await ((Func<object, object[], Task<IFlipflopContext>>)del.Clone()).Invoke(md.ActingInstance, parameters);// 开始等待全局触发器的触发
|
|
|
|
|
|
// var connectionType = flipflopContext.State.ToContentType();
|
|
|
|
|
|
// if (connectionType != ConnectionType.None)
|
|
|
|
|
|
// {
|
|
|
|
|
|
// await GlobalFlipflopExecute(context, singleFlipFlopNode, connectionType, cts);
|
|
|
|
|
|
// }
|
|
|
|
|
|
//}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public void Exit()
|
|
|
|
|
|
{
|
|
|
|
|
|
ExitAction?.Invoke();
|
2024-09-18 16:45:41 +08:00
|
|
|
|
}
|
2024-09-15 19:48:27 +08:00
|
|
|
|
|
2024-09-22 14:10:13 +08:00
|
|
|
|
|
|
|
|
|
|
#if false
|
|
|
|
|
|
|
2024-09-20 10:50:32 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 全局触发器开始执行相关分支
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="context">上下文</param>
|
|
|
|
|
|
/// <param name="singleFlipFlopNode">被触发的全局触发器</param>
|
|
|
|
|
|
/// <param name="connectionType">分支类型</param>
|
|
|
|
|
|
/// <returns></returns>
|
2024-09-21 10:06:44 +08:00
|
|
|
|
public async Task GlobalFlipflopExecute(IDynamicContext context, SingleFlipflopNode singleFlipFlopNode,
|
2024-09-22 14:10:13 +08:00
|
|
|
|
|
|
|
|
|
|
ConnectionType connectionType, CancellationTokenSource cts)
|
2024-09-18 16:45:41 +08:00
|
|
|
|
{
|
2024-09-22 14:10:13 +08:00
|
|
|
|
|
2024-09-20 10:50:32 +08:00
|
|
|
|
|
2024-09-18 16:45:41 +08:00
|
|
|
|
bool skip = true;
|
|
|
|
|
|
Stack<NodeModelBase> stack = new Stack<NodeModelBase>();
|
|
|
|
|
|
stack.Push(singleFlipFlopNode);
|
2024-08-06 16:09:46 +08:00
|
|
|
|
|
2024-09-15 22:07:10 +08:00
|
|
|
|
|
2024-09-18 16:45:41 +08:00
|
|
|
|
while (stack.Count > 0 && !cts.IsCancellationRequested) // 循环中直到栈为空才会退出循环
|
|
|
|
|
|
{
|
|
|
|
|
|
// 从栈中弹出一个节点作为当前节点进行处理
|
|
|
|
|
|
var currentNode = stack.Pop();
|
2024-09-17 21:43:49 +08:00
|
|
|
|
|
2024-09-18 16:45:41 +08:00
|
|
|
|
// 设置方法执行的对象
|
2024-09-21 10:06:44 +08:00
|
|
|
|
//if (currentNode.MethodDetails?.ActingInstance == null && currentNode.MethodDetails?.ActingInstanceType is not null)
|
|
|
|
|
|
//{
|
|
|
|
|
|
// currentNode.MethodDetails.ActingInstance ??= context.SereinIoc.GetOrRegisterInstantiate(currentNode.MethodDetails.ActingInstanceType);
|
|
|
|
|
|
//}
|
2024-09-18 16:45:41 +08:00
|
|
|
|
|
|
|
|
|
|
// 首先执行上游分支
|
|
|
|
|
|
var upstreamNodes = currentNode.SuccessorNodes[ConnectionType.Upstream];
|
|
|
|
|
|
for (int i = upstreamNodes.Count - 1; i >= 0; i--)
|
|
|
|
|
|
{
|
|
|
|
|
|
upstreamNodes[i].PreviousNode = currentNode;
|
2024-09-20 10:50:32 +08:00
|
|
|
|
await upstreamNodes[i].StartExecute(context); // 执行全局触发器的上游分支
|
2024-09-18 16:45:41 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 当前节点是已经触发了的全局触发器,所以跳过,难道每次都要判断一次?
|
|
|
|
|
|
if (skip)
|
|
|
|
|
|
{
|
|
|
|
|
|
skip = false;
|
|
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
2024-09-21 10:06:44 +08:00
|
|
|
|
currentNode.FlowData = await currentNode.ExecutingAsync(context);
|
2024-09-20 10:50:32 +08:00
|
|
|
|
|
2024-09-22 14:10:13 +08:00
|
|
|
|
|
|
|
|
|
|
if (currentNode.NextOrientation == ConnectionType.None)
|
2024-09-18 16:45:41 +08:00
|
|
|
|
{
|
|
|
|
|
|
break; // 不再执行
|
|
|
|
|
|
}
|
|
|
|
|
|
connectionType = currentNode.NextOrientation;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 获取下一分支
|
|
|
|
|
|
var nextNodes = currentNode.SuccessorNodes[connectionType];
|
|
|
|
|
|
|
|
|
|
|
|
// 将下一个节点集合中的所有节点逆序推入栈中
|
|
|
|
|
|
for (int i = nextNodes.Count - 1; i >= 0; i--)
|
|
|
|
|
|
{
|
|
|
|
|
|
nextNodes[i].PreviousNode = currentNode;
|
|
|
|
|
|
stack.Push(nextNodes[i]);
|
2024-08-06 16:09:46 +08:00
|
|
|
|
}
|
2024-09-22 14:10:13 +08:00
|
|
|
|
}}
|
|
|
|
|
|
#endif
|
2024-08-06 16:09:46 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-09-22 14:10:13 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|