CoreExpression.cs 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
  1. using Expressive;
  2. using Expressive.Expressions;
  3. using Expressive.Functions;
  4. using InABox.Clients;
  5. using System.Diagnostics.CodeAnalysis;
  6. using System;
  7. using System.Collections.Generic;
  8. using System.Linq;
  9. using System.Text;
  10. namespace InABox.Core
  11. {
  12. using FncType = Func<IExpression[], IDictionary<string, object>, object?>;
  13. public class CoreExpressionFunction
  14. {
  15. public string Group { get; set; }
  16. public string Name { get; set; }
  17. public string? Description { get; set; }
  18. public List<Parameter> Parameters { get; set; }
  19. public FncType Function { get; set; }
  20. public class Parameter
  21. {
  22. public string Name { get; set; }
  23. public string Type { get; set; }
  24. public Parameter(string name, string type)
  25. {
  26. Name = name;
  27. Type = type;
  28. }
  29. }
  30. public CoreExpressionFunction(string group, string name, FncType function, List<Parameter> parameters, string? description = null)
  31. {
  32. Group = group;
  33. Name = name;
  34. Parameters = parameters;
  35. Function = function;
  36. Description = description;
  37. }
  38. }
  39. public interface IExpressionModel { }
  40. public interface IExpressionModel<TReturn> : IExpressionModel { }
  41. public class CoreExpression
  42. {
  43. private Expression Expression;
  44. public IReadOnlyCollection<string> ReferencedVariables => Expression.ReferencedVariables;
  45. protected virtual Type? ReturnType { get; }
  46. public CoreExpression(string expressionString)
  47. {
  48. Expression = new Expression(expressionString);
  49. foreach (var function in Functions)
  50. {
  51. Expression.RegisterFunction(function.Name, function.Function);
  52. }
  53. }
  54. public object? Evaluate(Dictionary<string, object?>? variables)
  55. {
  56. var result = Expression.Evaluate(variables);
  57. if(ReturnType != null)
  58. {
  59. return CoreUtils.ChangeType(result, ReturnType);
  60. }
  61. return result;
  62. }
  63. public static List<string> GetModelVariables(Type modelType)
  64. {
  65. return CoreUtils.PropertyList(modelType, x => true).Select(x => x.Name).ToList();
  66. }
  67. public static List<string> GetModelVariables<TModel>() where TModel : IExpressionModel
  68. => GetModelVariables(typeof(TModel));
  69. #region Static
  70. public static List<CoreExpressionFunction> Functions = new List<CoreExpressionFunction>();
  71. static CoreExpression()
  72. {
  73. RegisterFunction("String", "Format", (p, v) =>
  74. {
  75. if (!(p[0].Evaluate(v) is string format)) throw new Exception("No format string given for Format()");
  76. return string.Format(format, p.Skip(1).Select(x => x.Evaluate(v)).ToArray());
  77. }, "string fmt", "...");
  78. RegisterFunction("Other", "Client_LoadDocument", "Loads a byte array from the database Document table, with the ID {docID}.", Fnc_LoadDocument, "Guid docID");
  79. }
  80. private static object? Fnc_LoadDocument(IExpression[] p, IDictionary<string, object> v)
  81. {
  82. var id = p[0].Evaluate(v);
  83. if (id is null)
  84. return null;
  85. if (!(id is Guid docID))
  86. return null;
  87. return new Client<Document>()
  88. .Query(
  89. new Filter<Document>(x => x.ID).IsEqualTo(docID),
  90. new Columns<Document>(x => x.Data))
  91. .Rows.FirstOrDefault()
  92. ?.Get<Document, byte[]>(x => x.Data);
  93. }
  94. public static void RegisterFunction(string group, string name, FncType function, params string[] parameters)
  95. => RegisterFunction(group, name, null, function, parameters);
  96. public static void RegisterFunction(string group, string name, string? description, FncType function, params string[] parameters) =>
  97. Functions.Add(new CoreExpressionFunction(group, name, function, parameters.Select(x =>
  98. {
  99. var parts = x.Split(' ');
  100. if (parts.Length == 1)
  101. return new CoreExpressionFunction.Parameter(parts[0], "");
  102. return new CoreExpressionFunction.Parameter(parts[1], parts[0]);
  103. }).ToList(), description));
  104. #endregion
  105. }
  106. public class CoreExpression<TModel, TReturn> : CoreExpression
  107. where TModel : IExpressionModel<TReturn>
  108. {
  109. protected override Type? ReturnType => typeof(TReturn);
  110. public CoreExpression(string expressionString): base(expressionString) { }
  111. [return: MaybeNull]
  112. public new TReturn Evaluate(Dictionary<string, object?>? variables)
  113. {
  114. var result = base.Evaluate(variables);
  115. if(result is TReturn ret)
  116. {
  117. return ret;
  118. }
  119. return default;
  120. }
  121. [return: MaybeNull]
  122. public TReturn Evaluate(TModel model)
  123. {
  124. var values = new Dictionary<string, object?>();
  125. foreach(var variable in ReferencedVariables)
  126. {
  127. values[variable] = CoreUtils.GetPropertyValue(model, variable);
  128. }
  129. var result = base.Evaluate(values);
  130. if(result is TReturn ret)
  131. {
  132. return ret;
  133. }
  134. return default;
  135. }
  136. }
  137. }