928 lines
39 KiB
C#
Raw Normal View History

//---------------------------------------------------------------------
// <copyright file="MetadataHelper.cs" company="Microsoft">
// Copyright (c) Microsoft Corporation. All rights reserved.
// </copyright>
//
// @owner Microsoft
// @backupOwner Microsoft
//---------------------------------------------------------------------
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.")]
[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.")]
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
}
}