You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

LuExpressionConverterVisitor.cs 12KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Linq.Expressions;
  5. using Luticate2.Auth.Utils.Dbo;
  6. using Luticate2.Auth.Utils.Dbo.Result;
  7. using Luticate2.Auth.Utils.Exceptions;
  8. using Luticate2.Auth.Utils.Interfaces;
  9. namespace Luticate2.Auth.Utils.Business.Converters.ExpressionConverter
  10. {
  11. public interface ILuExpressionConverterVisitorOptions
  12. {
  13. IDictionary<ParameterExpression, Expression> Parameters { get; }
  14. ILuConvertersTypeConverter TypeConverter { get; }
  15. ILuObjectConverterDescriptorOptions DescriptorOptions { get; }
  16. ILuExpressionParamReplaceVisitorOptions VisitorOptions { get; }
  17. }
  18. public class LuExpressionConverterVisitor : ExpressionVisitor
  19. {
  20. public ILuExpressionConverterVisitorOptions Options { get; }
  21. protected IServiceProvider ServiceProvider { get; }
  22. public LuExpressionConverterVisitor(ILuExpressionConverterVisitorOptions options, IServiceProvider serviceProvider)
  23. {
  24. Options = options;
  25. ServiceProvider = serviceProvider;
  26. }
  27. protected LuResult<Stack<Expression>> StackSimpleExpression(Expression expression, Stack<Expression> stack)
  28. {
  29. stack.Push(expression);
  30. if (expression is MemberExpression memberExpression)
  31. {
  32. if (memberExpression.Expression != null)
  33. {
  34. var result = StackSimpleExpression(memberExpression.Expression, stack);
  35. return result;
  36. }
  37. return LuResult<Stack<Expression>>.Ok(stack);
  38. }
  39. else if (expression is MethodCallExpression methodCallExpression)
  40. {
  41. if (methodCallExpression.Object != null)
  42. {
  43. var result = StackSimpleExpression(methodCallExpression.Object, stack);
  44. return result;
  45. }
  46. return LuResult<Stack<Expression>>.Ok(stack);
  47. }
  48. else
  49. {
  50. return LuResult<Stack<Expression>>.Ok(stack);
  51. }
  52. }
  53. protected LuResult<ILuObjectConverterDescriptor> GetConverterDescriptor(Type typeFrom, Type typeTo)
  54. {
  55. var type = typeof(ILuObjectConverterDescriptor<,>);
  56. var gtype = type.MakeGenericType(typeFrom, typeTo);
  57. var descriptor = (ILuObjectConverterDescriptor) ServiceProvider.GetService(gtype);
  58. if (descriptor == null && typeFrom == typeTo)
  59. {
  60. descriptor = (ILuObjectConverterDescriptor) ServiceProvider.GetService(typeof(ILuObjectConverterDescriptorIdentity));
  61. }
  62. if (descriptor != null)
  63. {
  64. return LuResult<ILuObjectConverterDescriptor>.Ok(descriptor);
  65. }
  66. return LuResult<ILuObjectConverterDescriptor>.Error(LuStatus.InternalError.ToInt(),
  67. $"Could not find converter descriptor for {typeFrom} => {typeTo}");
  68. }
  69. protected LuResult<Expression> ConvertMemberExpression(MemberExpression memberExpression, Expression newExpression, Type typeTo)
  70. {
  71. var typeFrom = memberExpression.Expression == null ? memberExpression.Member.DeclaringType : memberExpression.Expression.Type;
  72. var descriptorResult = GetConverterDescriptor(typeFrom, typeTo);
  73. if (!descriptorResult)
  74. {
  75. return descriptorResult.To<Expression>();
  76. }
  77. var descriptor = descriptorResult.Data;
  78. var valueExpressionResult = descriptor.GetMemberValueExpression(memberExpression.Member, Options.DescriptorOptions);
  79. if (!valueExpressionResult)
  80. {
  81. return valueExpressionResult.To<Expression>();
  82. }
  83. var valueExpression = valueExpressionResult.Data;
  84. if (valueExpression.Parameters[0].Type != typeTo)
  85. {
  86. return LuResult<Expression>.Error(LuStatus.InternalError.ToInt(),
  87. $"Invalid conversion lambda for {typeFrom} => {typeTo}");
  88. }
  89. if (newExpression != null)
  90. {
  91. Options.Parameters.Add(valueExpression.Parameters[0], newExpression);
  92. }
  93. var visitor = new LuExpressionParamReplaceVisitor(Options.VisitorOptions);
  94. newExpression = visitor.Visit(valueExpression.Body);
  95. Options.Parameters.Remove(valueExpression.Parameters[0]);
  96. return LuResult<Expression>.Ok(newExpression);
  97. }
  98. protected LuResult<Expression> ConvertMethodCallExpression(MethodCallExpression methodCallExpression, Expression newExpression, Type typeTo)
  99. {
  100. var typeFrom = methodCallExpression.Object == null ? methodCallExpression.Method.DeclaringType : methodCallExpression.Object.Type;
  101. var descriptorResult = GetConverterDescriptor(typeFrom, typeTo);
  102. if (!descriptorResult)
  103. {
  104. return descriptorResult.To<Expression>();
  105. }
  106. var descriptor = descriptorResult.Data;
  107. var valueExpressionResult = descriptor.GetMethodValueExpression(methodCallExpression.Method, Options.DescriptorOptions);
  108. if (!valueExpressionResult)
  109. {
  110. return valueExpressionResult.To<Expression>();
  111. }
  112. var valueExpression = valueExpressionResult.Data;
  113. if (valueExpression.Parameters.Count != methodCallExpression.Arguments.Count + 1)
  114. {
  115. return LuResult<Expression>.Error(LuStatus.InternalError.ToInt(),
  116. $"Converter descriptor lambda has incorrect number of arguments for {typeFrom} => {typeTo} => {methodCallExpression.Method}");
  117. }
  118. if (valueExpression.Parameters[0].Type != typeTo)
  119. {
  120. return LuResult<Expression>.Error(LuStatus.InternalError.ToInt(),
  121. $"Invalid conversion lambda for {typeFrom} => {typeTo}");
  122. }
  123. var visitorConverter = new LuExpressionConverterVisitor(Options, ServiceProvider);
  124. for (var i = 0; i < methodCallExpression.Arguments.Count; ++i)
  125. {
  126. var convertedArgument = visitorConverter.Visit(methodCallExpression.Arguments[i]);
  127. Options.Parameters.Add(valueExpression.Parameters[i + 1], convertedArgument);
  128. }
  129. Options.Parameters.Add(valueExpression.Parameters[0], newExpression);
  130. var visitor = new LuExpressionParamReplaceVisitor(Options.VisitorOptions);
  131. newExpression = visitor.Visit(valueExpression.Body);
  132. Options.Parameters.Remove(valueExpression.Parameters[0]);
  133. return LuResult<Expression>.Ok(newExpression);
  134. }
  135. protected LuResult<Expression> ReplaceExpression(Expression expression)
  136. {
  137. var stackResult = StackSimpleExpression(expression, new Stack<Expression>());
  138. if (!stackResult)
  139. {
  140. return stackResult.To<Expression>();
  141. }
  142. Expression newExpression;
  143. var currentExp = stackResult.Data.Pop();
  144. if (currentExp is ParameterExpression parameterExpression)
  145. {
  146. if (!Options.Parameters.ContainsKey(parameterExpression))
  147. {
  148. return LuResult<Expression>.Error(LuStatus.InternalError.ToInt(),
  149. $"Could not find a conversion for parameter {parameterExpression}");
  150. }
  151. newExpression = Options.Parameters[parameterExpression];
  152. }
  153. else if (currentExp is MemberExpression memberExpression)
  154. {
  155. var typeFrom = memberExpression.Member.DeclaringType;
  156. var typeToResult = Options.TypeConverter.ConvertType(typeFrom);
  157. if (!typeToResult)
  158. {
  159. return typeToResult.To<Expression>();
  160. }
  161. var typeTo = typeToResult.Data;
  162. var convertResult = ConvertMemberExpression(memberExpression, null, typeTo);
  163. if (!convertResult)
  164. {
  165. return convertResult;
  166. }
  167. newExpression = convertResult.Data;
  168. }
  169. else if (currentExp is MethodCallExpression methodCallExpression)
  170. {
  171. var typeFrom = methodCallExpression.Method.DeclaringType;
  172. var typeToResult = Options.TypeConverter.ConvertType(typeFrom);
  173. if (!typeToResult)
  174. {
  175. return typeToResult.To<Expression>();
  176. }
  177. var typeTo = typeToResult.Data;
  178. var convertResult = ConvertMethodCallExpression(methodCallExpression, null, typeTo);
  179. if (!convertResult)
  180. {
  181. return convertResult;
  182. }
  183. newExpression = convertResult.Data;
  184. }
  185. else
  186. {
  187. var visitor = new LuExpressionConverterVisitor(Options, ServiceProvider);
  188. newExpression = visitor.Visit(currentExp);
  189. }
  190. if (newExpression == null)
  191. {
  192. return LuResult<Expression>.Error(LuStatus.InternalError.ToInt(),
  193. $"Invalid simple expression first member: {currentExp}", "");
  194. }
  195. while (stackResult.Data.Any())
  196. {
  197. currentExp = stackResult.Data.Pop();
  198. if (currentExp is MemberExpression memberExpression)
  199. {
  200. var typeTo = newExpression.Type;
  201. var convertResult = ConvertMemberExpression(memberExpression, newExpression, typeTo);
  202. if (!convertResult)
  203. {
  204. return convertResult;
  205. }
  206. newExpression = convertResult.Data;
  207. }
  208. else if (currentExp is MethodCallExpression methodCallExpression)
  209. {
  210. var typeTo = newExpression.Type;
  211. var convertResult = ConvertMethodCallExpression(methodCallExpression, newExpression, typeTo);
  212. if (!convertResult)
  213. {
  214. return convertResult;
  215. }
  216. newExpression = convertResult.Data;
  217. }
  218. else
  219. {
  220. return LuResult<Expression>.Error(LuStatus.InternalError.ToInt(),
  221. $"Unknown expression type (should not happen) {currentExp}");
  222. }
  223. }
  224. return LuResult<Expression>.Ok(newExpression);
  225. }
  226. protected override Expression VisitMember(MemberExpression node)
  227. {
  228. return ReplaceExpression(node).ThrowIfNotSuccess().Data;
  229. }
  230. protected override Expression VisitMethodCall(MethodCallExpression node)
  231. {
  232. return ReplaceExpression(node).ThrowIfNotSuccess().Data;
  233. }
  234. protected override Expression VisitParameter(ParameterExpression node)
  235. {
  236. if (!Options.Parameters.ContainsKey(node))
  237. {
  238. LuResult<Expression>.Error(LuStatus.InternalError.ToInt(),
  239. $"Could not find a conversion for parameter {node}").Throw();
  240. }
  241. return Options.Parameters[node];
  242. }
  243. protected override Expression VisitLambda<T>(Expression<T> node)
  244. {
  245. var convertedParams = new List<ParameterExpression>();
  246. foreach (var parameter in node.Parameters)
  247. {
  248. var convertedTypeResult = Options.TypeConverter.ConvertType(parameter.Type).ThrowIfNotSuccess();
  249. var convertedType = convertedTypeResult.Data;
  250. var convertedParam = Expression.Parameter(convertedType);
  251. Options.Parameters.Add(parameter, convertedParam);
  252. convertedParams.Add(convertedParam);
  253. }
  254. var convertedBody = Visit(node.Body);
  255. foreach (var parameter in node.Parameters)
  256. {
  257. Options.Parameters.Remove(parameter);
  258. }
  259. var convertedLambda = Expression.Lambda(convertedBody, convertedParams);
  260. return convertedLambda;
  261. }
  262. }
  263. }