/src/EntityFramework/Core/Mapping/ViewGeneration/Structures/TypeRestriction.cs

# · C# · 240 lines · 181 code · 32 blank · 27 comment · 22 complexity · 97fccee8a9c4da9ad6417d76ac6987cf MD5 · raw file

  1. namespace System.Data.Entity.Core.Mapping.ViewGeneration.Structures
  2. {
  3. using System.Collections.Generic;
  4. using System.Data.Entity.Core.Common.CommandTrees;
  5. using System.Data.Entity.Core.Common.CommandTrees.ExpressionBuilder;
  6. using System.Data.Entity.Core.Common.Utils;
  7. using System.Data.Entity.Core.Mapping.ViewGeneration.CqlGeneration;
  8. using System.Data.Entity.Core.Metadata.Edm;
  9. using System.Diagnostics;
  10. using System.Linq;
  11. using System.Text;
  12. using DomainBoolExpr =
  13. System.Data.Entity.Core.Common.Utils.Boolean.BoolExpr<Common.Utils.Boolean.DomainConstraint<BoolLiteral, Constant>>;
  14. /// <summary>
  15. /// A class that denotes the boolean expression: "varType in values".
  16. /// See the comments in <see cref="MemberRestriction"/> for complete and incomplete restriction objects.
  17. /// </summary>
  18. internal class TypeRestriction : MemberRestriction
  19. {
  20. #region Constructors
  21. /// <summary>
  22. /// Creates an incomplete type restriction of the form "<paramref name="member"/> in <paramref name="values"/>".
  23. /// </summary>
  24. internal TypeRestriction(MemberPath member, IEnumerable<EdmType> values)
  25. : base(new MemberProjectedSlot(member), CreateTypeConstants(values))
  26. {
  27. }
  28. /// <summary>
  29. /// Creates an incomplete type restriction of the form "<paramref name="member"/> = <paramref name="value"/>".
  30. /// </summary>
  31. internal TypeRestriction(MemberPath member, Constant value)
  32. : base(new MemberProjectedSlot(member), value)
  33. {
  34. Debug.Assert(value is TypeConstant || value.IsNull(), "Type or NULL expected.");
  35. }
  36. /// <summary>
  37. /// Creates a complete type restriction of the form "<paramref name="slot"/> in <paramref name="domain"/>".
  38. /// </summary>
  39. internal TypeRestriction(MemberProjectedSlot slot, Domain domain)
  40. : base(slot, domain)
  41. {
  42. }
  43. #endregion
  44. #region Methods
  45. /// <summary>
  46. /// Requires: <see cref="MemberRestriction.IsComplete"/> is true.
  47. /// </summary>
  48. internal override DomainBoolExpr FixRange(Set<Constant> range, MemberDomainMap memberDomainMap)
  49. {
  50. Debug.Assert(IsComplete, "Ranges are fixed only for complete type restrictions.");
  51. var possibleValues = memberDomainMap.GetDomain(RestrictedMemberSlot.MemberPath);
  52. BoolLiteral newLiteral = new TypeRestriction(RestrictedMemberSlot, new Domain(range, possibleValues));
  53. return newLiteral.GetDomainBoolExpression(memberDomainMap);
  54. }
  55. internal override BoolLiteral RemapBool(Dictionary<MemberPath, MemberPath> remap)
  56. {
  57. var newVar = RestrictedMemberSlot.RemapSlot(remap);
  58. return new TypeRestriction(newVar, Domain);
  59. }
  60. internal override MemberRestriction CreateCompleteMemberRestriction(IEnumerable<Constant> possibleValues)
  61. {
  62. Debug.Assert(!IsComplete, "CreateCompleteMemberRestriction must be called only for incomplete restrictions.");
  63. return new TypeRestriction(RestrictedMemberSlot, new Domain(Domain.Values, possibleValues));
  64. }
  65. internal override StringBuilder AsEsql(StringBuilder builder, string blockAlias, bool skipIsNotNull)
  66. {
  67. // Add Cql of the form "(T.A IS OF (ONLY Person) OR .....)"
  68. // Important to enclose all the OR statements in parens.
  69. if (Domain.Count > 1)
  70. {
  71. builder.Append('(');
  72. }
  73. var isFirst = true;
  74. foreach (var constant in Domain.Values)
  75. {
  76. var typeConstant = constant as TypeConstant;
  77. Debug.Assert(typeConstant != null || constant.IsNull(), "Constants for type checks must be type constants or NULLs");
  78. if (isFirst == false)
  79. {
  80. builder.Append(" OR ");
  81. }
  82. isFirst = false;
  83. if (Helper.IsRefType(RestrictedMemberSlot.MemberPath.EdmType))
  84. {
  85. builder.Append("Deref(");
  86. RestrictedMemberSlot.MemberPath.AsEsql(builder, blockAlias);
  87. builder.Append(')');
  88. }
  89. else
  90. {
  91. // non-reference type
  92. RestrictedMemberSlot.MemberPath.AsEsql(builder, blockAlias);
  93. }
  94. if (constant.IsNull())
  95. {
  96. builder.Append(" IS NULL");
  97. }
  98. else
  99. {
  100. // type constant
  101. builder.Append(" IS OF (ONLY ");
  102. CqlWriter.AppendEscapedTypeName(builder, typeConstant.EdmType);
  103. builder.Append(')');
  104. }
  105. }
  106. if (Domain.Count > 1)
  107. {
  108. builder.Append(')');
  109. }
  110. return builder;
  111. }
  112. internal override DbExpression AsCqt(DbExpression row, bool skipIsNotNull)
  113. {
  114. var cqt = RestrictedMemberSlot.MemberPath.AsCqt(row);
  115. if (Helper.IsRefType(RestrictedMemberSlot.MemberPath.EdmType))
  116. {
  117. cqt = cqt.Deref();
  118. }
  119. if (Domain.Count == 1)
  120. {
  121. // Single value
  122. cqt = cqt.IsOfOnly(TypeUsage.Create(((TypeConstant)Domain.Values.Single()).EdmType));
  123. }
  124. else
  125. {
  126. // Multiple values: build list of var IsOnOnly(t1), var = IsOnOnly(t1), ..., then OR them all.
  127. var operands = Domain.Values.Select(t => (DbExpression)cqt.IsOfOnly(TypeUsage.Create(((TypeConstant)t).EdmType))).ToList();
  128. cqt = Helpers.BuildBalancedTreeInPlace(operands, (prev, next) => prev.Or(next));
  129. }
  130. return cqt;
  131. }
  132. internal override StringBuilder AsUserString(StringBuilder builder, string blockAlias, bool skipIsNotNull)
  133. {
  134. // Add user readable string of the form "T.A IS a (Person OR .....)"
  135. if (Helper.IsRefType(RestrictedMemberSlot.MemberPath.EdmType))
  136. {
  137. builder.Append("Deref(");
  138. RestrictedMemberSlot.MemberPath.AsEsql(builder, blockAlias);
  139. builder.Append(')');
  140. }
  141. else
  142. {
  143. // non-reference type
  144. RestrictedMemberSlot.MemberPath.AsEsql(builder, blockAlias);
  145. }
  146. if (Domain.Count > 1)
  147. {
  148. builder.Append(" is a (");
  149. }
  150. else
  151. {
  152. builder.Append(" is type ");
  153. }
  154. var isFirst = true;
  155. foreach (var constant in Domain.Values)
  156. {
  157. var typeConstant = constant as TypeConstant;
  158. Debug.Assert(typeConstant != null || constant.IsNull(), "Constants for type checks must be type constants or NULLs");
  159. if (isFirst == false)
  160. {
  161. builder.Append(" OR ");
  162. }
  163. if (constant.IsNull())
  164. {
  165. builder.Append(" NULL");
  166. }
  167. else
  168. {
  169. CqlWriter.AppendEscapedTypeName(builder, typeConstant.EdmType);
  170. }
  171. isFirst = false;
  172. }
  173. if (Domain.Count > 1)
  174. {
  175. builder.Append(')');
  176. }
  177. return builder;
  178. }
  179. /// <summary>
  180. /// Given a list of <paramref name="types"/> (which can contain nulls), returns a corresponding list of <see cref="TypeConstant"/>s for those types.
  181. /// </summary>
  182. private static IEnumerable<Constant> CreateTypeConstants(IEnumerable<EdmType> types)
  183. {
  184. foreach (var type in types)
  185. {
  186. if (type == null)
  187. {
  188. yield return Constant.Null;
  189. }
  190. else
  191. {
  192. yield return new TypeConstant(type);
  193. }
  194. }
  195. }
  196. #endregion
  197. #region String methods
  198. internal override void ToCompactString(StringBuilder builder)
  199. {
  200. builder.Append("type(");
  201. RestrictedMemberSlot.ToCompactString(builder);
  202. builder.Append(") IN (");
  203. StringUtil.ToCommaSeparatedStringSorted(builder, Domain.Values);
  204. builder.Append(")");
  205. }
  206. #endregion
  207. }
  208. }