2024-12-20 23:39:29 +08:00
|
|
|
|
using Serein.Library;
|
|
|
|
|
|
using Serein.Library.Api;
|
2025-07-30 11:29:12 +08:00
|
|
|
|
using Serein.Library.Utils;
|
2024-12-20 23:39:29 +08:00
|
|
|
|
using Serein.Script;
|
2025-07-13 17:34:03 +08:00
|
|
|
|
using Serein.Script.Node.FlowControl;
|
2024-12-20 23:39:29 +08:00
|
|
|
|
using System;
|
|
|
|
|
|
using System.Collections.Generic;
|
2025-07-23 15:57:57 +08:00
|
|
|
|
using System.Diagnostics;
|
2024-12-21 20:47:31 +08:00
|
|
|
|
using System.Dynamic;
|
2024-12-20 23:39:29 +08:00
|
|
|
|
using System.Linq;
|
2024-12-21 20:47:31 +08:00
|
|
|
|
using System.Linq.Expressions;
|
2025-07-30 11:29:12 +08:00
|
|
|
|
using System.Reactive;
|
2024-12-20 23:39:29 +08:00
|
|
|
|
using System.Reflection;
|
|
|
|
|
|
using System.Text;
|
|
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
|
using System.Xml.Linq;
|
|
|
|
|
|
|
2025-07-29 14:25:31 +08:00
|
|
|
|
namespace Serein.NodeFlow.Model.Nodes
|
2024-12-20 23:39:29 +08:00
|
|
|
|
{
|
2024-12-23 23:19:10 +08:00
|
|
|
|
|
2025-07-30 21:15:07 +08:00
|
|
|
|
[FlowDataProperty(ValuePath = NodeValuePath.Node, IsNodeImp = true)]
|
2024-12-20 23:39:29 +08:00
|
|
|
|
public partial class SingleScriptNode : NodeModelBase
|
|
|
|
|
|
{
|
2025-07-30 21:15:07 +08:00
|
|
|
|
[DataInfo(IsNotification = true)]
|
|
|
|
|
|
private string _script = string.Empty;
|
2024-12-20 23:39:29 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 流程脚本节点
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public partial class SingleScriptNode : NodeModelBase
|
|
|
|
|
|
{
|
2024-12-24 22:23:53 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 脚本节点是基础节点
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public override bool IsBase => true;
|
|
|
|
|
|
|
2025-05-30 15:42:59 +08:00
|
|
|
|
private bool IsScriptChanged = false;
|
|
|
|
|
|
|
2025-07-16 16:16:19 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 脚本解释器
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
private readonly SereinScript sereinScript;
|
|
|
|
|
|
|
2024-12-20 23:39:29 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 构建流程脚本节点
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="environment"></param>
|
2025-07-16 16:16:19 +08:00
|
|
|
|
public SingleScriptNode(IFlowEnvironment environment) : base(environment)
|
2024-12-20 23:39:29 +08:00
|
|
|
|
{
|
2025-07-16 16:16:19 +08:00
|
|
|
|
sereinScript = new SereinScript();
|
2024-12-21 20:47:31 +08:00
|
|
|
|
}
|
2024-12-20 23:39:29 +08:00
|
|
|
|
|
2024-12-21 20:47:31 +08:00
|
|
|
|
static SingleScriptNode()
|
|
|
|
|
|
{
|
2024-12-20 23:39:29 +08:00
|
|
|
|
// 挂载静态方法
|
2025-07-23 15:57:57 +08:00
|
|
|
|
var tempMethods = typeof(Serein.Library.ScriptBaseFunc).GetMethods().Where(method =>
|
|
|
|
|
|
method.IsStatic &&
|
2024-12-20 23:39:29 +08:00
|
|
|
|
!(method.Name.Equals("GetHashCode")
|
|
|
|
|
|
|| method.Name.Equals("Equals")
|
|
|
|
|
|
|| method.Name.Equals("ToString")
|
|
|
|
|
|
|| method.Name.Equals("GetType")
|
|
|
|
|
|
)).Select(method => (method.Name, method)).ToArray();
|
2024-12-21 20:47:31 +08:00
|
|
|
|
// 加载基础方法
|
2024-12-20 23:39:29 +08:00
|
|
|
|
foreach ((string name, MethodInfo method) item in tempMethods)
|
|
|
|
|
|
{
|
2025-07-16 16:16:19 +08:00
|
|
|
|
SereinScript.AddStaticFunction(item.name, item.method);
|
2024-12-20 23:39:29 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-05-30 15:42:59 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 代码改变后
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="value"></param>
|
|
|
|
|
|
/// <exception cref="NotImplementedException"></exception>
|
|
|
|
|
|
partial void OnScriptChanged(string value)
|
|
|
|
|
|
{
|
|
|
|
|
|
IsScriptChanged = true;
|
|
|
|
|
|
}
|
2025-03-14 21:38:07 +08:00
|
|
|
|
|
2025-05-30 15:42:59 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 节点创建时
|
|
|
|
|
|
/// </summary>
|
2024-12-21 20:47:31 +08:00
|
|
|
|
public override void OnCreating()
|
|
|
|
|
|
{
|
|
|
|
|
|
var md = MethodDetails;
|
|
|
|
|
|
var pd = md.ParameterDetailss ??= new ParameterDetails[1];
|
|
|
|
|
|
md.ParamsArgIndex = 0;
|
|
|
|
|
|
pd[0] = new ParameterDetails
|
|
|
|
|
|
{
|
|
|
|
|
|
Index = 0,
|
2025-07-23 15:57:57 +08:00
|
|
|
|
Name = "arg",
|
2024-12-21 20:47:31 +08:00
|
|
|
|
IsExplicitData = true,
|
|
|
|
|
|
DataValue = string.Empty,
|
2025-07-17 22:46:40 +08:00
|
|
|
|
DataType = typeof(string),
|
|
|
|
|
|
ExplicitType = typeof(string),
|
2024-12-21 20:47:31 +08:00
|
|
|
|
ArgDataSourceNodeGuid = string.Empty,
|
|
|
|
|
|
ArgDataSourceType = ConnectionArgSourceType.GetPreviousNodeData,
|
|
|
|
|
|
NodeModel = this,
|
2025-03-15 15:43:42 +08:00
|
|
|
|
InputType = ParameterValueInputType.Input,
|
2024-12-21 20:47:31 +08:00
|
|
|
|
Items = null,
|
|
|
|
|
|
IsParams = true,
|
2025-05-30 15:42:59 +08:00
|
|
|
|
//Description = "脚本节点入参"
|
2024-12-21 20:47:31 +08:00
|
|
|
|
};
|
2025-07-17 22:46:40 +08:00
|
|
|
|
md.ReturnType = typeof(void); // 默认无返回
|
2024-12-21 20:47:31 +08:00
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
2025-07-30 11:29:12 +08:00
|
|
|
|
/// 保存项目时保存脚本代码、方法入参类型、返回值类型
|
2024-12-21 20:47:31 +08:00
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="nodeInfo"></param>
|
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
|
public override NodeInfo SaveCustomData(NodeInfo nodeInfo)
|
|
|
|
|
|
{
|
2025-07-30 11:29:12 +08:00
|
|
|
|
var paramsTypeName = MethodDetails.ParameterDetailss.Select(pd =>
|
|
|
|
|
|
{
|
|
|
|
|
|
return new ScriptArgInfo
|
|
|
|
|
|
{
|
|
|
|
|
|
Index = pd.Index,
|
|
|
|
|
|
ArgName = pd.Name,
|
|
|
|
|
|
ArgType = pd.DataType.FullName,
|
|
|
|
|
|
};
|
|
|
|
|
|
}).ToArray();
|
|
|
|
|
|
|
2024-12-21 20:47:31 +08:00
|
|
|
|
dynamic data = new ExpandoObject();
|
|
|
|
|
|
data.Script = Script ?? "";
|
2025-07-30 11:29:12 +08:00
|
|
|
|
data.ParamsTypeName = paramsTypeName;
|
|
|
|
|
|
data.ReturnTypeName = MethodDetails.ReturnType;
|
2024-12-21 20:47:31 +08:00
|
|
|
|
nodeInfo.CustomData = data;
|
|
|
|
|
|
return nodeInfo;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-07-30 11:29:12 +08:00
|
|
|
|
private class ScriptArgInfo
|
|
|
|
|
|
{
|
|
|
|
|
|
public int Index { get; set; }
|
|
|
|
|
|
public string? ArgName { get; set; }
|
|
|
|
|
|
public string? ArgType { get; set; }
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2024-12-20 23:39:29 +08:00
|
|
|
|
/// <summary>
|
2024-12-21 20:47:31 +08:00
|
|
|
|
/// 加载自定义数据
|
2024-12-20 23:39:29 +08:00
|
|
|
|
/// </summary>
|
2024-12-21 20:47:31 +08:00
|
|
|
|
/// <param name="nodeInfo"></param>
|
|
|
|
|
|
public override void LoadCustomData(NodeInfo nodeInfo)
|
|
|
|
|
|
{
|
|
|
|
|
|
this.Script = nodeInfo.CustomData?.Script ?? "";
|
2025-07-30 11:29:12 +08:00
|
|
|
|
|
2025-03-14 21:38:07 +08:00
|
|
|
|
|
2025-07-30 11:29:12 +08:00
|
|
|
|
var paramCount = Math.Min(MethodDetails.ParameterDetailss.Length, nodeInfo.ParameterData.Length);
|
2025-03-14 21:38:07 +08:00
|
|
|
|
// 更新变量名
|
2025-07-30 11:29:12 +08:00
|
|
|
|
for (int i = 0; i < paramCount; i++)
|
2025-03-14 21:38:07 +08:00
|
|
|
|
{
|
2025-07-30 11:29:12 +08:00
|
|
|
|
var pd = MethodDetails.ParameterDetailss[i];
|
|
|
|
|
|
pd.Name = nodeInfo.ParameterData[i].ArgName;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
|
{
|
|
|
|
|
|
string paramsTypeNameJson = nodeInfo.CustomData?.ParamsTypeName.ToString() ?? "[]";
|
|
|
|
|
|
ScriptArgInfo[] array = JsonHelper.Deserialize<ScriptArgInfo[]>(paramsTypeNameJson);
|
|
|
|
|
|
|
|
|
|
|
|
string returnTypeName = nodeInfo.CustomData?.ReturnTypeName ?? typeof(object);
|
|
|
|
|
|
Type?[] argType = array.Select(item => string.IsNullOrWhiteSpace(item.ArgType) ? null : Type.GetType(item.ArgType) ?? typeof(Unit)).ToArray();
|
|
|
|
|
|
Type? resType = Type.GetType(returnTypeName);
|
|
|
|
|
|
for (int i = 0; i < paramCount; i++)
|
|
|
|
|
|
{
|
|
|
|
|
|
var pd = MethodDetails.ParameterDetailss[i];
|
|
|
|
|
|
pd.DataType = argType[i];
|
|
|
|
|
|
}
|
|
|
|
|
|
MethodDetails.ReturnType = resType;
|
|
|
|
|
|
}
|
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
|
{
|
|
|
|
|
|
SereinEnv.WriteLine(InfoType.ERROR ,$"加载脚本自定义数据类型信息时发生异常:{ex.Message}");
|
2025-03-14 21:38:07 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-07-17 22:46:40 +08:00
|
|
|
|
//ReloadScript();// 加载时重新解析
|
2025-07-06 14:34:49 +08:00
|
|
|
|
IsScriptChanged = false; // 重置脚本改变标志
|
2024-12-21 20:47:31 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-07-30 11:29:12 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2024-12-21 20:47:31 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 重新加载脚本代码
|
|
|
|
|
|
/// </summary>
|
2025-07-17 22:46:40 +08:00
|
|
|
|
public bool ReloadScript()
|
2024-12-20 23:39:29 +08:00
|
|
|
|
{
|
|
|
|
|
|
try
|
|
|
|
|
|
{
|
2025-07-26 19:36:54 +08:00
|
|
|
|
HashSet<string> varNames = new HashSet<string>();
|
|
|
|
|
|
foreach (var pd in MethodDetails.ParameterDetailss)
|
|
|
|
|
|
{
|
|
|
|
|
|
if (varNames.Contains(pd.Name))
|
|
|
|
|
|
{
|
|
|
|
|
|
throw new Exception($"脚本节点重复的变量名称:{pd.Name} - {Guid}");
|
|
|
|
|
|
}
|
|
|
|
|
|
varNames.Add(pd.Name);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
var argTypes = MethodDetails.ParameterDetailss
|
|
|
|
|
|
.Select(pd =>
|
|
|
|
|
|
{
|
|
|
|
|
|
if (pd.ArgDataSourceType == ConnectionArgSourceType.GetPreviousNodeData)
|
|
|
|
|
|
{
|
|
|
|
|
|
var Type = pd.DataType;
|
|
|
|
|
|
return (pd.Name, Type);
|
|
|
|
|
|
}
|
|
|
|
|
|
if (Env.TryGetNodeModel(pd.ArgDataSourceNodeGuid, out var node) &&
|
|
|
|
|
|
node.MethodDetails?.ReturnType is not null)
|
|
|
|
|
|
{
|
|
|
|
|
|
pd.DataType = node.MethodDetails.ReturnType;
|
|
|
|
|
|
var Type = node.MethodDetails.ReturnType;
|
|
|
|
|
|
return (pd.Name, Type);
|
|
|
|
|
|
}
|
|
|
|
|
|
return default;
|
|
|
|
|
|
})
|
|
|
|
|
|
.Where(x => x != default)
|
|
|
|
|
|
.ToDictionary(x => x.Name, x => x.Type); // 准备预定义类型
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
var returnType = sereinScript.ParserScript(Script, argTypes); // 开始解析获取程序主节点
|
|
|
|
|
|
MethodDetails.ReturnType = returnType;
|
|
|
|
|
|
return true;
|
|
|
|
|
|
}
|
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
|
{
|
|
|
|
|
|
SereinEnv.WriteLine(InfoType.WARN, ex.Message);
|
|
|
|
|
|
return false; // 解析失败
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 转换为 C# 代码,并且附带方法信息
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public SereinScriptMethodInfo? ToCsharpMethodInfo(string methodName)
|
|
|
|
|
|
{
|
|
|
|
|
|
try
|
|
|
|
|
|
{
|
|
|
|
|
|
if (string.IsNullOrWhiteSpace(methodName))
|
|
|
|
|
|
{
|
|
|
|
|
|
var tmp = Guid.Replace("-", "");
|
|
|
|
|
|
methodName = $"FlowMethod_{tmp}";
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-03-14 21:38:07 +08:00
|
|
|
|
HashSet<string> varNames = new HashSet<string>();
|
|
|
|
|
|
foreach (var pd in MethodDetails.ParameterDetailss)
|
|
|
|
|
|
{
|
|
|
|
|
|
if (varNames.Contains(pd.Name))
|
|
|
|
|
|
{
|
|
|
|
|
|
throw new Exception($"脚本节点重复的变量名称:{pd.Name} - {Guid}");
|
|
|
|
|
|
}
|
|
|
|
|
|
varNames.Add(pd.Name);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-07-17 22:46:40 +08:00
|
|
|
|
var argTypes = MethodDetails.ParameterDetailss
|
|
|
|
|
|
.Select(pd =>
|
|
|
|
|
|
{
|
2025-07-23 15:57:57 +08:00
|
|
|
|
if(pd.ArgDataSourceType == ConnectionArgSourceType.GetPreviousNodeData)
|
|
|
|
|
|
{
|
|
|
|
|
|
var Type = pd.DataType;
|
|
|
|
|
|
return (pd.Name, Type);
|
|
|
|
|
|
}
|
2025-07-17 22:46:40 +08:00
|
|
|
|
if (Env.TryGetNodeModel(pd.ArgDataSourceNodeGuid, out var node) &&
|
|
|
|
|
|
node.MethodDetails?.ReturnType is not null)
|
|
|
|
|
|
{
|
|
|
|
|
|
pd.DataType = node.MethodDetails.ReturnType;
|
2025-07-23 15:57:57 +08:00
|
|
|
|
var Type = node.MethodDetails.ReturnType;
|
|
|
|
|
|
return (pd.Name, Type);
|
2025-07-17 22:46:40 +08:00
|
|
|
|
}
|
|
|
|
|
|
return default;
|
|
|
|
|
|
})
|
|
|
|
|
|
.Where(x => x != default)
|
2025-07-23 15:57:57 +08:00
|
|
|
|
.ToDictionary(x => x.Name, x => x.Type); // 准备预定义类型
|
2025-07-06 14:34:49 +08:00
|
|
|
|
|
2025-07-17 22:46:40 +08:00
|
|
|
|
|
|
|
|
|
|
var returnType = sereinScript.ParserScript(Script, argTypes); // 开始解析获取程序主节点
|
|
|
|
|
|
MethodDetails.ReturnType = returnType;
|
2025-07-28 12:16:29 +08:00
|
|
|
|
var scriptMethodInfo = sereinScript.ConvertCSharpCode(methodName, argTypes);
|
2025-07-26 19:36:54 +08:00
|
|
|
|
return scriptMethodInfo;
|
2024-12-20 23:39:29 +08:00
|
|
|
|
}
|
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
|
{
|
2025-07-23 15:57:57 +08:00
|
|
|
|
SereinEnv.WriteLine(InfoType.WARN, ex.Message);
|
2025-07-26 19:36:54 +08:00
|
|
|
|
return null; // 解析失败
|
2024-12-20 23:39:29 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 执行脚本
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="context"></param>
|
2025-07-30 21:15:07 +08:00
|
|
|
|
/// <param name="token"></param>
|
2024-12-20 23:39:29 +08:00
|
|
|
|
/// <returns></returns>
|
2025-07-23 16:20:41 +08:00
|
|
|
|
public override async Task<FlowResult> ExecutingAsync(IFlowContext context, CancellationToken token)
|
2025-05-30 15:42:59 +08:00
|
|
|
|
{
|
2025-07-11 20:52:21 +08:00
|
|
|
|
var result = await ExecutingAsync(this, context, token);
|
|
|
|
|
|
return result;
|
2025-05-30 15:42:59 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 流程接口提供参数进行调用脚本节点
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="flowCallNode"></param>
|
|
|
|
|
|
/// <param name="context"></param>
|
|
|
|
|
|
/// <param name="token"></param>
|
|
|
|
|
|
/// <returns></returns>
|
2025-07-23 16:20:41 +08:00
|
|
|
|
public async Task<FlowResult> ExecutingAsync(NodeModelBase flowCallNode, IFlowContext context, CancellationToken token)
|
2024-12-20 23:39:29 +08:00
|
|
|
|
{
|
2025-07-30 11:29:12 +08:00
|
|
|
|
if (token.IsCancellationRequested) return FlowResult.Fail(this.Guid, context, "流程已通过token取消");
|
2025-05-30 15:42:59 +08:00
|
|
|
|
var @params = await flowCallNode.GetParametersAsync(context, token);
|
2025-03-14 21:38:07 +08:00
|
|
|
|
|
2024-12-26 22:24:44 +08:00
|
|
|
|
IScriptInvokeContext scriptContext = new ScriptInvokeContext(context);
|
|
|
|
|
|
|
2025-03-14 21:38:07 +08:00
|
|
|
|
if (@params[0] is object[] agrDatas)
|
|
|
|
|
|
{
|
|
|
|
|
|
for (int i = 0; i < agrDatas.Length; i++)
|
|
|
|
|
|
{
|
2025-05-30 15:42:59 +08:00
|
|
|
|
var argName = flowCallNode.MethodDetails.ParameterDetailss[i].Name;
|
2025-03-14 21:38:07 +08:00
|
|
|
|
var argData = agrDatas[i];
|
|
|
|
|
|
scriptContext.SetVarValue(argName, argData);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2025-03-15 14:02:12 +08:00
|
|
|
|
|
|
|
|
|
|
FlowRunCompleteHandler onFlowStop = (e) =>
|
|
|
|
|
|
{
|
|
|
|
|
|
scriptContext.OnExit();
|
|
|
|
|
|
};
|
2025-03-14 21:38:07 +08:00
|
|
|
|
|
2025-06-22 21:53:37 +08:00
|
|
|
|
var envEvent = context.Env.Event;
|
2025-06-02 19:17:30 +08:00
|
|
|
|
envEvent.FlowRunComplete += onFlowStop; // 防止运行后台流程
|
2025-03-20 22:54:10 +08:00
|
|
|
|
|
2025-07-30 11:29:12 +08:00
|
|
|
|
if (token.IsCancellationRequested) return FlowResult.Fail(this.Guid, context, "流程已通过token取消");
|
2025-03-20 22:54:10 +08:00
|
|
|
|
|
2025-07-16 16:16:19 +08:00
|
|
|
|
var result = await sereinScript.InterpreterAsync(scriptContext); // 从入口节点执行
|
2025-06-02 19:17:30 +08:00
|
|
|
|
envEvent.FlowRunComplete -= onFlowStop;
|
2025-07-30 11:29:12 +08:00
|
|
|
|
return FlowResult.OK(this.Guid, context, result);
|
2024-12-20 23:39:29 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|