//--------------------------------------------------------------------- // // Copyright (c) Microsoft Corporation. All rights reserved. // // // @owner Microsoft // @backupOwner Microsoft //--------------------------------------------------------------------- using System; using System.Collections.Generic; using System.Text; using System.Diagnostics; using System.Threading; using System.Data.Common.Utils; using System.Collections.ObjectModel; namespace System.Data.Metadata.Edm { /// /// concrete Class for representing a entity set /// public class EntitySet : EntitySetBase { #region Constructors /// /// The constructor for constructing the EntitySet with a given name and an entity type /// /// The name of the EntitySet /// The db schema /// The db table /// The provider specific query that should be used to retrieve the EntitySet /// The entity type of the entities that this entity set type contains /// Thrown if the argument name or entityType is null internal EntitySet(string name, string schema, string table, string definingQuery, EntityType entityType) : base(name, schema, table, definingQuery, entityType) { } #endregion #region Fields private ReadOnlyCollection> _foreignKeyDependents; private ReadOnlyCollection> _foreignKeyPrincipals; private volatile bool _hasForeignKeyRelationships; private volatile bool _hasIndependentRelationships; #endregion #region Properties /// /// Returns the kind of the type /// public override BuiltInTypeKind BuiltInTypeKind { get { return BuiltInTypeKind.EntitySet; } } /// /// Gets/Sets the entity type of this entity set /// public new EntityType ElementType { get { return (EntityType)base.ElementType; } } /// /// Returns the associations and constraints where "this" EntitySet particpates as the Principal end. /// From the results of this list, you can retrieve the Dependent IRelatedEnds /// internal ReadOnlyCollection> ForeignKeyDependents { get { if (_foreignKeyDependents == null) { InitializeForeignKeyLists(); } return _foreignKeyDependents; } } /// /// Returns the associations and constraints where "this" EntitySet particpates as the Dependent end. /// From the results of this list, you can retrieve the Principal IRelatedEnds /// internal ReadOnlyCollection> ForeignKeyPrincipals { get { if (_foreignKeyPrincipals == null) { InitializeForeignKeyLists(); } return _foreignKeyPrincipals; } } /// /// True if this entity set participates in any foreign key relationships, otherwise false. /// internal bool HasForeignKeyRelationships { get { if (_foreignKeyPrincipals == null) { InitializeForeignKeyLists(); } return _hasForeignKeyRelationships; } } /// /// True if this entity set participates in any independent relationships, otherwise false. /// internal bool HasIndependentRelationships { get { if (_foreignKeyPrincipals == null) { InitializeForeignKeyLists(); } return _hasIndependentRelationships; } } #endregion #region Methods private void InitializeForeignKeyLists() { var dependents = new List>(); var principals = new List>(); bool foundFkRelationship = false; bool foundIndependentRelationship = false; foreach (AssociationSet associationSet in MetadataHelper.GetAssociationsForEntitySet(this)) { if (associationSet.ElementType.IsForeignKey) { foundFkRelationship = true; Debug.Assert(associationSet.ElementType.ReferentialConstraints.Count == 1, "Expected exactly one constraint for FK"); ReferentialConstraint constraint = associationSet.ElementType.ReferentialConstraints[0]; if (constraint.ToRole.GetEntityType().IsAssignableFrom(this.ElementType) || this.ElementType.IsAssignableFrom(constraint.ToRole.GetEntityType())) { // Dependents dependents.Add(new Tuple(associationSet, constraint)); } if (constraint.FromRole.GetEntityType().IsAssignableFrom(this.ElementType) || this.ElementType.IsAssignableFrom(constraint.FromRole.GetEntityType())) { // Principals principals.Add(new Tuple(associationSet, constraint)); } } else { foundIndependentRelationship = true; } } _hasForeignKeyRelationships = foundFkRelationship; _hasIndependentRelationships = foundIndependentRelationship; var readOnlyDependents = dependents.AsReadOnly(); var readOnlyPrincipals = principals.AsReadOnly(); Interlocked.CompareExchange(ref _foreignKeyDependents, readOnlyDependents, null); Interlocked.CompareExchange(ref _foreignKeyPrincipals, readOnlyPrincipals, null); } #endregion } }