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 11KB

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