QueryBuilder.cs 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287
  1. using System;
  2. using System.Collections;
  3. using System.Collections.Generic;
  4. using System.Linq;
  5. using System.Linq.Expressions;
  6. namespace Vit.Linq.QueryBuilder
  7. {
  8. public static class QueryBuilder
  9. {
  10. public static Func<T, bool> ToPredicate<T>(IFilterRule rule)
  11. {
  12. return ToExpression<T>(rule)?.Compile();
  13. }
  14. public static string ToExpressionString<T>(IFilterRule rule)
  15. {
  16. return ToExpression<T>(rule)?.ToString();
  17. }
  18. public static Expression<Func<T, bool>> ToExpression<T>(IFilterRule rule)
  19. {
  20. var exp = ToLambdaExpression(rule, typeof(T));
  21. return (Expression<Func<T, bool>>)exp;
  22. }
  23. public static LambdaExpression ToLambdaExpression(IFilterRule rule, Type targetType)
  24. {
  25. ParameterExpression parameter = Expression.Parameter(targetType);
  26. var expression = ConvertToExpression(rule, parameter);
  27. if (expression == null)
  28. {
  29. return null;
  30. }
  31. return Expression.Lambda(expression, parameter);
  32. }
  33. public static ECondition GetCondition(IFilterRule filter)
  34. {
  35. return filter.condition?.ToLower() == "or" ? ECondition.or : ECondition.and;
  36. }
  37. static Expression ConvertToExpression(IFilterRule rule, ParameterExpression parameter)
  38. {
  39. if (rule == null) return null;
  40. // #1 nested filter rules
  41. if (rule.rules?.Any() == true)
  42. {
  43. return ConvertToExpression(rule.rules, parameter, GetCondition(rule));
  44. }
  45. // #2 simple rule
  46. if (string.IsNullOrWhiteSpace(rule.field))
  47. {
  48. return null;
  49. }
  50. MemberExpression memberExp = LinqHelp.GetFieldMemberExpression(parameter, rule.field);
  51. #region get Expression
  52. Type fieldType = memberExp.Type;
  53. switch (rule.@operator)
  54. {
  55. #region ##1 null
  56. case "is null":
  57. {
  58. return IsNull();
  59. }
  60. case "is not null":
  61. {
  62. return Expression.Not(IsNull());
  63. }
  64. #endregion
  65. #region ##2 number
  66. case "=":
  67. {
  68. return Expression.Equal(memberExp, ConvertValueExp());
  69. }
  70. case "!=":
  71. {
  72. return Expression.NotEqual(memberExp, ConvertValueExp());
  73. }
  74. case ">":
  75. {
  76. return Expression.GreaterThan(memberExp, ConvertValueExp());
  77. }
  78. case ">=":
  79. {
  80. return Expression.GreaterThanOrEqual(memberExp, ConvertValueExp());
  81. }
  82. case "<":
  83. {
  84. return Expression.LessThan(memberExp, ConvertValueExp());
  85. }
  86. case "<=":
  87. {
  88. return Expression.LessThanOrEqual(memberExp, ConvertValueExp());
  89. }
  90. #endregion
  91. #region ##3 array
  92. case "in":
  93. {
  94. return In();
  95. }
  96. case "not in":
  97. {
  98. return Expression.Not(In());
  99. }
  100. #endregion
  101. #region ##4 string
  102. case "contains":
  103. {
  104. var nullCheck = Expression.Call(typeof(string), "IsNullOrEmpty", null, memberExp);
  105. var contains = Expression.Call(memberExp, "Contains", null, ConvertValueExp());
  106. return Expression.AndAlso(Expression.Not(nullCheck), contains);
  107. }
  108. case "not contains":
  109. {
  110. var nullCheck = Expression.Call(typeof(string), "IsNullOrEmpty", null, memberExp);
  111. var contains = Expression.Call(memberExp, "Contains", null, ConvertValueExp());
  112. return Expression.OrElse(nullCheck, Expression.Not(contains));
  113. }
  114. case "starts with":
  115. {
  116. var nullCheck = Expression.Not(Expression.Call(typeof(string), "IsNullOrEmpty", null, memberExp));
  117. var startsWith = Expression.Call(memberExp, "StartsWith", null, ConvertValueExp());
  118. return Expression.AndAlso(nullCheck, startsWith);
  119. }
  120. case "ends with":
  121. {
  122. var nullCheck = Expression.Not(Expression.Call(typeof(string), "IsNullOrEmpty", null, memberExp));
  123. var endsWith = Expression.Call(memberExp, "EndsWith", null, ConvertValueExp());
  124. return Expression.AndAlso(nullCheck, endsWith);
  125. }
  126. case "is null or empty":
  127. {
  128. return Expression.Call(typeof(string), "IsNullOrEmpty", null, memberExp);
  129. }
  130. case "is not null or empty":
  131. {
  132. return Expression.Not(Expression.Call(typeof(string), "IsNullOrEmpty", null, memberExp));
  133. }
  134. #endregion
  135. }
  136. #endregion
  137. return null;
  138. #region Method ConvertValueExp
  139. UnaryExpression ConvertValueExp()
  140. {
  141. object value = rule.value;
  142. if (value != null)
  143. {
  144. Type valueType = Nullable.GetUnderlyingType(fieldType) ?? fieldType;
  145. value = Convert.ChangeType(value, valueType);
  146. }
  147. Expression<Func<object>> valueLamba = () => value;
  148. return Expression.Convert(valueLamba.Body, fieldType);
  149. }
  150. Expression IsNull()
  151. {
  152. var isNullable = !fieldType.IsValueType || Nullable.GetUnderlyingType(fieldType) != null;
  153. if (isNullable)
  154. {
  155. var nullValue = Expression.Constant(null, fieldType);
  156. return Expression.Equal(memberExp, nullValue);
  157. }
  158. return Expression.Constant(false, typeof(bool));
  159. }
  160. Expression In()
  161. {
  162. Expression arrayExp = null;
  163. #region build arrayExp
  164. {
  165. Type valueType = typeof(IEnumerable<>).MakeGenericType(fieldType);
  166. object value = null;
  167. if (rule.value != null)
  168. {
  169. //value = Vit.Core.Module.Serialization.Json.Deserialize(Vit.Core.Module.Serialization.Json.Serialize(rule.value), valueType);
  170. if (rule.value is IEnumerable arr)
  171. {
  172. value = ConvertToList(arr, fieldType);
  173. }
  174. }
  175. Expression<Func<object>> valueLamba = () => value;
  176. arrayExp = Expression.Convert(valueLamba.Body, valueType);
  177. }
  178. #endregion
  179. var inCheck = Expression.Call(typeof(System.Linq.Enumerable), "Contains", new[] { fieldType }, arrayExp, memberExp);
  180. return inCheck;
  181. }
  182. #endregion
  183. }
  184. static Expression ConvertToExpression(IEnumerable<IFilterRule> rules, ParameterExpression parameter, ECondition condition = ECondition.and)
  185. {
  186. if (rules?.Any() != true)
  187. {
  188. return null;
  189. }
  190. Expression expression = null;
  191. foreach (var rule in rules)
  192. {
  193. var curExp = ConvertToExpression(rule, parameter);
  194. if (curExp != null)
  195. expression = Append(expression, curExp);
  196. }
  197. return expression;
  198. #region Method Append
  199. Expression Append(Expression exp1, Expression exp2)
  200. {
  201. if (exp1 == null)
  202. {
  203. return exp2;
  204. }
  205. if (exp2 == null)
  206. {
  207. return exp1;
  208. }
  209. return condition == ECondition.or ? Expression.OrElse(exp1, exp2) : Expression.AndAlso(exp1, exp2);
  210. }
  211. #endregion
  212. }
  213. #region ConvertToList
  214. internal static object ConvertToList(IEnumerable values, Type fieldType)
  215. {
  216. var methodInfo = typeof(QueryBuilder).GetMethod("ConvertToListByType", System.Reflection.BindingFlags.Static | System.Reflection.BindingFlags.NonPublic).MakeGenericMethod(fieldType);
  217. return methodInfo.Invoke(null, new object[] { values });
  218. }
  219. internal static List<T> ConvertToListByType<T>(IEnumerable values)
  220. {
  221. Type valueType = typeof(T);
  222. var list = new List<T>();
  223. foreach (var item in values)
  224. {
  225. var value = (T)Convert.ChangeType(item, valueType);
  226. list.Add(value);
  227. }
  228. return list;
  229. }
  230. #endregion
  231. }
  232. }