You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			565 lines
		
	
	
		
			22 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			565 lines
		
	
	
		
			22 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|   | //--------------------------------------------------------------------- | ||
|  | // <copyright file="EntityContainer.cs" company="Microsoft"> | ||
|  | //      Copyright (c) Microsoft Corporation.  All rights reserved. | ||
|  | // </copyright> | ||
|  | // | ||
|  | // @owner       [....] | ||
|  | // @backupOwner [....] | ||
|  | //--------------------------------------------------------------------- | ||
|  | 
 | ||
|  | namespace System.Data.EntityModel.SchemaObjectModel | ||
|  | { | ||
|  |     using System; | ||
|  |     using System.Collections.Generic; | ||
|  |     using System.Data.Common.Utils; | ||
|  |     using System.Data.Entity; | ||
|  |     using System.Data.Metadata.Edm; | ||
|  |     using System.Diagnostics; | ||
|  |     using System.Xml; | ||
|  | 
 | ||
|  |     /// <summary> | ||
|  |     /// Represents an EntityContainer element. | ||
|  |     /// </summary> | ||
|  |     [System.Diagnostics.DebuggerDisplay("Name={Name}")] | ||
|  |     internal sealed class EntityContainer : SchemaType | ||
|  |     { | ||
|  |         #region Instance Fields | ||
|  | 
 | ||
|  |         private SchemaElementLookUpTable<SchemaElement> _members; | ||
|  |         private ISchemaElementLookUpTable<EntityContainerEntitySet> _entitySets; | ||
|  |         private ISchemaElementLookUpTable<EntityContainerRelationshipSet> _relationshipSets; | ||
|  |         private ISchemaElementLookUpTable<Function> _functionImports; | ||
|  |         private string _unresolvedExtendedEntityContainerName; | ||
|  |         private EntityContainer _entityContainerGettingExtended; | ||
|  |         private bool _isAlreadyValidated; | ||
|  |         private bool _isAlreadyResolved; | ||
|  | 
 | ||
|  |         #endregion | ||
|  | 
 | ||
|  |         #region Constructors | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// Constructs an EntityContainer | ||
|  |         /// </summary> | ||
|  |         /// <param name="parentElement">Reference to the schema element.</param> | ||
|  |         public EntityContainer(Schema parentElement) | ||
|  |             : base(parentElement) | ||
|  |         { | ||
|  |             if(Schema.DataModel == SchemaDataModelOption.EntityDataModel) | ||
|  |                 OtherContent.Add(Schema.SchemaSource); | ||
|  |         } | ||
|  | 
 | ||
|  |         #endregion | ||
|  | 
 | ||
|  |         #region Properties, Methods, Events & Delegates | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         ///  | ||
|  |         /// </summary> | ||
|  |         SchemaElementLookUpTable<SchemaElement> Members | ||
|  |         { | ||
|  |             get | ||
|  |             { | ||
|  |                 if (_members == null) | ||
|  |                 { | ||
|  |                     _members = new SchemaElementLookUpTable<SchemaElement>(); | ||
|  |                 } | ||
|  |                 return _members; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         ///  | ||
|  |         /// </summary> | ||
|  |         public ISchemaElementLookUpTable<EntityContainerEntitySet> EntitySets | ||
|  |         { | ||
|  |             get | ||
|  |             { | ||
|  |                 if (_entitySets == null) | ||
|  |                 { | ||
|  |                     _entitySets = new FilteredSchemaElementLookUpTable<EntityContainerEntitySet, SchemaElement>(Members); | ||
|  |                 } | ||
|  |                 return _entitySets; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         ///  | ||
|  |         /// </summary> | ||
|  |         public ISchemaElementLookUpTable<EntityContainerRelationshipSet> RelationshipSets | ||
|  |         { | ||
|  |             get | ||
|  |             { | ||
|  |                 if (_relationshipSets == null) | ||
|  |                 { | ||
|  |                     _relationshipSets = new FilteredSchemaElementLookUpTable<EntityContainerRelationshipSet, SchemaElement>(Members); | ||
|  |                 } | ||
|  |                 return _relationshipSets; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         ///  | ||
|  |         /// </summary> | ||
|  |         public ISchemaElementLookUpTable<Function> FunctionImports | ||
|  |         { | ||
|  |             get | ||
|  |             { | ||
|  |                 if (_functionImports == null) | ||
|  |                 { | ||
|  |                     _functionImports = new FilteredSchemaElementLookUpTable<Function, SchemaElement>(Members); | ||
|  |                 } | ||
|  |                 return _functionImports; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         ///  | ||
|  |         /// </summary> | ||
|  |         public EntityContainer ExtendingEntityContainer | ||
|  |         { | ||
|  |             get | ||
|  |             { | ||
|  |                 return _entityContainerGettingExtended; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         protected override bool HandleAttribute(XmlReader reader) | ||
|  |         { | ||
|  |             if (base.HandleAttribute(reader)) | ||
|  |             { | ||
|  |                 return true; | ||
|  |             } | ||
|  |             else if (CanHandleAttribute(reader, XmlConstants.Extends)) | ||
|  |             { | ||
|  |                 HandleExtendsAttribute(reader); | ||
|  |                 return true; | ||
|  |             } | ||
|  | 
 | ||
|  |             return false; | ||
|  |         } | ||
|  | 
 | ||
|  |         protected override bool HandleElement(XmlReader reader) | ||
|  |         { | ||
|  |             if (base.HandleElement(reader)) | ||
|  |             { | ||
|  |                 return true; | ||
|  |             } | ||
|  |             else if (CanHandleElement(reader, XmlConstants.EntitySet)) | ||
|  |             { | ||
|  |                 HandleEntitySetElement(reader); | ||
|  |                 return true; | ||
|  |             } | ||
|  |             else if (CanHandleElement(reader, XmlConstants.AssociationSet)) | ||
|  |             { | ||
|  |                 HandleAssociationSetElement(reader); | ||
|  |                 return true; | ||
|  |             } | ||
|  |             else if (CanHandleElement(reader, XmlConstants.FunctionImport)) | ||
|  |             { | ||
|  |                 HandleFunctionImport(reader); | ||
|  |                 return true; | ||
|  |             } | ||
|  |             else if (Schema.DataModel == SchemaDataModelOption.EntityDataModel) | ||
|  |             { | ||
|  |                 if (CanHandleElement(reader, XmlConstants.ValueAnnotation)) | ||
|  |                 { | ||
|  |                     // EF does not support this EDM 3.0 element, so ignore it. | ||
|  |                     SkipElement(reader); | ||
|  |                     return true; | ||
|  |                 } | ||
|  |                 else if (CanHandleElement(reader, XmlConstants.TypeAnnotation)) | ||
|  |                 { | ||
|  |                     // EF does not support this EDM 3.0 element, so ignore it. | ||
|  |                     SkipElement(reader); | ||
|  |                     return true; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             return false; | ||
|  |         } | ||
|  | 
 | ||
|  |         private void HandleEntitySetElement(XmlReader reader) | ||
|  |         { | ||
|  |             Debug.Assert(reader != null); | ||
|  |             EntityContainerEntitySet set = new EntityContainerEntitySet(this); | ||
|  |             set.Parse(reader); | ||
|  |             Members.Add(set, true, Strings.DuplicateEntityContainerMemberName); | ||
|  |         } | ||
|  | 
 | ||
|  |         private void HandleAssociationSetElement(XmlReader reader) | ||
|  |         { | ||
|  |             Debug.Assert(reader != null); | ||
|  |             EntityContainerAssociationSet set = new EntityContainerAssociationSet(this); | ||
|  |             set.Parse(reader); | ||
|  |             Members.Add(set, true, Strings.DuplicateEntityContainerMemberName); | ||
|  |         } | ||
|  | 
 | ||
|  |         private void HandleFunctionImport(XmlReader reader) | ||
|  |         { | ||
|  |             Debug.Assert(null != reader); | ||
|  |             FunctionImportElement functionImport = new FunctionImportElement(this); | ||
|  |             functionImport.Parse(reader); | ||
|  |             Members.Add(functionImport, true, Strings.DuplicateEntityContainerMemberName); | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         ///  | ||
|  |         /// </summary> | ||
|  |         /// <param name="reader"></param> | ||
|  |         private void HandleExtendsAttribute(XmlReader reader) | ||
|  |         { | ||
|  |             _unresolvedExtendedEntityContainerName = HandleUndottedNameAttribute(reader, _unresolvedExtendedEntityContainerName); | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// Resolves the names to element references. | ||
|  |         /// </summary> | ||
|  |         internal override void ResolveTopLevelNames() | ||
|  |         { | ||
|  |             if (!_isAlreadyResolved) | ||
|  |             { | ||
|  |                 base.ResolveTopLevelNames(); | ||
|  | 
 | ||
|  |                 SchemaType extendingEntityContainer; | ||
|  |                 // If this entity container extends some other entity container, we should validate the entity container name. | ||
|  |                 if (!String.IsNullOrEmpty(_unresolvedExtendedEntityContainerName)) | ||
|  |                 { | ||
|  |                     if (_unresolvedExtendedEntityContainerName == this.Name) | ||
|  |                     { | ||
|  |                         AddError(ErrorCode.EntityContainerCannotExtendItself, EdmSchemaErrorSeverity.Error, | ||
|  |                             System.Data.Entity.Strings.EntityContainerCannotExtendItself(this.Name)); | ||
|  |                     } | ||
|  |                     else if (!Schema.SchemaManager.TryResolveType(null, _unresolvedExtendedEntityContainerName, out extendingEntityContainer)) | ||
|  |                     { | ||
|  |                         AddError(ErrorCode.InvalidEntityContainerNameInExtends, EdmSchemaErrorSeverity.Error, | ||
|  |                                 System.Data.Entity.Strings.InvalidEntityContainerNameInExtends(_unresolvedExtendedEntityContainerName)); | ||
|  |                     } | ||
|  |                     else | ||
|  |                     { | ||
|  |                         _entityContainerGettingExtended = (EntityContainer)extendingEntityContainer; | ||
|  | 
 | ||
|  |                         // Once you have successfully resolved the entity container, then you should call ResolveNames on the | ||
|  |                         // extending entity containers as well. This is because we will need to look up the chain for resolving | ||
|  |                         // entity set names, since there might be association sets/ function imports that refer to entity sets | ||
|  |                         // belonging in extended entity containers | ||
|  |                         _entityContainerGettingExtended.ResolveTopLevelNames(); | ||
|  |                     } | ||
|  |                 } | ||
|  | 
 | ||
|  |                 foreach (SchemaElement element in Members) | ||
|  |                 { | ||
|  |                     element.ResolveTopLevelNames(); | ||
|  |                 } | ||
|  | 
 | ||
|  |                 _isAlreadyResolved = true; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         internal override void ResolveSecondLevelNames() | ||
|  |         { | ||
|  |             base.ResolveSecondLevelNames(); | ||
|  | 
 | ||
|  |             foreach (SchemaElement element in Members) | ||
|  |             { | ||
|  |                 element.ResolveSecondLevelNames(); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// Do all validation for this element here, and delegate to all sub elements | ||
|  |         /// </summary> | ||
|  |         internal override void Validate() | ||
|  |         { | ||
|  |             // Now before we clone all the entity sets from the entity container that this entity container is extending, | ||
|  |             // we need to make sure that the entity container that is getting extended is already validated. since it might | ||
|  |             // be extending some other entity container, and we might want to populate this entity container, before | ||
|  |             // it gets extended | ||
|  |             if (!_isAlreadyValidated) | ||
|  |             { | ||
|  |                 base.Validate(); | ||
|  | 
 | ||
|  |                 // If this entity container extends some other entity container, then we should add all the  | ||
|  |                 // sets and function imports from that entity container to this entity container | ||
|  |                 if (this.ExtendingEntityContainer != null) | ||
|  |                 { | ||
|  |                     // Call Validate on the entity container that is getting extended, so that its entity set | ||
|  |                     // is populated | ||
|  |                     this.ExtendingEntityContainer.Validate(); | ||
|  | 
 | ||
|  |                     foreach (SchemaElement element in this.ExtendingEntityContainer.Members) | ||
|  |                     { | ||
|  |                         AddErrorKind error = this.Members.TryAdd(element.Clone(this)); | ||
|  |                         DuplicateOrEquivalentMemberNameWhileExtendingEntityContainer(element, error); | ||
|  |                     } | ||
|  |                 } | ||
|  | 
 | ||
|  |                 HashSet<string> tableKeys = new HashSet<string>(); | ||
|  |                  | ||
|  |                 foreach (SchemaElement element in Members) | ||
|  |                 { | ||
|  |                     EntityContainerEntitySet entitySet = element as EntityContainerEntitySet; | ||
|  |                     if (entitySet != null && Schema.DataModel == SchemaDataModelOption.ProviderDataModel) | ||
|  |                     { | ||
|  |                         CheckForDuplicateTableMapping(tableKeys, entitySet); | ||
|  |                     } | ||
|  |                     element.Validate(); | ||
|  |                 } | ||
|  | 
 | ||
|  |                 ValidateRelationshipSetHaveUniqueEnds(); | ||
|  | 
 | ||
|  |                 ValidateOnlyBaseEntitySetTypeDefinesConcurrency(); | ||
|  | 
 | ||
|  |                 // Set isAlreadyValidated to true | ||
|  |                 _isAlreadyValidated = true; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// Find the EntityContainerEntitySet in the same EntityContainer with the name from the extent  | ||
|  |         /// attribute | ||
|  |         /// </summary> | ||
|  |         /// <param name="name">the name of the EntityContainerProperty to find</param> | ||
|  |         /// <returns>The EntityContainerProperty it found or null if it fails to find it</returns> | ||
|  |         internal EntityContainerEntitySet FindEntitySet(string name) | ||
|  |         { | ||
|  |             EntityContainer current = this; | ||
|  |             while(current != null) | ||
|  |             { | ||
|  |                 foreach (EntityContainerEntitySet set in current.EntitySets) | ||
|  |                 { | ||
|  |                     if (Utils.CompareNames(set.Name, name) == 0) | ||
|  |                     { | ||
|  |                         return set; | ||
|  |                     } | ||
|  |                 } | ||
|  | 
 | ||
|  |                 current = current.ExtendingEntityContainer; | ||
|  |             } | ||
|  | 
 | ||
|  |             return null; | ||
|  |         } | ||
|  | 
 | ||
|  |         private void DuplicateOrEquivalentMemberNameWhileExtendingEntityContainer(SchemaElement schemaElement, | ||
|  |             AddErrorKind error) | ||
|  |         { | ||
|  |             Debug.Assert(error != AddErrorKind.MissingNameError, "Since entity container members are already resolved, name must never be empty"); | ||
|  |             Debug.Assert(this.ExtendingEntityContainer != null, "ExtendingEntityContainer must not be null"); | ||
|  | 
 | ||
|  |             if (error != AddErrorKind.Succeeded) | ||
|  |             { | ||
|  |                 Debug.Assert(error == AddErrorKind.DuplicateNameError, "Error must be duplicate name error"); | ||
|  |                 schemaElement.AddError(ErrorCode.AlreadyDefined, EdmSchemaErrorSeverity.Error, | ||
|  |                             System.Data.Entity.Strings.DuplicateMemberNameInExtendedEntityContainer( | ||
|  |                                 schemaElement.Name, ExtendingEntityContainer.Name, this.Name)); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         private void ValidateOnlyBaseEntitySetTypeDefinesConcurrency() | ||
|  |         { | ||
|  |             // collect all the base entitySet types | ||
|  |             Dictionary<SchemaEntityType, EntityContainerEntitySet> baseEntitySetTypes = new Dictionary<SchemaEntityType, EntityContainerEntitySet>(); | ||
|  |             foreach (SchemaElement element in Members) | ||
|  |             { | ||
|  |                 EntityContainerEntitySet entitySet = element as EntityContainerEntitySet; | ||
|  |                 if (entitySet != null && !baseEntitySetTypes.ContainsKey(entitySet.EntityType)) | ||
|  |                 { | ||
|  |                     baseEntitySetTypes.Add(entitySet.EntityType, entitySet); | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             // look through each type in this schema and see if it is derived from a base | ||
|  |             // type if it is then see if it has some "new" Concurrency fields | ||
|  |             foreach (SchemaType type in Schema.SchemaTypes) | ||
|  |             { | ||
|  |                 SchemaEntityType itemType = type as SchemaEntityType; | ||
|  |                 if (itemType != null) | ||
|  |                 { | ||
|  |                     EntityContainerEntitySet set; | ||
|  |                     if (TypeIsSubTypeOf(itemType, baseEntitySetTypes, out set) && | ||
|  |                        TypeDefinesNewConcurrencyProperties(itemType)) | ||
|  |                     { | ||
|  |                         AddError(ErrorCode.ConcurrencyRedefinedOnSubTypeOfEntitySetType, | ||
|  |                             EdmSchemaErrorSeverity.Error, | ||
|  |                             System.Data.Entity.Strings.ConcurrencyRedefinedOnSubTypeOfEntitySetType(itemType.FQName, set.EntityType.FQName, set.FQName)); | ||
|  |                     } | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// Validates that if there are more than one relationship set referring to the same type, each role of the relationship type | ||
|  |         /// never refers to the same entity set | ||
|  |         /// </summary> | ||
|  |         private void ValidateRelationshipSetHaveUniqueEnds() | ||
|  |         { | ||
|  |             // Contains the list of ends that have been visited and validated | ||
|  |             List<EntityContainerRelationshipSetEnd> alreadyValidatedEnds = new List<EntityContainerRelationshipSetEnd>(); | ||
|  |             bool error = true; | ||
|  | 
 | ||
|  |             foreach (EntityContainerRelationshipSet currentSet in this.RelationshipSets) | ||
|  |             { | ||
|  |                 foreach (EntityContainerRelationshipSetEnd currentSetEnd in currentSet.Ends) | ||
|  |                 { | ||
|  |                     error = false; | ||
|  |                     foreach (EntityContainerRelationshipSetEnd alreadyValidatedEnd in alreadyValidatedEnds) | ||
|  |                     { | ||
|  |                         if (AreRelationshipEndsEqual(alreadyValidatedEnd, currentSetEnd)) | ||
|  |                         { | ||
|  |                             AddError(ErrorCode.SimilarRelationshipEnd, | ||
|  |                                      EdmSchemaErrorSeverity.Error, | ||
|  |                                      System.Data.Entity.Strings.SimilarRelationshipEnd(alreadyValidatedEnd.Name, alreadyValidatedEnd.ParentElement.Name, | ||
|  |                                                          currentSetEnd.ParentElement.Name, alreadyValidatedEnd.EntitySet.Name, this.FQName)); | ||
|  |                             error = true; | ||
|  |                             break; | ||
|  |                         } | ||
|  |                     } | ||
|  |                     if (!error) | ||
|  |                     { | ||
|  |                         alreadyValidatedEnds.Add(currentSetEnd); | ||
|  |                     } | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         private static bool TypeIsSubTypeOf(SchemaEntityType itemType, Dictionary<SchemaEntityType, EntityContainerEntitySet> baseEntitySetTypes, out EntityContainerEntitySet set) | ||
|  |         { | ||
|  |             if (itemType.IsTypeHierarchyRoot) | ||
|  |             { | ||
|  |                 // can't be a sub type if we are a base type | ||
|  |                 set = null; | ||
|  |                 return false; | ||
|  |             } | ||
|  | 
 | ||
|  |             // walk up the hierarchy looking for a base that is the base type of an entityset | ||
|  |             for (SchemaEntityType baseType = itemType.BaseType as SchemaEntityType; baseType != null; baseType = baseType.BaseType as SchemaEntityType) | ||
|  |             { | ||
|  |                 if (baseEntitySetTypes.ContainsKey(baseType)) | ||
|  |                 { | ||
|  |                     set = baseEntitySetTypes[baseType]; | ||
|  |                     return true; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             set = null; | ||
|  |             return false; | ||
|  |         } | ||
|  | 
 | ||
|  |         private static bool TypeDefinesNewConcurrencyProperties(SchemaEntityType itemType) | ||
|  |         { | ||
|  |             foreach (StructuredProperty property in itemType.Properties) | ||
|  |             { | ||
|  |                 if (property.Type is ScalarType && MetadataHelper.GetConcurrencyMode(property.TypeUsage) != ConcurrencyMode.None) | ||
|  |                 { | ||
|  |                     return true; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             return false; | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// Return the fully qualified name for entity container. Since EntityContainer no longer lives in a schema, | ||
|  |         /// the FQName should be same as that of the Name | ||
|  |         /// </summary> | ||
|  |         public override string FQName | ||
|  |         { | ||
|  |             get | ||
|  |             { | ||
|  |                 return this.Name; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         ///  | ||
|  |         /// </summary> | ||
|  |         public override string Identity | ||
|  |         { | ||
|  |             get | ||
|  |             { | ||
|  |                 return Name; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// Adds a child EntitySet's tableKey (Schema/Table combination) to the validation collection | ||
|  |         /// This is used to validate that no child EntitySets share a Schema.Table combination | ||
|  |         /// </summary> | ||
|  |         private void CheckForDuplicateTableMapping(HashSet<string> tableKeys, EntityContainerEntitySet entitySet) | ||
|  |         { | ||
|  |             string schema; | ||
|  |             string table; | ||
|  | 
 | ||
|  |             if (String.IsNullOrEmpty(entitySet.DbSchema)) | ||
|  |             { | ||
|  |                 // if there is no specified DbSchema, use the parent EntityContainer's name | ||
|  |                 schema = this.Name; | ||
|  |             } | ||
|  |             else | ||
|  |             { | ||
|  |                 schema = entitySet.DbSchema; | ||
|  |             } | ||
|  | 
 | ||
|  | 
 | ||
|  |             if (String.IsNullOrEmpty(entitySet.Table)) | ||
|  |             { | ||
|  |                 // if there is no specified Table, use the EntitySet's name | ||
|  |                 table = entitySet.Name; | ||
|  |             } | ||
|  |             else | ||
|  |             { | ||
|  |                 table = entitySet.Table; | ||
|  |             } | ||
|  | 
 | ||
|  |             // create a key using the DbSchema and Table | ||
|  |             string tableKey = String.Format(System.Globalization.CultureInfo.InvariantCulture, "{0}.{1}", schema, table); | ||
|  |             if (entitySet.DefiningQuery != null) | ||
|  |             { | ||
|  |                 // don't consider the schema name for defining queries, because | ||
|  |                 // we can't say for sure that it is the same as the entity container | ||
|  |                 // so in this example | ||
|  |                 // | ||
|  |                 // <EntityContainer Name="dbo"> | ||
|  |                 //   <EntitySet Name="ByVal"> | ||
|  |                 //     <DefiningQuery>Select col1 from dbi.ByVal</DefiningQuery> | ||
|  |                 //   </EntitySet> | ||
|  |                 //   <EntitySet Name="ByVal1" Table="ByVal"/> | ||
|  |                 //   ... | ||
|  |                 // | ||
|  |                 // ByVal and ByVal1 should not conflict in our check | ||
|  |                 tableKey = entitySet.Name; | ||
|  |             } | ||
|  | 
 | ||
|  |             bool alreadyExisted = !tableKeys.Add(tableKey); | ||
|  |             if (alreadyExisted) | ||
|  |             { | ||
|  |                 entitySet.AddError(ErrorCode.AlreadyDefined, EdmSchemaErrorSeverity.Error, System.Data.Entity.Strings.DuplicateEntitySetTable(entitySet.Name, schema, table)); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// Returns true if the given two ends are similar - the relationship type that this ends belongs to is the same | ||
|  |         /// and the entity set refered by the ends are same and they have the same role name | ||
|  |         /// </summary> | ||
|  |         /// <param name="left"></param> | ||
|  |         /// <param name="right"></param> | ||
|  |         /// <returns></returns> | ||
|  |         private static bool AreRelationshipEndsEqual(EntityContainerRelationshipSetEnd left, EntityContainerRelationshipSetEnd right) | ||
|  |         { | ||
|  |             Debug.Assert(left.ParentElement.ParentElement == right.ParentElement.ParentElement, "both end should belong to the same entity container"); | ||
|  | 
 | ||
|  |             if (object.ReferenceEquals(left.EntitySet, right.EntitySet) && | ||
|  |                 object.ReferenceEquals(left.ParentElement.Relationship, right.ParentElement.Relationship) && | ||
|  |                 left.Name == right.Name) | ||
|  |             { | ||
|  |                 return true; | ||
|  |             } | ||
|  | 
 | ||
|  |             return false; | ||
|  |         } | ||
|  |         #endregion | ||
|  |     } | ||
|  | 
 | ||
|  | } |