CoreExpression.cs 5.4 KB

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