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;
|
|
|
|
|
|
using Serein.NodeFlow.Base;
|
2024-08-06 16:09:46 +08:00
|
|
|
|
using Serein.NodeFlow.Model;
|
|
|
|
|
|
|
|
|
|
|
|
namespace Serein.NodeFlow
|
|
|
|
|
|
{
|
|
|
|
|
|
|
2024-09-15 12:15:32 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 流程启动器
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="serviceContainer"></param>
|
|
|
|
|
|
/// <param name="methodDetails"></param>
|
|
|
|
|
|
public class FlowStarter(ISereinIoc serviceContainer, List<MethodDetails> methodDetails)
|
2024-08-06 16:09:46 +08:00
|
|
|
|
|
|
|
|
|
|
{
|
2024-09-12 20:32:54 +08:00
|
|
|
|
private readonly ISereinIoc ServiceContainer = serviceContainer;
|
2024-08-06 16:09:46 +08:00
|
|
|
|
private readonly List<MethodDetails> methodDetails = methodDetails;
|
2024-09-15 12:15:32 +08:00
|
|
|
|
private Action ExitAction = null; //退出方法
|
|
|
|
|
|
private IDynamicContext context = null; //上下文
|
|
|
|
|
|
public NodeRunCts MainCts;
|
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-15 12:15:32 +08:00
|
|
|
|
/// <param name="nodes"></param>
|
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
|
// public async Task RunAsync(List<NodeModelBase> nodes, IFlowEnvironment flowEnvironment)
|
|
|
|
|
|
public async Task RunAsync(NodeModelBase startNode, IFlowEnvironment flowEnvironment, List<SingleFlipflopNode> flipflopNodes)
|
2024-08-06 16:09:46 +08:00
|
|
|
|
{
|
2024-09-15 12:15:32 +08:00
|
|
|
|
// var startNode = nodes.FirstOrDefault(p => p.IsStart);
|
2024-08-06 16:09:46 +08:00
|
|
|
|
if (startNode == null) { return; }
|
2024-09-15 12:15:32 +08:00
|
|
|
|
|
|
|
|
|
|
var isNetFramework = true;
|
|
|
|
|
|
|
|
|
|
|
|
if (isNetFramework)
|
2024-09-12 20:32:54 +08:00
|
|
|
|
{
|
2024-09-15 12:15:32 +08:00
|
|
|
|
context = new Serein.Library.Framework.NodeFlow.DynamicContext(ServiceContainer, flowEnvironment);
|
2024-09-12 20:32:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
2024-09-15 12:15:32 +08:00
|
|
|
|
context = new Serein.Library.Core.NodeFlow.DynamicContext(ServiceContainer, flowEnvironment);
|
2024-09-12 20:32:54 +08:00
|
|
|
|
}
|
2024-08-06 16:09:46 +08:00
|
|
|
|
|
2024-09-15 12:15:32 +08:00
|
|
|
|
MainCts = ServiceContainer.CreateServiceInstance<NodeRunCts>();
|
2024-08-06 16:09:46 +08:00
|
|
|
|
|
2024-09-12 20:32:54 +08:00
|
|
|
|
var initMethods = methodDetails.Where(it => it.MethodDynamicType == NodeType.Init).ToList();
|
|
|
|
|
|
var loadingMethods = methodDetails.Where(it => it.MethodDynamicType == NodeType.Loading).ToList();
|
|
|
|
|
|
var exitMethods = methodDetails.Where(it => it.MethodDynamicType == NodeType.Exit).ToList();
|
2024-08-06 16:09:46 +08:00
|
|
|
|
ExitAction = () =>
|
|
|
|
|
|
{
|
2024-09-12 20:32:54 +08:00
|
|
|
|
//ServiceContainer.Run<WebServer>((web) =>
|
|
|
|
|
|
//{
|
|
|
|
|
|
// web?.Stop();
|
|
|
|
|
|
//});
|
2024-08-06 16:09:46 +08:00
|
|
|
|
foreach (MethodDetails? md in exitMethods)
|
|
|
|
|
|
{
|
|
|
|
|
|
object?[]? args = [context];
|
|
|
|
|
|
object?[]? data = [md.ActingInstance, args];
|
|
|
|
|
|
md.MethodDelegate.DynamicInvoke(data);
|
|
|
|
|
|
}
|
|
|
|
|
|
if (context != null && context.NodeRunCts != null && !context.NodeRunCts.IsCancellationRequested)
|
|
|
|
|
|
{
|
|
|
|
|
|
context.NodeRunCts.Cancel();
|
|
|
|
|
|
}
|
|
|
|
|
|
if (MainCts != null && !MainCts.IsCancellationRequested) MainCts.Cancel();
|
|
|
|
|
|
ServiceContainer.Reset();
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
foreach (var md in initMethods) // 初始化 - 调用方法
|
|
|
|
|
|
{
|
|
|
|
|
|
object?[]? args = [context];
|
|
|
|
|
|
object?[]? data = [md.ActingInstance, args];
|
|
|
|
|
|
md.MethodDelegate.DynamicInvoke(data);
|
|
|
|
|
|
}
|
2024-09-12 20:32:54 +08:00
|
|
|
|
context.SereinIoc.Build();
|
2024-08-06 16:09:46 +08:00
|
|
|
|
|
|
|
|
|
|
foreach (var md in loadingMethods) // 加载
|
|
|
|
|
|
{
|
|
|
|
|
|
object?[]? args = [context];
|
|
|
|
|
|
object?[]? data = [md.ActingInstance, args];
|
|
|
|
|
|
md.MethodDelegate.DynamicInvoke(data);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-09-15 12:15:32 +08:00
|
|
|
|
// 运行触发器节点
|
2024-08-06 16:09:46 +08:00
|
|
|
|
var singleFlipflopNodes = flipflopNodes.Select(it => (SingleFlipflopNode)it).ToArray();
|
|
|
|
|
|
|
|
|
|
|
|
// 使用 TaskCompletionSource 创建未启动的任务
|
|
|
|
|
|
var tasks = singleFlipflopNodes.Select(async node =>
|
|
|
|
|
|
{
|
2024-09-15 12:15:32 +08:00
|
|
|
|
await FlipflopExecute(node, flowEnvironment);
|
2024-08-06 16:09:46 +08:00
|
|
|
|
}).ToArray();
|
|
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
|
{
|
2024-09-09 16:42:01 +08:00
|
|
|
|
await Task.Run(async () =>
|
|
|
|
|
|
{
|
|
|
|
|
|
await Task.WhenAll([startNode.StartExecution(context), .. tasks]);
|
|
|
|
|
|
});
|
2024-08-06 16:09:46 +08:00
|
|
|
|
}
|
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
|
{
|
|
|
|
|
|
await Console.Out.WriteLineAsync(ex.ToString());
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-09-15 12:15:32 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 启动触发器
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
private async Task FlipflopExecute(SingleFlipflopNode singleFlipFlopNode, IFlowEnvironment flowEnvironment)
|
2024-08-06 16:09:46 +08:00
|
|
|
|
{
|
2024-09-15 12:15:32 +08:00
|
|
|
|
DynamicContext context = new DynamicContext(ServiceContainer, flowEnvironment);
|
2024-08-06 16:09:46 +08:00
|
|
|
|
MethodDetails md = singleFlipFlopNode.MethodDetails;
|
2024-09-15 12:15:32 +08:00
|
|
|
|
var del = md.MethodDelegate;
|
2024-08-06 16:09:46 +08:00
|
|
|
|
try
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
|
|
|
2024-09-12 20:32:54 +08:00
|
|
|
|
//var func = md.ExplicitDatas.Length == 0 ? (Func<object, object, Task<FlipflopContext<dynamic>>>)del : (Func<object, object[], Task<FlipflopContext<dynamic>>>)del;
|
|
|
|
|
|
var func = md.ExplicitDatas.Length == 0 ? (Func<object, object, Task<IFlipflopContext>>)del : (Func<object, object[], Task<IFlipflopContext>>)del;
|
2024-08-06 16:09:46 +08:00
|
|
|
|
|
|
|
|
|
|
while (!MainCts.IsCancellationRequested) // 循环中直到栈为空才会退出
|
|
|
|
|
|
{
|
|
|
|
|
|
object?[]? parameters = singleFlipFlopNode.GetParameters(context, md);
|
|
|
|
|
|
// 调用委托并获取结果
|
|
|
|
|
|
|
2024-09-12 20:32:54 +08:00
|
|
|
|
IFlipflopContext flipflopContext = await func.Invoke(md.ActingInstance, parameters);
|
2024-08-06 16:09:46 +08:00
|
|
|
|
|
2024-09-15 12:15:32 +08:00
|
|
|
|
if (flipflopContext.State == FlowStateType.Succeed)
|
2024-08-06 16:09:46 +08:00
|
|
|
|
{
|
2024-09-09 16:42:01 +08:00
|
|
|
|
singleFlipFlopNode.FlowState = FlowStateType.Succeed;
|
2024-08-06 16:09:46 +08:00
|
|
|
|
singleFlipFlopNode.FlowData = flipflopContext.Data;
|
2024-09-15 12:15:32 +08:00
|
|
|
|
var tasks = singleFlipFlopNode.PreviousNodes[ConnectionType.IsSucceed].Select(nextNode =>
|
2024-08-06 16:09:46 +08:00
|
|
|
|
{
|
2024-09-15 12:15:32 +08:00
|
|
|
|
var context = new DynamicContext(ServiceContainer,flowEnvironment);
|
2024-08-06 16:09:46 +08:00
|
|
|
|
nextNode.PreviousNode = singleFlipFlopNode;
|
2024-09-09 16:42:01 +08:00
|
|
|
|
return nextNode.StartExecution(context);
|
2024-08-06 16:09:46 +08:00
|
|
|
|
}).ToArray();
|
|
|
|
|
|
Task.WaitAll(tasks);
|
|
|
|
|
|
}
|
2024-09-15 12:15:32 +08:00
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
break;
|
|
|
|
|
|
}
|
2024-08-06 16:09:46 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
|
{
|
|
|
|
|
|
await Console.Out.WriteLineAsync(ex.ToString());
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public void Exit()
|
|
|
|
|
|
{
|
|
|
|
|
|
ExitAction?.Invoke();
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|