mirror of
https://gitee.com/langsisi_admin/serein-flow
synced 2026-03-03 00:00:49 +08:00
新增了脚本节点对于集合对象[]下标/key取值的语法支持。修复了加载项目文件时无法加载脚本节点变量名称的问题
This commit is contained in:
@@ -102,6 +102,7 @@ namespace Serein.Library
|
|||||||
{
|
{
|
||||||
return new ParameterData[0];
|
return new ParameterData[0];
|
||||||
}
|
}
|
||||||
|
|
||||||
if (MethodDetails.ParameterDetailss.Length > 0)
|
if (MethodDetails.ParameterDetailss.Length > 0)
|
||||||
{
|
{
|
||||||
return MethodDetails.ParameterDetailss
|
return MethodDetails.ParameterDetailss
|
||||||
@@ -182,8 +183,7 @@ namespace Serein.Library
|
|||||||
{
|
{
|
||||||
md.ParameterDetailss = new ParameterDetails[0];
|
md.ParameterDetailss = new ParameterDetails[0];
|
||||||
}
|
}
|
||||||
LoadCustomData(nodeInfo); // 加载自定义数据
|
|
||||||
|
|
||||||
var pds = md.ParameterDetailss; // 当前节点的入参描述数组
|
var pds = md.ParameterDetailss; // 当前节点的入参描述数组
|
||||||
#region 类库方法型节点加载参数
|
#region 类库方法型节点加载参数
|
||||||
if (nodeInfo.ParameterData.Length > pds.Length && md.HasParamsArg)
|
if (nodeInfo.ParameterData.Length > pds.Length && md.HasParamsArg)
|
||||||
@@ -215,7 +215,10 @@ namespace Serein.Library
|
|||||||
pd.ArgDataSourceType = EnumHelper.ConvertEnum<ConnectionArgSourceType>(pdInfo.SourceType);
|
pd.ArgDataSourceType = EnumHelper.ConvertEnum<ConnectionArgSourceType>(pdInfo.SourceType);
|
||||||
pd.ArgDataSourceNodeGuid = pdInfo.SourceNodeGuid;
|
pd.ArgDataSourceNodeGuid = pdInfo.SourceNodeGuid;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
LoadCustomData(nodeInfo); // 加载自定义数据
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -143,14 +143,14 @@ namespace Serein.NodeFlow.Model
|
|||||||
varNames.Add(pd.Name);
|
varNames.Add(pd.Name);
|
||||||
}
|
}
|
||||||
|
|
||||||
//StringBuilder sb = new StringBuilder();
|
StringBuilder sb = new StringBuilder();
|
||||||
//foreach (var pd in MethodDetails.ParameterDetailss)
|
foreach (var pd in MethodDetails.ParameterDetailss)
|
||||||
//{
|
{
|
||||||
// sb.AppendLine($"let {pd.Name};"); // 提前声明这些变量
|
sb.AppendLine($"let {pd.Name};"); // 提前声明这些变量
|
||||||
//}
|
}
|
||||||
//sb.Append(Script);
|
sb.Append(Script);
|
||||||
//var p = new SereinScriptParser(sb.ToString());
|
var p = new SereinScriptParser(sb.ToString());
|
||||||
var p = new SereinScriptParser(Script);
|
//var p = new SereinScriptParser(Script);
|
||||||
mainNode = p.Parse(); // 开始解析
|
mainNode = p.Parse(); // 开始解析
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
@@ -168,6 +168,7 @@ namespace Serein.NodeFlow.Model
|
|||||||
public override async Task<object?> ExecutingAsync(IDynamicContext context)
|
public override async Task<object?> ExecutingAsync(IDynamicContext context)
|
||||||
{
|
{
|
||||||
var @params = await GetParametersAsync(context);
|
var @params = await GetParametersAsync(context);
|
||||||
|
|
||||||
|
|
||||||
//context.AddOrUpdate($"{context.Guid}_{this.Guid}_Params", @params[0]); // 后面再改
|
//context.AddOrUpdate($"{context.Guid}_{this.Guid}_Params", @params[0]); // 后面再改
|
||||||
ReloadScript();// 每次都重新解析
|
ReloadScript();// 每次都重新解析
|
||||||
@@ -183,9 +184,17 @@ namespace Serein.NodeFlow.Model
|
|||||||
scriptContext.SetVarValue(argName, argData);
|
scriptContext.SetVarValue(argName, argData);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
FlowRunCompleteHandler onFlowStop = (e) =>
|
||||||
|
{
|
||||||
|
scriptContext.OnExit();
|
||||||
|
};
|
||||||
|
|
||||||
|
var envEvent = (IFlowEnvironmentEvent)context.Env;
|
||||||
|
envEvent.OnFlowRunComplete += onFlowStop; // 防止运行后台流程
|
||||||
var result = await ScriptInterpreter.InterpretAsync(scriptContext, mainNode); // 从入口节点执行
|
var result = await ScriptInterpreter.InterpretAsync(scriptContext, mainNode); // 从入口节点执行
|
||||||
|
envEvent.OnFlowRunComplete -= onFlowStop;
|
||||||
//SereinEnv.WriteLine(InfoType.INFO, "FlowContext Guid : " + context.Guid);
|
//SereinEnv.WriteLine(InfoType.INFO, "FlowContext Guid : " + context.Guid);
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
@@ -237,7 +246,6 @@ namespace Serein.NodeFlow.Model
|
|||||||
}
|
}
|
||||||
else if (value is TimeSpan timeSpan)
|
else if (value is TimeSpan timeSpan)
|
||||||
{
|
{
|
||||||
|
|
||||||
Console.WriteLine($"等待{timeSpan}");
|
Console.WriteLine($"等待{timeSpan}");
|
||||||
await Task.Delay(timeSpan);
|
await Task.Delay(timeSpan);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -11,9 +11,11 @@ namespace Serein.Script.Node
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
public class CollectionIndexNode : ASTNode
|
public class CollectionIndexNode : ASTNode
|
||||||
{
|
{
|
||||||
|
public ASTNode TargetValue { get; }
|
||||||
public ASTNode IndexValue { get; }
|
public ASTNode IndexValue { get; }
|
||||||
public CollectionIndexNode(ASTNode indexValue)
|
public CollectionIndexNode(ASTNode collectionValue,ASTNode indexValue)
|
||||||
{
|
{
|
||||||
|
this.TargetValue = collectionValue;
|
||||||
this.IndexValue = indexValue;
|
this.IndexValue = indexValue;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,7 +3,9 @@ using Serein.Library;
|
|||||||
using Serein.Library.Api;
|
using Serein.Library.Api;
|
||||||
using Serein.Library.Utils;
|
using Serein.Library.Utils;
|
||||||
using Serein.Script.Node;
|
using Serein.Script.Node;
|
||||||
|
using System.ComponentModel.Design;
|
||||||
using System.Reflection;
|
using System.Reflection;
|
||||||
|
using System.Reflection.Metadata.Ecma335;
|
||||||
using System.Runtime.CompilerServices;
|
using System.Runtime.CompilerServices;
|
||||||
using System.Xml.Linq;
|
using System.Xml.Linq;
|
||||||
|
|
||||||
@@ -21,12 +23,15 @@ namespace Serein.Script
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 脚本运行上下文
|
/// 脚本运行上下文
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public interface IScriptInvokeContext
|
public interface IScriptInvokeContext
|
||||||
{
|
{
|
||||||
IDynamicContext FlowContext { get; }
|
IDynamicContext FlowContext { get; }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 是否该退出了
|
/// 是否该退出了
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -35,7 +40,7 @@ namespace Serein.Script
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// 是否严格检查 Null 值 (禁止使用 Null)
|
/// 是否严格检查 Null 值 (禁止使用 Null)
|
||||||
/// </summary>
|
/// </summary>
|
||||||
bool IsCheckNullValue { get; }
|
bool IsCheckNullValue { get; set; }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 获取变量的值
|
/// 获取变量的值
|
||||||
@@ -56,7 +61,7 @@ namespace Serein.Script
|
|||||||
/// 结束调用
|
/// 结束调用
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <returns></returns>
|
/// <returns></returns>
|
||||||
bool Exit();
|
void OnExit();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -73,11 +78,13 @@ namespace Serein.Script
|
|||||||
/// 定义的变量
|
/// 定义的变量
|
||||||
/// </summary>
|
/// </summary>
|
||||||
private Dictionary<string, object> _variables = new Dictionary<string, object>();
|
private Dictionary<string, object> _variables = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
private CancellationTokenSource _tokenSource = new CancellationTokenSource();
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 是否该退出了
|
/// 是否该退出了
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public bool IsReturn { get; set; }
|
public bool IsReturn => _tokenSource.IsCancellationRequested;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 是否严格检查 Null 值 (禁止使用 Null)
|
/// 是否严格检查 Null 值 (禁止使用 Null)
|
||||||
@@ -104,7 +111,7 @@ namespace Serein.Script
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
bool IScriptInvokeContext.Exit()
|
void IScriptInvokeContext.OnExit()
|
||||||
{
|
{
|
||||||
// 清理脚本中加载的非托管资源
|
// 清理脚本中加载的非托管资源
|
||||||
foreach (var nodeObj in _variables.Values)
|
foreach (var nodeObj in _variables.Values)
|
||||||
@@ -121,8 +128,8 @@ namespace Serein.Script
|
|||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
_tokenSource.Cancel();
|
||||||
_variables.Clear();
|
_variables.Clear();
|
||||||
return true ;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
@@ -213,6 +220,7 @@ namespace Serein.Script
|
|||||||
private async Task<object?> ExecutionProgramNodeAsync(IScriptInvokeContext context, ProgramNode programNode)
|
private async Task<object?> ExecutionProgramNodeAsync(IScriptInvokeContext context, ProgramNode programNode)
|
||||||
{
|
{
|
||||||
// 加载变量
|
// 加载变量
|
||||||
|
|
||||||
|
|
||||||
// 遍历 ProgramNode 中的所有语句并执行它们
|
// 遍历 ProgramNode 中的所有语句并执行它们
|
||||||
foreach (var statement in programNode.Statements)
|
foreach (var statement in programNode.Statements)
|
||||||
@@ -290,8 +298,13 @@ namespace Serein.Script
|
|||||||
/// <exception cref="Exception"></exception>
|
/// <exception cref="Exception"></exception>
|
||||||
private async Task ExectutionWhileNodeAsync(IScriptInvokeContext context, WhileNode whileNode)
|
private async Task ExectutionWhileNodeAsync(IScriptInvokeContext context, WhileNode whileNode)
|
||||||
{
|
{
|
||||||
|
|
||||||
while (true)
|
while (true)
|
||||||
{
|
{
|
||||||
|
if (context.IsReturn) // 停止流程
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
var result = await EvaluateAsync(context, whileNode.Condition) ?? throw new SereinSciptException(whileNode, $"条件语句返回了 null");
|
var result = await EvaluateAsync(context, whileNode.Condition) ?? throw new SereinSciptException(whileNode, $"条件语句返回了 null");
|
||||||
if (result is not bool condition)
|
if (result is not bool condition)
|
||||||
{
|
{
|
||||||
@@ -316,7 +329,11 @@ namespace Serein.Script
|
|||||||
private async Task ExecutionAssignmentNodeAsync(IScriptInvokeContext context, AssignmentNode assignmentNode)
|
private async Task ExecutionAssignmentNodeAsync(IScriptInvokeContext context, AssignmentNode assignmentNode)
|
||||||
{
|
{
|
||||||
var tmp = await EvaluateAsync(context, assignmentNode.Value);
|
var tmp = await EvaluateAsync(context, assignmentNode.Value);
|
||||||
context.SetVarValue(assignmentNode.Variable, tmp);
|
if(tmp is not null)
|
||||||
|
{
|
||||||
|
context.SetVarValue(assignmentNode.Variable, tmp);
|
||||||
|
|
||||||
|
}
|
||||||
}
|
}
|
||||||
private async Task<object> InterpretFunctionCallAsync(IScriptInvokeContext context, FunctionCallNode functionCallNode)
|
private async Task<object> InterpretFunctionCallAsync(IScriptInvokeContext context, FunctionCallNode functionCallNode)
|
||||||
{
|
{
|
||||||
@@ -378,6 +395,7 @@ namespace Serein.Script
|
|||||||
switch (node)
|
switch (node)
|
||||||
{
|
{
|
||||||
case ProgramNode programNode: // AST树入口
|
case ProgramNode programNode: // AST树入口
|
||||||
|
|
||||||
var scritResult = await ExecutionProgramNodeAsync(context, programNode);
|
var scritResult = await ExecutionProgramNodeAsync(context, programNode);
|
||||||
return scritResult; // 遍历 ProgramNode 中的所有语句并执行它们
|
return scritResult; // 遍历 ProgramNode 中的所有语句并执行它们
|
||||||
case ClassTypeDefinitionNode classTypeDefinitionNode: // 定义类型
|
case ClassTypeDefinitionNode classTypeDefinitionNode: // 定义类型
|
||||||
@@ -463,16 +481,18 @@ namespace Serein.Script
|
|||||||
throw new SereinSciptException(objectInstantiationNode, $"使用了未定义的类型\"{objectInstantiationNode.TypeName}\"");
|
throw new SereinSciptException(objectInstantiationNode, $"使用了未定义的类型\"{objectInstantiationNode.TypeName}\"");
|
||||||
|
|
||||||
}
|
}
|
||||||
case FunctionCallNode callNode:
|
case FunctionCallNode callNode: // 调用方法
|
||||||
return await InterpretFunctionCallAsync(context, callNode); // 调用方法返回函数的返回值
|
return await InterpretFunctionCallAsync(context, callNode); // 调用方法返回函数的返回值
|
||||||
case MemberFunctionCallNode memberFunctionCallNode:
|
case MemberFunctionCallNode memberFunctionCallNode: // 对象方法调用
|
||||||
return await CallMemberFunction(context, memberFunctionCallNode);
|
return await CallMemberFunction(context, memberFunctionCallNode);
|
||||||
case MemberAccessNode memberAccessNode:
|
case MemberAccessNode memberAccessNode: // 对象成员访问
|
||||||
return await GetValue(context, memberAccessNode);
|
return await GetValue(context, memberAccessNode);
|
||||||
case ReturnNode returnNode: //
|
case CollectionIndexNode collectionIndexNode:
|
||||||
|
return await GetCollectionValue(context, collectionIndexNode);
|
||||||
|
case ReturnNode returnNode: // 返回内容
|
||||||
return await EvaluateAsync(context, returnNode.Value); // 直接返回响应的内容
|
return await EvaluateAsync(context, returnNode.Value); // 直接返回响应的内容
|
||||||
default:
|
default:
|
||||||
throw new SereinSciptException(node, "解释器 EvaluateAsync() 未实现节点行为");
|
throw new SereinSciptException(node, $"解释器 EvaluateAsync() 未实现{node}节点行为");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -593,6 +613,74 @@ namespace Serein.Script
|
|||||||
return lastProperty.GetValue(target);
|
return lastProperty.GetValue(target);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 获取集合中的成员
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="memberAccessNode"></param>
|
||||||
|
/// <returns></returns>
|
||||||
|
/// <exception cref="SereinSciptException"></exception>
|
||||||
|
public async Task<object?> GetCollectionValue(IScriptInvokeContext context, CollectionIndexNode collectionIndexNode)
|
||||||
|
{
|
||||||
|
var target = await EvaluateAsync(context, collectionIndexNode.TargetValue); // 获取对象
|
||||||
|
if (target is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException($"解析{collectionIndexNode}节点时,TargetValue返回空。");
|
||||||
|
}
|
||||||
|
|
||||||
|
// 解析数组/集合名与索引部分
|
||||||
|
var targetType = target.GetType(); // 目标对象的类型
|
||||||
|
#region 处理键值对
|
||||||
|
if (targetType.IsGenericType && targetType.GetGenericTypeDefinition() == typeof(Dictionary<,>))
|
||||||
|
{
|
||||||
|
// 目标是键值对
|
||||||
|
var method = targetType.GetMethod("get_Item", BindingFlags.Public | BindingFlags.Instance);
|
||||||
|
if (method is not null)
|
||||||
|
{
|
||||||
|
var key = await EvaluateAsync(context, collectionIndexNode.IndexValue); // 获取索引值;
|
||||||
|
var result = method.Invoke(target, new object[] { key });
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endregion
|
||||||
|
#region 处理集合对象
|
||||||
|
else
|
||||||
|
{
|
||||||
|
var indexValue = await EvaluateAsync(context, collectionIndexNode.IndexValue); // 获取索引值
|
||||||
|
object? result;
|
||||||
|
if (indexValue is int index)
|
||||||
|
{
|
||||||
|
// 获取数组或集合对象
|
||||||
|
// 访问数组或集合中的指定索引
|
||||||
|
if (target is Array array)
|
||||||
|
{
|
||||||
|
if (index < 0 || index >= array.Length)
|
||||||
|
{
|
||||||
|
throw new ArgumentException($"解析{collectionIndexNode}节点时,数组下标越界。");
|
||||||
|
}
|
||||||
|
result = array.GetValue(index);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
else if (target is IList<object> list)
|
||||||
|
{
|
||||||
|
if (index < 0 || index >= list.Count)
|
||||||
|
{
|
||||||
|
throw new ArgumentException($"解析{collectionIndexNode}节点时,数组下标越界。");
|
||||||
|
}
|
||||||
|
result = list[index];
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
throw new ArgumentException($"解析{collectionIndexNode}节点时,左值并非有效集合。");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
throw new ArgumentException($"解析{collectionIndexNode}节点时,左值并非有效集合。");
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 缓存method委托
|
/// 缓存method委托
|
||||||
|
|||||||
@@ -127,6 +127,11 @@ namespace Serein.Script
|
|||||||
// 对象成员的获取
|
// 对象成员的获取
|
||||||
return ParseMemberAccessOrAssignment();
|
return ParseMemberAccessOrAssignment();
|
||||||
}
|
}
|
||||||
|
else if (_tempToken.Type == TokenType.SquareBracketsLeft)
|
||||||
|
{
|
||||||
|
// 数组 index; 字典 key obj.Member[xxx];
|
||||||
|
return ParseCollectionIndex();
|
||||||
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
// 不是函数调用,是变量赋值或其他
|
// 不是函数调用,是变量赋值或其他
|
||||||
@@ -292,10 +297,13 @@ namespace Serein.Script
|
|||||||
|
|
||||||
public ASTNode ParseCollectionIndex()
|
public ASTNode ParseCollectionIndex()
|
||||||
{
|
{
|
||||||
string functionName = _currentToken.Value.ToString();
|
var identifierNode = new IdentifierNode(_currentToken.Value.ToString()).SetTokenInfo(_currentToken);
|
||||||
_currentToken = _lexer.NextToken(); // consume identifier
|
|
||||||
|
|
||||||
if (_currentToken.Type != TokenType.ParenthesisLeft)
|
string collectionName = _currentToken.Value.ToString();
|
||||||
|
//_lexer.NextToken(); // consume "["
|
||||||
|
_currentToken = _lexer.NextToken(); // consume identifier
|
||||||
|
// ParenthesisLeft
|
||||||
|
if (_currentToken.Type != TokenType.SquareBracketsLeft)
|
||||||
throw new Exception("Expected '[' after function name");
|
throw new Exception("Expected '[' after function name");
|
||||||
|
|
||||||
_currentToken = _lexer.NextToken(); // consume "["
|
_currentToken = _lexer.NextToken(); // consume "["
|
||||||
@@ -303,7 +311,7 @@ namespace Serein.Script
|
|||||||
ASTNode indexValue = Expression(); // get index value
|
ASTNode indexValue = Expression(); // get index value
|
||||||
|
|
||||||
_currentToken = _lexer.NextToken(); // consume "]"
|
_currentToken = _lexer.NextToken(); // consume "]"
|
||||||
return new CollectionIndexNode(indexValue).SetTokenInfo(_currentToken);
|
return new CollectionIndexNode(identifierNode,indexValue).SetTokenInfo(_currentToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@@ -631,6 +639,14 @@ namespace Serein.Script
|
|||||||
{
|
{
|
||||||
return ParseMemberAccessOrAssignment();
|
return ParseMemberAccessOrAssignment();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// 数组 index; 字典 key obj.Member[xxx];
|
||||||
|
if (_identifierPeekToken.Type == TokenType.SquareBracketsLeft)
|
||||||
|
{
|
||||||
|
return ParseCollectionIndex();
|
||||||
|
}
|
||||||
|
|
||||||
_currentToken = _lexer.NextToken(); // 消耗标识符
|
_currentToken = _lexer.NextToken(); // 消耗标识符
|
||||||
return new IdentifierNode(identifier.ToString()).SetTokenInfo(_currentToken);
|
return new IdentifierNode(identifier.ToString()).SetTokenInfo(_currentToken);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -60,7 +60,7 @@
|
|||||||
<Setter Property="ContentTemplate">
|
<Setter Property="ContentTemplate">
|
||||||
<Setter.Value>
|
<Setter.Value>
|
||||||
<DataTemplate>
|
<DataTemplate>
|
||||||
<TextBox MinWidth="50" Text="{Binding Name, Mode=TwoWay, UpdateSourceTrigger=LostFocus}"/>
|
<TextBox MinWidth="50" Text="{Binding Name, Mode=TwoWay}"/>
|
||||||
</DataTemplate>
|
</DataTemplate>
|
||||||
</Setter.Value>
|
</Setter.Value>
|
||||||
</Setter>
|
</Setter>
|
||||||
|
|||||||
Reference in New Issue
Block a user