using Serein.Library; using Serein.Library.Api; using Serein.Library.Utils; using Serein.Script; using Serein.Script.Node; using System; using System.Collections.Generic; using System.Dynamic; using System.Linq; using System.Linq.Expressions; using System.Reflection; using System.Text; using System.Threading.Tasks; using System.Xml.Linq; namespace Serein.NodeFlow.Model { [NodeProperty(ValuePath = NodeValuePath.Node)] public partial class SingleScriptNode : NodeModelBase { [PropertyInfo(IsNotification = true)] private string _script; } /// /// 流程脚本节点 /// public partial class SingleScriptNode : NodeModelBase { /// /// 脚本节点是基础节点 /// public override bool IsBase => true; private IScriptFlowApi ScriptFlowApi { get; } private ASTNode mainNode; private SereinScriptInterpreter ScriptInterpreter; /// /// 构建流程脚本节点 /// /// public SingleScriptNode(IFlowEnvironment environment):base(environment) { //ScriptFlowApi = environment.IOC.Get(); ScriptFlowApi = new ScriptFlowApi(environment, this); ScriptInterpreter = new SereinScriptInterpreter(); } static SingleScriptNode() { // 挂载静态方法 var tempMethods = typeof(BaseFunc).GetMethods().Where(method => !(method.Name.Equals("GetHashCode") || method.Name.Equals("Equals") || method.Name.Equals("ToString") || method.Name.Equals("GetType") )).Select(method => (method.Name, method)).ToArray(); // 加载基础方法 foreach ((string name, MethodInfo method) item in tempMethods) { SereinScriptInterpreter.AddStaticFunction(item.name, item.method); } } public override void OnCreating() { MethodInfo? method = this.GetType().GetMethod(nameof(GetFlowApi)); if (method != null) { ScriptInterpreter.AddFunction(nameof(GetFlowApi), method, () => this); // 挂载获取流程接口 } var md = MethodDetails; var pd = md.ParameterDetailss ??= new ParameterDetails[1]; md.ParamsArgIndex = 0; pd[0] = new ParameterDetails { Index = 0, Name = "object", IsExplicitData = true, DataValue = string.Empty, DataType = typeof(object), ExplicitType = typeof(object), ArgDataSourceNodeGuid = string.Empty, ArgDataSourceType = ConnectionArgSourceType.GetPreviousNodeData, NodeModel = this, ExplicitTypeName = "Value", Items = null, IsParams = true, }; } /// /// 导出脚本代码 /// /// /// public override NodeInfo SaveCustomData(NodeInfo nodeInfo) { dynamic data = new ExpandoObject(); data.Script = Script ?? ""; nodeInfo.CustomData = data; return nodeInfo; } /// /// 加载自定义数据 /// /// public override void LoadCustomData(NodeInfo nodeInfo) { this.Script = nodeInfo.CustomData?.Script ?? ""; } /// /// 重新加载脚本代码 /// public void ReloadScript() { try { var p = new SereinScriptParser(Script); mainNode = p.Parse(); } catch (Exception ex) { SereinEnv.WriteLine(InfoType.ERROR, ex.ToString()); } } /// /// 执行脚本 /// /// /// public override async Task ExecutingAsync(IDynamicContext context) { var @params = await GetParametersAsync(context); //dynamic obj = ((object[])@params[0])[0]; //try //{ // SereinEnv.WriteLine(InfoType.INFO, "Dynamic Object Value :" + obj.VarInfo); //} //catch (Exception ex) //{ // SereinEnv.WriteLine(ex); //} //ScriptFlowApi.Context = context; // 并发破坏了数据状态 context.AddOrUpdate($"{context.Guid}_{this.Guid}_Params", @params[0]); // 后面再改 mainNode ??= new SereinScriptParser(Script).Parse(); IScriptInvokeContext scriptContext = new ScriptInvokeContext(context); var result = await ScriptInterpreter.InterpretAsync(scriptContext, mainNode); // 从入口节点执行 //SereinEnv.WriteLine(InfoType.INFO, "FlowContext Guid : " + context.Guid); return result; } #region 挂载的方法 public IScriptFlowApi GetFlowApi() { return ScriptFlowApi; } private static class BaseFunc { public static DateTime GetNow() => DateTime.Now; public static Type TypeOf(object type) { return type.GetType(); } public static void Print(object value) { SereinEnv.WriteLine(InfoType.INFO, value?.ToString()); } #region 数据转换 public static int ToInt(object value) { return int.Parse(value.ToString()); } public static double ToDouble(object value) { return double.Parse(value.ToString()); } public static bool ToBool(object value) { return bool.Parse(value.ToString()); } #endregion public static async Task Delay(object value) { if (value is int @int) { Console.WriteLine($"等待{@int}ms"); await Task.Delay(@int); } else if (value is TimeSpan timeSpan) { Console.WriteLine($"等待{timeSpan}"); await Task.Delay(timeSpan); } } } #endregion } }