2016-08-03 10:59:49 +00:00
//---------------------------------------------------------------------
// <copyright file="MetadataHelper.cs" company="Microsoft">
// Copyright (c) Microsoft Corporation. All rights reserved.
// </copyright>
//
2017-08-21 15:34:15 +00:00
// @owner Microsoft
// @backupOwner Microsoft
2016-08-03 10:59:49 +00:00
//---------------------------------------------------------------------
using System.Collections.Generic ;
using System.Data.Mapping ;
using System.Data.Metadata.Edm ;
using System.Data.Objects.ELinq ;
using System.Diagnostics ;
using System.Linq ;
using System.Security.Cryptography ;
namespace System.Data.Common.Utils
{
// Helper functions to get metadata information
internal static class MetadataHelper
{
/// <summary>
/// Returns an element type of the collection returned by the function import.
/// Returns false, if element type cannot be determined.
/// </summary>
internal static bool TryGetFunctionImportReturnType < T > ( EdmFunction functionImport , int resultSetIndex , out T returnType ) where T : EdmType
{
T resultType ;
if ( TryGetWrappedReturnEdmTypeFromFunctionImport < T > ( functionImport , resultSetIndex , out resultType ) )
{
if ( typeof ( EntityType ) . Equals ( typeof ( T ) ) & & resultType is EntityType
| | typeof ( ComplexType ) . Equals ( typeof ( T ) ) & & resultType is ComplexType
| | typeof ( StructuralType ) . Equals ( typeof ( T ) ) & & resultType is StructuralType
| | typeof ( EdmType ) . Equals ( typeof ( T ) ) & & resultType is EdmType )
{
returnType = resultType ;
return true ;
}
}
returnType = null ;
return false ;
}
private static bool TryGetWrappedReturnEdmTypeFromFunctionImport < T > ( EdmFunction functionImport , int resultSetIndex , out T resultType ) where T : EdmType
{
resultType = null ;
CollectionType collectionType ;
if ( TryGetFunctionImportReturnCollectionType ( functionImport , resultSetIndex , out collectionType ) )
{
resultType = collectionType . TypeUsage . EdmType as T ;
return true ;
}
return false ;
}
/// <summary>
/// effects: determines if the given function import returns collection type, and if so returns the type
/// </summary>
internal static bool TryGetFunctionImportReturnCollectionType ( EdmFunction functionImport , int resultSetIndex , out CollectionType collectionType )
{
FunctionParameter returnParameter = GetReturnParameter ( functionImport , resultSetIndex ) ;
if ( returnParameter ! = null
& & returnParameter . TypeUsage . EdmType . BuiltInTypeKind = = BuiltInTypeKind . CollectionType )
{
collectionType = ( CollectionType ) returnParameter . TypeUsage . EdmType ;
return true ;
}
collectionType = null ;
return false ;
}
/// <summary>
/// Gets the resultSetIndexth return parameter for functionImport, or null if resultSetIndex is out of range
/// </summary>
internal static FunctionParameter GetReturnParameter ( EdmFunction functionImport , int resultSetIndex )
{
return functionImport . ReturnParameters . Count > resultSetIndex
? functionImport . ReturnParameters [ resultSetIndex ]
: null ;
}
internal static EdmFunction GetFunctionImport (
string functionName , string defaultContainerName , MetadataWorkspace workspace ,
out string containerName , out string functionImportName )
{
// find FunctionImport
CommandHelper . ParseFunctionImportCommandText ( functionName , defaultContainerName ,
out containerName , out functionImportName ) ;
return CommandHelper . FindFunctionImport ( workspace , containerName , functionImportName ) ;
}
/// <summary>
/// Gets the resultSetIndexth result edm type, and ensure that it is consistent with EntityType.
/// </summary>
internal static EdmType GetAndCheckFunctionImportReturnType < TElement > ( EdmFunction functionImport , int resultSetIndex , MetadataWorkspace workspace )
{
EdmType expectedEdmType ;
if ( ! MetadataHelper . TryGetFunctionImportReturnType < EdmType > ( functionImport , resultSetIndex , out expectedEdmType ) )
{
throw EntityUtil . ExecuteFunctionCalledWithNonReaderFunction ( functionImport ) ;
}
CheckFunctionImportReturnType < TElement > ( expectedEdmType , workspace ) ;
return expectedEdmType ;
}
/// <summary>
/// check that the type TElement and function metadata are consistent
/// </summary>
internal static void CheckFunctionImportReturnType < TElement > ( EdmType expectedEdmType , MetadataWorkspace workspace )
{
// currently there are only two possible spatial O-space types, but 16 C-space types.
// Normalize the C-space type to the base type before we check to see if it matches the O-space type.
bool isGeographic ;
EdmType spatialNormalizedEdmType = expectedEdmType ;
if ( Helper . IsSpatialType ( expectedEdmType , out isGeographic ) )
{
spatialNormalizedEdmType = PrimitiveType . GetEdmPrimitiveType ( isGeographic ? PrimitiveTypeKind . Geography : PrimitiveTypeKind . Geometry ) ;
}
EdmType modelEdmType ;
if ( ! MetadataHelper . TryDetermineCSpaceModelType < TElement > ( workspace , out modelEdmType ) | |
! modelEdmType . EdmEquals ( spatialNormalizedEdmType ) )
{
throw EntityUtil . ExecuteFunctionTypeMismatch ( typeof ( TElement ) , expectedEdmType ) ;
}
}
// Returns ParameterDirection corresponding to given ParameterMode
internal static ParameterDirection ParameterModeToParameterDirection ( ParameterMode mode )
{
switch ( mode )
{
case ParameterMode . In :
return ParameterDirection . Input ;
case ParameterMode . InOut :
return ParameterDirection . InputOutput ;
case ParameterMode . Out :
return ParameterDirection . Output ;
case ParameterMode . ReturnValue :
return ParameterDirection . ReturnValue ;
default :
Debug . Fail ( "unrecognized mode " + mode . ToString ( ) ) ;
return default ( ParameterDirection ) ;
}
}
// requires: workspace
// Determines CSpace EntityType associated with the type argument T
internal static bool TryDetermineCSpaceModelType < T > ( MetadataWorkspace workspace , out EdmType modelEdmType )
{
return TryDetermineCSpaceModelType ( typeof ( T ) , workspace , out modelEdmType ) ;
}
internal static bool TryDetermineCSpaceModelType ( Type type , MetadataWorkspace workspace , out EdmType modelEdmType )
{
Debug . Assert ( null ! = workspace ) ;
Type nonNullabelType = TypeSystem . GetNonNullableType ( type ) ;
// make sure the workspace knows about T
workspace . ImplicitLoadAssemblyForType ( nonNullabelType , System . Reflection . Assembly . GetCallingAssembly ( ) ) ;
ObjectItemCollection objectItemCollection = ( ObjectItemCollection ) workspace . GetItemCollection ( DataSpace . OSpace ) ;
EdmType objectEdmType ;
if ( objectItemCollection . TryGetItem < EdmType > ( nonNullabelType . FullName , out objectEdmType ) )
{
Map map ;
if ( workspace . TryGetMap ( objectEdmType , DataSpace . OCSpace , out map ) )
{
ObjectTypeMapping objectMapping = ( ObjectTypeMapping ) map ;
modelEdmType = objectMapping . EdmType ;
return true ;
}
}
modelEdmType = null ;
return false ;
}
// effects: Returns true iff member is present in type.Members
internal static bool DoesMemberExist ( StructuralType type , EdmMember member )
{
foreach ( EdmMember child in type . Members )
{
if ( child . Equals ( member ) )
{
return true ;
}
}
return false ;
}
/// <summary>
/// Returns true iff member's is a simple non-structures scalar such as primitive or enum.
/// </summary>
internal static bool IsNonRefSimpleMember ( EdmMember member )
{
return member . TypeUsage . EdmType . BuiltInTypeKind = = BuiltInTypeKind . PrimitiveType | |
member . TypeUsage . EdmType . BuiltInTypeKind = = BuiltInTypeKind . EnumType ;
}
// effects: Returns true if member's type has a discrete domain (i.e. is bolean type)
// Note: enums don't have discrete domains as we allow domain of the underlying type.
internal static bool HasDiscreteDomain ( EdmType edmType )
{
var primitiveType = edmType as PrimitiveType ;
return primitiveType ! = null & & primitiveType . PrimitiveTypeKind = = PrimitiveTypeKind . Boolean ;
}
// requires: end is given
// effects: determine the entity type for an association end member
internal static EntityType GetEntityTypeForEnd ( AssociationEndMember end )
{
Debug . Assert ( null ! = end ) ;
Debug . Assert ( end . TypeUsage . EdmType . BuiltInTypeKind = = BuiltInTypeKind . RefType ,
"type of association end member must be ref" ) ;
RefType refType = ( RefType ) end . TypeUsage . EdmType ;
EntityTypeBase endType = refType . ElementType ;
Debug . Assert ( endType . BuiltInTypeKind = = BuiltInTypeKind . EntityType ,
"type of association end reference element must be entity type" ) ;
return ( EntityType ) endType ;
}
// effects: Returns the entity set at the end corresponding to endMember
internal static EntitySet GetEntitySetAtEnd ( AssociationSet associationSet ,
AssociationEndMember endMember )
{
return associationSet . AssociationSetEnds [ endMember . Name ] . EntitySet ;
}
// effects: Returns the AssociationEndMember at the other end of the parent association (first found)
internal static AssociationEndMember GetOtherAssociationEnd ( AssociationEndMember endMember )
{
ReadOnlyMetadataCollection < EdmMember > members = endMember . DeclaringType . Members ;
Debug . Assert ( members . Count = = 2 , "only expecting two end members" ) ;
EdmMember otherMember = members [ 0 ] ;
if ( ! Object . ReferenceEquals ( endMember , otherMember ) )
{
Debug . Assert ( Object . ReferenceEquals ( endMember , members [ 1 ] ) , "didn't match other member" ) ;
return ( AssociationEndMember ) otherMember ;
}
return ( AssociationEndMember ) members [ 1 ] ;
}
// effects: Returns true iff every end other than "endPropery" has a lower
// multiplicity of at least one
internal static bool IsEveryOtherEndAtLeastOne ( AssociationSet associationSet ,
AssociationEndMember member )
{
foreach ( AssociationSetEnd end in associationSet . AssociationSetEnds )
{
AssociationEndMember endMember = end . CorrespondingAssociationEndMember ;
if ( endMember . Equals ( member ) = = false & &
GetLowerBoundOfMultiplicity ( endMember . RelationshipMultiplicity ) = = 0 )
{
return false ;
}
}
return true ;
}
// requires: toEnd and type are given
// effects: determines whether the given association end can be referenced by an entity of the given type
internal static bool IsAssociationValidForEntityType ( AssociationSetEnd toEnd , EntityType type )
{
Debug . Assert ( null ! = toEnd ) ;
Debug . Assert ( null ! = type ) ;
// get the opposite end which includes the relevant type information
AssociationSetEnd fromEnd = GetOppositeEnd ( toEnd ) ;
EntityType fromType = GetEntityTypeForEnd ( fromEnd . CorrespondingAssociationEndMember ) ;
return ( fromType . IsAssignableFrom ( type ) ) ;
}
// requires: end is given
// effects: returns the opposite end in the association
internal static AssociationSetEnd GetOppositeEnd ( AssociationSetEnd end )
{
Debug . Assert ( null ! = end ) ;
// there must be exactly one ("Single") other end that isn't ("Filter") this end
AssociationSetEnd otherEnd = end . ParentAssociationSet . AssociationSetEnds . Where (
e = > ! e . EdmEquals ( end ) ) . Single ( ) ;
return otherEnd ;
}
// requires: function is not null
// effects: Returns true if the given function is composable.
internal static bool IsComposable ( EdmFunction function )
{
Debug . Assert ( function ! = null ) ;
MetadataProperty isComposableProperty ;
if ( function . MetadataProperties . TryGetValue ( "IsComposableAttribute" , false , out isComposableProperty ) )
{
return ( bool ) isComposableProperty . Value ;
} else
{
return ! function . IsFunctionImport ;
}
}
// requires: member is EdmProperty or AssociationEndMember
// effects: Returns true if member is nullable
internal static bool IsMemberNullable ( EdmMember member )
{
Debug . Assert ( member ! = null ) ;
Debug . Assert ( Helper . IsEdmProperty ( member ) | | Helper . IsAssociationEndMember ( member ) ) ;
if ( Helper . IsEdmProperty ( member ) )
{
return ( ( EdmProperty ) member ) . Nullable ;
}
return false ;
}
/// <summary>
/// Given a table EntitySet this function finds out all C-side EntitySets that are mapped to the table.
/// </summary>
internal static IEnumerable < EntitySet > GetInfluencingEntitySetsForTable ( EntitySet table , MetadataWorkspace workspace )
{
Debug . Assert ( table . EntityContainer . GetDataSpace ( ) = = DataSpace . SSpace ) ;
ItemCollection itemCollection = null ;
workspace . TryGetItemCollection ( DataSpace . CSSpace , out itemCollection ) ;
StorageEntityContainerMapping containerMapping = MappingMetadataHelper . GetEntityContainerMap ( ( StorageMappingItemCollection ) itemCollection , table . EntityContainer ) ;
//find EntitySetMappings where one of the mapping fragment maps some type to the given table
return containerMapping . EntitySetMaps
. Where (
map = > map . TypeMappings . Any (
typeMap = > typeMap . MappingFragments . Any (
mappingFrag = > mappingFrag . TableSet . EdmEquals ( table )
)
)
)
. Select ( m = > m . Set )
. Cast < EntitySet > ( )
. Distinct ( ) ;
}
// effects: Returns this type and its sub types - for refs, gets the
// type and subtypes of the entity type
internal static IEnumerable < EdmType > GetTypeAndSubtypesOf ( EdmType type , MetadataWorkspace workspace , bool includeAbstractTypes )
{
return GetTypeAndSubtypesOf ( type , workspace . GetItemCollection ( DataSpace . CSpace ) , includeAbstractTypes ) ;
}
internal static IEnumerable < EdmType > GetTypeAndSubtypesOf ( EdmType type , ItemCollection itemCollection , bool includeAbstractTypes )
{
// We have to collect subtypes in ref to support conditional association mappings
if ( Helper . IsRefType ( type ) )
{
type = ( ( RefType ) type ) . ElementType ;
}
if ( includeAbstractTypes | | ! type . Abstract )
{
yield return type ;
}
// Get entity sub-types
foreach ( EdmType subType in GetTypeAndSubtypesOf < EntityType > ( type , itemCollection , includeAbstractTypes ) )
{
yield return subType ;
}
// Get complex sub-types
foreach ( EdmType subType in GetTypeAndSubtypesOf < ComplexType > ( type , itemCollection , includeAbstractTypes ) )
{
yield return subType ;
}
}
private static IEnumerable < EdmType > GetTypeAndSubtypesOf < T_EdmType > ( EdmType type , ItemCollection itemCollection , bool includeAbstractTypes )
where T_EdmType : EdmType
{
// Get the subtypes of the type from the WorkSpace
T_EdmType specificType = type as T_EdmType ;
if ( specificType ! = null )
{
IEnumerable < T_EdmType > typesInWorkSpace = itemCollection . GetItems < T_EdmType > ( ) ;
foreach ( T_EdmType typeInWorkSpace in typesInWorkSpace )
{
if ( specificType . Equals ( typeInWorkSpace ) = = false & & Helper . IsSubtypeOf ( typeInWorkSpace , specificType ) )
{
if ( includeAbstractTypes | | ! typeInWorkSpace . Abstract )
{
yield return typeInWorkSpace ;
}
}
}
}
yield break ;
}
internal static IEnumerable < EdmType > GetTypeAndParentTypesOf ( EdmType type , ItemCollection itemCollection , bool includeAbstractTypes )
{
// We have to collect subtypes in ref to support conditional association mappings
if ( Helper . IsRefType ( type ) )
{
type = ( ( RefType ) type ) . ElementType ;
}
EdmType specificType = type ;
while ( specificType ! = null )
{
if ( includeAbstractTypes | | ! specificType . Abstract )
{
yield return specificType ;
}
specificType = specificType . BaseType as EntityType ; //The cast is guaranteed to work. See use of GetItems<T_EdmType> in GetTypesAndSubTypesOf()
}
}
/// <summary>
/// Builds an undirected graph (represented as a directional graph with reciprocal navigation edges) of the all the types in the workspace.
/// This is used to traverse inheritance hierarchy up and down.
/// O(n), where n=number of types
/// </summary>
/// <returns>A dictionary of type t -> set of types {s}, such that there is an edge between t and elem(s) iff t and s are related DIRECTLY via inheritance (child or parent type) </returns>
internal static Dictionary < EntityType , Set < EntityType > > BuildUndirectedGraphOfTypes ( EdmItemCollection edmItemCollection )
{
Dictionary < EntityType , Set < EntityType > > graph = new Dictionary < EntityType , Set < EntityType > > ( ) ;
IEnumerable < EntityType > typesInWorkSpace = edmItemCollection . GetItems < EntityType > ( ) ;
foreach ( EntityType childType in typesInWorkSpace )
{
if ( childType . BaseType = = null ) //root type
{
continue ;
}
EntityType parentType = childType . BaseType as EntityType ;
Debug . Assert ( parentType ! = null , "Parent type not Entity Type ??" ) ;
AddDirectedEdgeBetweenEntityTypes ( graph , childType , parentType ) ;
AddDirectedEdgeBetweenEntityTypes ( graph , parentType , childType ) ;
}
return graph ;
}
/// <summary>
/// is A parent of b?
/// </summary>
internal static bool IsParentOf ( EntityType a , EntityType b )
{
EntityType parent = b . BaseType as EntityType ;
while ( parent ! = null )
{
if ( parent . EdmEquals ( a ) )
{
return true ;
}
else
{
parent = parent . BaseType as EntityType ;
}
}
return false ;
}
/// <summary>
/// Add and Edge a --> b
/// Assumes edge does not exist
/// O(1)
/// </summary>
private static void AddDirectedEdgeBetweenEntityTypes ( Dictionary < EntityType , Set < EntityType > > graph , EntityType a , EntityType b )
{
Set < EntityType > references ;
if ( graph . ContainsKey ( a ) )
{
references = graph [ a ] ;
}
else
{
references = new Set < EntityType > ( ) ;
graph . Add ( a , references ) ;
}
Debug . Assert ( ! references . Contains ( b ) , "Dictionary already has a --> b reference" ) ;
references . Add ( b ) ;
}
/// <summary>
/// Checks wither the given AssociationEnd's keys are sufficient for identifying a unique tuple in the AssociationSet.
/// This is possible because refconstraints make certain Keys redundant. We subtract such redundant key sof "other" ends
/// and see if what is left is contributed only from the given end's keys.
/// </summary>
[System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Security", "CA2140:TransparentMethodsMustNotReferenceCriticalCode", Justification = "Based on Bug VSTS Pioneer #433188: IsVisibleOutsideAssembly is wrong on generic instantiations.")]
internal static bool DoesEndKeySubsumeAssociationSetKey ( AssociationSet assocSet , AssociationEndMember thisEnd , HashSet < Pair < EdmMember , EntityType > > associationkeys )
{
AssociationType assocType = assocSet . ElementType ;
EntityType thisEndsEntityType = ( EntityType ) ( ( RefType ) thisEnd . TypeUsage . EdmType ) . ElementType ;
HashSet < Pair < EdmMember , EntityType > > thisEndKeys = new HashSet < Pair < EdmMember , EntityType > > (
thisEndsEntityType . KeyMembers . Select ( edmMember = > new Pair < EdmMember , EntityType > ( edmMember , thisEndsEntityType ) ) ) ;
foreach ( ReferentialConstraint constraint in assocType . ReferentialConstraints )
{
IEnumerable < EdmMember > otherEndProperties ;
EntityType otherEndType ;
if ( thisEnd . Equals ( ( AssociationEndMember ) constraint . ToRole ) )
{
otherEndProperties = Helpers . AsSuperTypeList < EdmProperty , EdmMember > ( constraint . FromProperties ) ;
otherEndType = ( EntityType ) ( ( RefType ) ( ( AssociationEndMember ) constraint . FromRole ) . TypeUsage . EdmType ) . ElementType ;
}
else if ( thisEnd . Equals ( ( AssociationEndMember ) constraint . FromRole ) )
{
otherEndProperties = Helpers . AsSuperTypeList < EdmProperty , EdmMember > ( constraint . ToProperties ) ;
otherEndType = ( EntityType ) ( ( RefType ) ( ( AssociationEndMember ) constraint . ToRole ) . TypeUsage . EdmType ) . ElementType ;
}
else
{
//this end not part of the referential constraint
continue ;
}
//Essentially ref constraints is an equality condition, so remove redundant members from entity set key
foreach ( EdmMember member in otherEndProperties )
{
associationkeys . Remove ( new Pair < EdmMember , EntityType > ( member , otherEndType ) ) ;
}
}
//Now that all redundant members have been removed, is thisEnd the key of the entity set?
return associationkeys . IsSubsetOf ( thisEndKeys ) ;
}
// effects: Returns true if end forms a key in relationshipSet
internal static bool DoesEndFormKey ( AssociationSet associationSet , AssociationEndMember end )
{
// Look at all other ends. if their multiplicities are at most 1, return true
foreach ( AssociationEndMember endMember in associationSet . ElementType . Members )
{
if ( endMember . Equals ( end ) = = false & &
endMember . RelationshipMultiplicity = = RelationshipMultiplicity . Many ) // some other end has multiplicity 0..*
{
return false ;
}
}
return true ;
}
// effects: Returns true if extent is at one of the ends of relationshipSet
internal static bool IsExtentAtSomeRelationshipEnd ( AssociationSet relationshipSet , EntitySetBase extent )
{
if ( Helper . IsEntitySet ( extent ) )
{
return GetSomeEndForEntitySet ( relationshipSet , ( EntitySet ) extent ) ! = null ;
}
return false ;
}
// effects: Returns some end corresponding to entity set in
// association set. If no such end exists, return null
internal static AssociationEndMember GetSomeEndForEntitySet ( AssociationSet associationSet , EntitySetBase entitySet )
{
foreach ( AssociationSetEnd associationEnd in associationSet . AssociationSetEnds )
{
if ( associationEnd . EntitySet . Equals ( entitySet ) )
{
return associationEnd . CorrespondingAssociationEndMember ;
}
}
return null ;
}
// requires: entitySet1 and entitySet2 belong to the same container
// effects: Returns the associations that occur between entitySet1
// and entitySet2. If none is found, returns an empty set
internal static List < AssociationSet > GetAssociationsForEntitySets ( EntitySet entitySet1 , EntitySet entitySet2 )
{
Debug . Assert ( entitySet1 ! = null ) ;
Debug . Assert ( entitySet2 ! = null ) ;
Debug . Assert ( entitySet1 . EntityContainer = = entitySet2 . EntityContainer , "EntityContainer must be the same for both the entity sets" ) ;
List < AssociationSet > result = new List < AssociationSet > ( ) ;
foreach ( EntitySetBase extent in entitySet1 . EntityContainer . BaseEntitySets )
{
if ( Helper . IsRelationshipSet ( extent ) )
{
AssociationSet assocSet = ( AssociationSet ) extent ;
if ( IsExtentAtSomeRelationshipEnd ( assocSet , entitySet1 ) & &
IsExtentAtSomeRelationshipEnd ( assocSet , entitySet2 ) )
{
result . Add ( assocSet ) ;
}
}
}
return result ;
}
// requires: entitySet and associationType
// effects: Returns the associations that refer to associationType and refer to entitySet in one of its end.
// If none is found, returns an empty set
internal static AssociationSet GetAssociationsForEntitySetAndAssociationType ( EntityContainer entityContainer , string entitySetName ,
AssociationType associationType , string endName , out EntitySet entitySet )
{
Debug . Assert ( associationType . Members . Contains ( endName ) , "EndName should be a valid name" ) ;
entitySet = null ;
AssociationSet retValue = null ;
ReadOnlyMetadataCollection < EntitySetBase > baseEntitySets = entityContainer . BaseEntitySets ;
int count = baseEntitySets . Count ;
for ( int i = 0 ; i < count ; + + i )
{
EntitySetBase extent = baseEntitySets [ i ] ;
if ( Object . ReferenceEquals ( extent . ElementType , associationType ) )
{
AssociationSet assocSet = ( AssociationSet ) extent ;
EntitySet es = assocSet . AssociationSetEnds [ endName ] . EntitySet ;
if ( es . Name = = entitySetName )
{
Debug . Assert ( retValue = = null , "There should be only one AssociationSet, given an assocationtype, end name and entity set" ) ;
retValue = assocSet ;
entitySet = es ;
#if ! DEBUG
break ;
#endif
}
}
}
return retValue ;
}
// requires: entitySet
// effects: Returns the associations that occur between entitySet
// and other entitySets. If none is found, returns an empty set
internal static List < AssociationSet > GetAssociationsForEntitySet ( EntitySetBase entitySet )
{
Debug . Assert ( entitySet ! = null ) ;
List < AssociationSet > result = new List < AssociationSet > ( ) ;
foreach ( EntitySetBase extent in entitySet . EntityContainer . BaseEntitySets )
{
if ( Helper . IsRelationshipSet ( extent ) )
{
AssociationSet assocSet = ( AssociationSet ) extent ;
if ( IsExtentAtSomeRelationshipEnd ( assocSet , entitySet ) )
{
result . Add ( assocSet ) ;
}
}
}
return result ;
}
// effects: Returns true iff superType is an ancestor of subType in
// the type hierarchy or superType and subType are the same
internal static bool IsSuperTypeOf ( EdmType superType , EdmType subType )
{
EdmType currentType = subType ;
while ( currentType ! = null )
{
if ( currentType . Equals ( superType ) )
{
return true ;
}
currentType = currentType . BaseType ;
}
return false ;
}
// requires: typeUsage wraps a primitive type
internal static PrimitiveTypeKind GetPrimitiveTypeKind ( TypeUsage typeUsage )
{
Debug . Assert ( null ! = typeUsage & & null ! = typeUsage . EdmType & & typeUsage . EdmType . BuiltInTypeKind = = BuiltInTypeKind . PrimitiveType ) ;
PrimitiveType primitiveType = ( PrimitiveType ) typeUsage . EdmType ;
return primitiveType . PrimitiveTypeKind ;
}
// determines whether the given member is a key of an entity set
internal static bool IsPartOfEntityTypeKey ( EdmMember member )
{
if ( Helper . IsEntityType ( member . DeclaringType ) & &
Helper . IsEdmProperty ( member ) )
{
return ( ( EntityType ) member . DeclaringType ) . KeyMembers . Contains ( member ) ;
}
return false ;
}
// Given a type usage, returns the element type (unwraps collections)
internal static TypeUsage GetElementType ( TypeUsage typeUsage )
{
if ( BuiltInTypeKind . CollectionType = = typeUsage . EdmType . BuiltInTypeKind )
{
TypeUsage elementType = ( ( CollectionType ) typeUsage . EdmType ) . TypeUsage ;
// recursively unwrap
return GetElementType ( elementType ) ;
}
return typeUsage ;
}
internal static int GetLowerBoundOfMultiplicity ( RelationshipMultiplicity multiplicity )
{
if ( multiplicity = = RelationshipMultiplicity . Many | |
multiplicity = = RelationshipMultiplicity . ZeroOrOne )
{
return 0 ;
}
else
{
return 1 ;
}
}
internal static int? GetUpperBoundOfMultiplicity ( RelationshipMultiplicity multiplicity )
{
if ( multiplicity = = RelationshipMultiplicity . One | |
multiplicity = = RelationshipMultiplicity . ZeroOrOne )
{
return 1 ;
}
else
{
return null ;
}
}
// effects: Returns all the concurrency token members in superType and its subtypes
internal static Set < EdmMember > GetConcurrencyMembersForTypeHierarchy ( EntityTypeBase superType , EdmItemCollection edmItemCollection )
{
Set < EdmMember > result = new Set < EdmMember > ( ) ;
foreach ( StructuralType type in GetTypeAndSubtypesOf ( superType , edmItemCollection , true /*includeAbstractTypes */ ) )
{
// Go through all the members -- Can call Members instead of AllMembers since we are
// running through the whole hierarchy
foreach ( EdmMember member in type . Members )
{
// check for the concurrency facet
ConcurrencyMode concurrencyMode = GetConcurrencyMode ( member ) ;
if ( concurrencyMode = = ConcurrencyMode . Fixed )
{
result . Add ( member ) ;
}
}
}
return result ;
}
// Determines whether the given member is declared as a concurrency property
internal static ConcurrencyMode GetConcurrencyMode ( EdmMember member )
{
return GetConcurrencyMode ( member . TypeUsage ) ;
}
// Determines whether the given member is declared as a concurrency property
internal static ConcurrencyMode GetConcurrencyMode ( TypeUsage typeUsage )
{
Facet concurrencyFacet ;
if ( typeUsage . Facets . TryGetValue ( EdmProviderManifest . ConcurrencyModeFacetName , false , out concurrencyFacet ) & &
concurrencyFacet . Value ! = null )
{
ConcurrencyMode concurrencyMode = ( ConcurrencyMode ) concurrencyFacet . Value ;
return concurrencyMode ;
}
return ConcurrencyMode . None ;
}
// Determines the store generated pattern for this member
internal static StoreGeneratedPattern GetStoreGeneratedPattern ( EdmMember member )
{
Facet storeGeneratedFacet ;
if ( member . TypeUsage . Facets . TryGetValue ( EdmProviderManifest . StoreGeneratedPatternFacetName , false , out storeGeneratedFacet ) & &
storeGeneratedFacet . Value ! = null )
{
StoreGeneratedPattern pattern = ( StoreGeneratedPattern ) storeGeneratedFacet . Value ;
return pattern ;
}
return StoreGeneratedPattern . None ;
}
/// <summary>
/// Check if all the SchemaErrors have the serverity of SchemaErrorSeverity.Warning
/// </summary>
/// <param name="schemaErrors"></param>
/// <returns></returns>
internal static bool CheckIfAllErrorsAreWarnings ( IList < EdmSchemaError > schemaErrors )
{
int length = schemaErrors . Count ;
for ( int i = 0 ; i < length ; + + i )
{
EdmSchemaError error = schemaErrors [ i ] ;
if ( error . Severity ! = EdmSchemaErrorSeverity . Warning )
{
return false ;
}
}
return true ;
}
/// <summary>
///
/// </summary>
/// <param name="dictionaryExtentViews"></param>
/// <returns></returns>
internal static string GenerateHashForAllExtentViewsContent ( double schemaVersion , IEnumerable < KeyValuePair < string , string > > extentViews )
{
CompressingHashBuilder builder = new CompressingHashBuilder ( CreateMetadataHashAlgorithm ( schemaVersion ) ) ;
foreach ( var view in extentViews )
{
builder . AppendLine ( view . Key ) ;
builder . AppendLine ( view . Value ) ;
}
return builder . ComputeHash ( ) ;
}
[ System . Diagnostics . CodeAnalysis . SuppressMessage ( "Microsoft.Cryptographic.Standard" , "CA5350:Microsoft.Cryptographic.Standard" ,
Justification = "MD5CryptoServiceProvider is not used for cryptography/security purposes and we do it only for v1 and v1.1 for compatibility reasons." ) ]
2016-11-10 13:04:39 +00:00
[ System . Diagnostics . CodeAnalysis . SuppressMessage ( "Microsoft.Security.Cryptography" , "CA5350:Microsoft.Cryptographic.Standard" ,
Justification = "MD5CryptoServiceProvider is not used for cryptography/security purposes and we do it only for v1 and v1.1 for compatibility reasons." ) ]
2016-08-03 10:59:49 +00:00
internal static HashAlgorithm CreateMetadataHashAlgorithm ( double schemaVersion )
{
HashAlgorithm hashAlgorithm ;
if ( schemaVersion < XmlConstants . EdmVersionForV2 )
{
// v1 and v1.1 use old hash to remain compatible
hashAlgorithm = new MD5CryptoServiceProvider ( ) ;
}
else
{
// v2 and above use a FIPS approved provider
// so that when FIPS only is enforced by the OS
// we still work
hashAlgorithm = CreateSHA256HashAlgorithm ( ) ;
}
return hashAlgorithm ;
}
internal static SHA256 CreateSHA256HashAlgorithm ( )
{
SHA256 sha256HashAlgorith ;
try
{
// use the FIPS compliant SHA256 implementation
sha256HashAlgorith = new SHA256CryptoServiceProvider ( ) ;
}
catch ( PlatformNotSupportedException )
{
// the FIPS compliant (and faster) algorith was not available, create the managed version
// this will throw if FIPS only is enforced
sha256HashAlgorith = new SHA256Managed ( ) ;
}
return sha256HashAlgorith ;
}
internal static TypeUsage ConvertStoreTypeUsageToEdmTypeUsage ( TypeUsage storeTypeUsage )
{
TypeUsage edmTypeUsage = storeTypeUsage . GetModelTypeUsage ( ) . ShallowCopy ( FacetValues . NullFacetValues ) ;
// we don't reason the facets during the function resolution any more
return edmTypeUsage ;
}
internal static byte GetPrecision ( this TypeUsage type )
{
return type . GetFacetValue < byte > ( "Precision" ) ;
}
internal static byte GetScale ( this TypeUsage type )
{
return type . GetFacetValue < byte > ( "Scale" ) ;
}
internal static int GetMaxLength ( this TypeUsage type )
{
return type . GetFacetValue < int > ( "MaxLength" ) ;
}
internal static T GetFacetValue < T > ( this TypeUsage type , string facetName )
{
return ( T ) type . Facets [ facetName ] . Value ;
}
#region NavigationPropertyAccessor Helpers
internal static NavigationPropertyAccessor GetNavigationPropertyAccessor ( EntityType sourceEntityType , AssociationEndMember sourceMember , AssociationEndMember targetMember )
{
Debug . Assert ( sourceEntityType . DataSpace = = DataSpace . OSpace & & sourceEntityType . ClrType ! = null , "sourceEntityType must contain an ospace type" ) ;
return GetNavigationPropertyAccessor ( sourceEntityType , sourceMember . DeclaringType . FullName , sourceMember . Name , targetMember . Name ) ;
}
internal static NavigationPropertyAccessor GetNavigationPropertyAccessor ( EntityType entityType , string relationshipType , string fromName , string toName )
{
NavigationProperty navigationProperty ;
if ( entityType . TryGetNavigationProperty ( relationshipType , fromName , toName , out navigationProperty ) )
{
return navigationProperty . Accessor ;
}
else
{
return NavigationPropertyAccessor . NoNavigationProperty ;
}
}
#endregion
}
}