//--------------------------------------------------------------------- // // Copyright (c) Microsoft Corporation. All rights reserved. // // // @owner [....] // @backupOwner [....] //--------------------------------------------------------------------- using System; using System.Collections.Generic; using System.Collections.ObjectModel; using System.Linq; using System.Text; using System.Xml; using System.Data.Metadata.Edm; using System.Diagnostics; using System.Data.Mapping.ViewGeneration; using System.Data.Common.Utils; using System.Data.Mapping.ViewGeneration.Structures; using System.Data.Mapping.ViewGeneration.Validation; namespace System.Data.Mapping { using CellGroup = Set; /// /// Represents the Mapping metadata for the EntityContainer map in CS space. /// Only one EntityContainerMapping element is allowed in the MSL file for CS mapping. /// /// For Example if conceptually you could represent the CS MSL file as following /// ---Mapping /// --EntityContainerMapping ( CNorthwind-->SNorthwind ) /// --EntitySetMapping /// --AssociationSetMapping /// The type represents the metadata for EntityContainerMapping element in the above example. /// The SetMapping elements that are children of the EntityContainerMapping element /// can be accessed through the properties on this type. /// /// /// We currently assume that an Entity Container on the C side /// is mapped to a single Entity Container in the S - space. /// /// internal class StorageEntityContainerMapping : Map { #region Constructors /// /// Construct a new EntityContainer mapping object /// passing in the C-space EntityContainer and /// the s-space Entity container metadata objects. /// /// Entity Continer type that is being mapped on the C-side /// Entity Continer type that is being mapped on the S-side internal StorageEntityContainerMapping(EntityContainer entityContainer, EntityContainer storageEntityContainer, StorageMappingItemCollection storageMappingItemCollection, bool validate, bool generateUpdateViews) { this.m_entityContainer = entityContainer; this.m_storageEntityContainer = storageEntityContainer; this.m_storageMappingItemCollection = storageMappingItemCollection; this.m_memoizedCellGroupEvaluator = new Memoizer(ComputeCellGroups, new InputForComputingCellGroups()); this.identity = entityContainer.Identity; this.m_validate = validate; this.m_generateUpdateViews = generateUpdateViews; } #endregion #region Fields private string identity; private bool m_validate; private bool m_generateUpdateViews; private EntityContainer m_entityContainer; //Entity Continer type that is being mapped on the C-side private EntityContainer m_storageEntityContainer; //Entity Continer type that the C-space container is being mapped to private Dictionary m_entitySetMappings = new Dictionary(StringComparer.Ordinal); //A collection of EntitySetMappings under this EntityContainer mapping private Dictionary m_associationSetMappings = new Dictionary(StringComparer.Ordinal); //A collection of AssociationSetMappings under this EntityContainer mapping private Dictionary m_functionImportMappings = new Dictionary(); private string m_sourceLocation; //Schema URI for the mapping private int m_startLineNumber; //Line Number for EntityContainer Mapping element start tag private int m_startLinePosition; //Line position for EntityContainer Mapping element start tag private readonly StorageMappingItemCollection m_storageMappingItemCollection; private readonly Memoizer m_memoizedCellGroupEvaluator; #endregion #region Properties public StorageMappingItemCollection StorageMappingItemCollection { get { return m_storageMappingItemCollection; } } /// /// Gets the type kind for this item /// public override BuiltInTypeKind BuiltInTypeKind { get { return BuiltInTypeKind.MetadataItem; } } /// /// The Entity Container Metadata object on the C-side /// for which the mapping is being represented. /// internal override MetadataItem EdmItem { get { return this.m_entityContainer; } } internal override string Identity { get { return identity; } } /// /// Indicates whether there are no Set mappings /// in the container mapping. /// internal bool IsEmpty { get { return ((m_entitySetMappings.Count == 0) && (m_associationSetMappings.Count == 0)); } } /// /// Determine whether the container includes any views. /// Returns true if there is at least one query or update view specified by the mapping. /// internal bool HasViews { get { return HasMappingFragments() || AllSetMaps.Any((StorageSetMapping setMap) => setMap.QueryView != null); } } internal string SourceLocation { get { return m_sourceLocation; } set { m_sourceLocation = value; } } /// /// The Entity Container Metadata object on the C-side /// for which the mapping is being represented. /// internal EntityContainer EdmEntityContainer { get { return this.m_entityContainer; } } /// /// The Entity Container Metadata object on the C-side /// for which the mapping is being represented. /// internal EntityContainer StorageEntityContainer { get { return this.m_storageEntityContainer; } } /// /// a list of all the entity set maps under this /// container. In CS mapping, the mapping is done /// at the extent level as opposed to the type level. /// internal ReadOnlyCollection EntitySetMaps { get { return new List(this.m_entitySetMappings.Values).AsReadOnly(); } } /// /// a list of all the entity set maps under this /// container. In CS mapping, the mapping is done /// at the extent level as opposed to the type level. /// RelationshipSetMaps will be CompositionSetMaps and /// AssociationSetMaps put together. /// /// /// The reason we have RelationshipSetMaps is to be consistent with CDM metadata /// which treats both associations and compositions as Relationships. /// internal ReadOnlyCollection RelationshipSetMaps { get { return new List(this.m_associationSetMappings.Values).AsReadOnly(); } } /// /// a list of all the set maps under this /// container. /// internal IEnumerable AllSetMaps { get { return System.Linq.Enumerable.Concat(this.m_entitySetMappings.Values, this.m_associationSetMappings.Values); } } /// /// Line Number in MSL file where the EntityContainer Mapping Element's Start Tag is present. /// internal int StartLineNumber { get { return m_startLineNumber; } set { m_startLineNumber = value; } } /// /// Line Position in MSL file where the EntityContainer Mapping Element's Start Tag is present. /// internal int StartLinePosition { get { return m_startLinePosition; } set { m_startLinePosition = value; } } /// /// Indicates whether to validate the mapping or not. /// internal bool Validate { get { return m_validate; } } /// /// Indicates whether to generate the update views or not. /// internal bool GenerateUpdateViews { get { return m_generateUpdateViews; } } #endregion #region Methods /// /// get an EntitySet mapping based upon the name of the entity set. /// /// /// the name of the entity set internal StorageSetMapping GetEntitySetMapping(String entitySetName) { EntityUtil.CheckArgumentNull(entitySetName, "entitySetName"); //Key for EntitySetMapping should be EntitySet name and Entoty type name StorageSetMapping setMapping = null; m_entitySetMappings.TryGetValue(entitySetName, out setMapping); return setMapping; } /// /// Get a RelationShip set mapping based upon the name of the relationship set /// /// the name of the relationship set /// the mapping for the entity set if it exists, null if it does not exist internal StorageSetMapping GetRelationshipSetMapping(string relationshipSetName) { EntityUtil.CheckArgumentNull(relationshipSetName, "relationshipSetName"); StorageSetMapping setMapping = null; m_associationSetMappings.TryGetValue(relationshipSetName, out setMapping); return setMapping; } /// /// Get a RelationShipSet mapping that has the passed in EntitySet as one of the ends and is mapped to the /// table. /// internal IEnumerable GetRelationshipSetMappingsFor(EntitySetBase edmEntitySet, EntitySetBase storeEntitySet ) { //First select the association set maps that are mapped to this table IEnumerable associationSetMappings = m_associationSetMappings.Values.Cast().Where(w => ((w.StoreEntitySet != null) && (w.StoreEntitySet == storeEntitySet))); //From this again filter the ones that have the specified EntitySet on atleast one end associationSetMappings = associationSetMappings.Where(associationSetMap => ((associationSetMap.Set as AssociationSet).AssociationSetEnds.Any(associationSetEnd => associationSetEnd.EntitySet == edmEntitySet))); return associationSetMappings; } /// /// Get a set mapping based upon the name of the set /// /// /// internal StorageSetMapping GetSetMapping(string setName) { StorageSetMapping setMap = GetEntitySetMapping(setName); if (setMap == null) { setMap = GetRelationshipSetMapping(setName); } return setMap; } /// /// Adds an entity set mapping to the list of EntitySetMaps /// under this entity container mapping. The method will be called /// by the Mapping loader. /// internal void AddEntitySetMapping(StorageSetMapping setMapping) { if (!this.m_entitySetMappings.ContainsKey(setMapping.Set.Name)) this.m_entitySetMappings.Add(setMapping.Set.Name, setMapping); } /// /// Adds a association set mapping to the list of AssociationSetMaps /// under this entity container mapping. The method will be called /// by the Mapping loader. /// internal void AddAssociationSetMapping(StorageSetMapping setMapping) { this.m_associationSetMappings.Add(setMapping.Set.Name, setMapping); } /// /// check whether the EntityContainerMapping contains /// the map for the given AssociationSet /// /// /// internal bool ContainsAssociationSetMapping(AssociationSet associationSet) { return this.m_associationSetMappings.ContainsKey(associationSet.Name); } /// /// Returns whether the Set Map for the given set has a query view or not /// /// /// internal bool HasQueryViewForSetMap(string setName) { StorageSetMapping set = GetSetMapping(setName); if (set != null) { return (set.QueryView != null); } return false; } internal bool HasMappingFragments() { foreach (var extentMap in this.AllSetMaps) { foreach (var typeMap in extentMap.TypeMappings) { if (typeMap.MappingFragments.Count > 0) { return true; } } } return false; } /// /// The method builds up the spaces required for pretty printing each /// part of the mapping. /// internal static string GetPrettyPrintString(ref int index) { string spaces = ""; spaces = spaces.PadLeft(index, ' '); Console.WriteLine(spaces + "|"); Console.WriteLine(spaces + "|"); index++; spaces = spaces.PadLeft(index, ' '); Console.Write(spaces + "-"); index++; spaces = spaces.PadLeft(index, ' '); Console.Write("-"); index++; spaces = spaces.PadLeft(index, ' '); return spaces; } /// /// This method is primarily for debugging purposes. /// Will be removed shortly. /// /// internal void Print(int index) { string spaces = ""; StringBuilder sb = new StringBuilder(); sb.Append(spaces); sb.Append("EntityContainerMapping"); sb.Append(" "); sb.Append("Name:"); sb.Append(this.m_entityContainer.Name); sb.Append(" "); Console.WriteLine(sb.ToString()); foreach (StorageSetMapping extentMapping in m_entitySetMappings.Values) { extentMapping.Print(index + 5); } foreach (StorageSetMapping extentMapping in m_associationSetMappings.Values) { extentMapping.Print(index + 5); } } // Methods to modify and access function imports, which association a "functionImport" declared // in the model entity container with a targetFunction declared in the target internal void AddFunctionImportMapping(EdmFunction functionImport, FunctionImportMapping mapping) { m_functionImportMappings.Add(functionImport, mapping); } internal bool TryGetFunctionImportMapping(EdmFunction functionImport, out FunctionImportMapping mapping) { return m_functionImportMappings.TryGetValue(functionImport, out mapping); } internal OutputFromComputeCellGroups GetCellgroups(InputForComputingCellGroups args) { Debug.Assert(object.ReferenceEquals(this, args.ContainerMapping)); return m_memoizedCellGroupEvaluator.Evaluate(args); } private OutputFromComputeCellGroups ComputeCellGroups(InputForComputingCellGroups args) { OutputFromComputeCellGroups result = new OutputFromComputeCellGroups(); result.Success = true; CellCreator cellCreator = new CellCreator(args.ContainerMapping); result.Cells = cellCreator.GenerateCells(args.Config); result.Identifiers = cellCreator.Identifiers; if (result.Cells.Count <= 0) { //When type-specific QVs are asked for but not defined in the MSL we should return without generating // Query pipeline will handle this appropriately by asking for UNION ALL view. result.Success = false; return result; } result.ForeignKeyConstraints = ForeignConstraint.GetForeignConstraints(args.ContainerMapping.StorageEntityContainer); // Go through each table and determine their foreign key constraints CellPartitioner partitioner = new CellPartitioner(result.Cells, result.ForeignKeyConstraints); List cellGroups = partitioner.GroupRelatedCells(); //Clone cell groups- i.e, List> - upto cell before storing it in the cache because viewgen modified the Cell structure result.CellGroups = cellGroups.Select(setOfcells => new CellGroup(setOfcells.Select(cell => new Cell(cell)))).ToList(); return result; } #endregion } internal struct InputForComputingCellGroups : IEquatable, IEqualityComparer { internal readonly StorageEntityContainerMapping ContainerMapping; internal readonly ConfigViewGenerator Config; internal InputForComputingCellGroups(StorageEntityContainerMapping containerMapping, ConfigViewGenerator config) { this.ContainerMapping = containerMapping; this.Config = config; } public bool Equals(InputForComputingCellGroups other) { // Isn't this funny? We are not using Memoizer for function memoization. Args Entity and Config don't matter! // If I were to compare Entity this would not use the cache for cases when I supply different entity set. However, // the cell groups belong to ALL entity sets. return (this.ContainerMapping.Equals(other.ContainerMapping) && this.Config.Equals(other.Config)); } public bool Equals(InputForComputingCellGroups one, InputForComputingCellGroups two) { if (object.ReferenceEquals(one, two)) { return true; } if (object.ReferenceEquals(one, null) || object.ReferenceEquals(two, null)) { return false; } return one.Equals(two); } public int GetHashCode(InputForComputingCellGroups value) { if (value == null) { return 0; } return value.GetHashCode(); } public override int GetHashCode() { return this.ContainerMapping.GetHashCode(); } public override bool Equals(object obj) { if (obj is InputForComputingCellGroups) { return Equals((InputForComputingCellGroups)obj); } else { return false; } } public static bool operator ==(InputForComputingCellGroups input1, InputForComputingCellGroups input2) { if (object.ReferenceEquals(input1, input2)) { return true; } return input1.Equals(input2); } public static bool operator !=(InputForComputingCellGroups input1, InputForComputingCellGroups input2) { return !(input1 == input2); } } internal struct OutputFromComputeCellGroups { internal List Cells; internal CqlIdentifiers Identifiers; internal List CellGroups; internal List ForeignKeyConstraints; internal bool Success; } }