130 lines
5.7 KiB
C#
130 lines
5.7 KiB
C#
|
using System.Collections.Generic;
|
||
|
using System.Collections.ObjectModel;
|
||
|
using System.Data.Metadata.Edm;
|
||
|
using System.Data.Objects;
|
||
|
using System.Diagnostics;
|
||
|
using System.Globalization;
|
||
|
using System.Linq;
|
||
|
|
||
|
namespace System.Web.DynamicData.ModelProviders {
|
||
|
internal sealed class EFDataModelProvider : DataModelProvider {
|
||
|
private ReadOnlyCollection<TableProvider> _tables;
|
||
|
|
||
|
internal Dictionary<long, EFColumnProvider> RelationshipEndLookup { get; private set; }
|
||
|
internal Dictionary<EntityType, EFTableProvider> TableEndLookup { get; private set; }
|
||
|
private Func<object> ContextFactory { get; set; }
|
||
|
private Dictionary<EdmType, Type> _entityTypeToClrType = new Dictionary<EdmType, Type>();
|
||
|
private ObjectContext _context;
|
||
|
private ObjectItemCollection _objectSpaceItems;
|
||
|
|
||
|
public EFDataModelProvider(object contextInstance, Func<object> contextFactory) {
|
||
|
ContextFactory = contextFactory;
|
||
|
RelationshipEndLookup = new Dictionary<long, EFColumnProvider>();
|
||
|
TableEndLookup = new Dictionary<EntityType, EFTableProvider>();
|
||
|
|
||
|
_context = (ObjectContext)contextInstance ?? (ObjectContext)CreateContext();
|
||
|
ContextType = _context.GetType();
|
||
|
|
||
|
// get a "container" (a scope at the instance level)
|
||
|
EntityContainer container = _context.MetadataWorkspace.GetEntityContainer(_context.DefaultContainerName, DataSpace.CSpace);
|
||
|
// load object space metadata
|
||
|
_context.MetadataWorkspace.LoadFromAssembly(ContextType.Assembly);
|
||
|
_objectSpaceItems = (ObjectItemCollection)_context.MetadataWorkspace.GetItemCollection(DataSpace.OSpace);
|
||
|
|
||
|
var tables = new List<TableProvider>();
|
||
|
|
||
|
// Create a dictionary from entity type to entity set. The entity type should be at the root of any inheritance chain.
|
||
|
IDictionary<EntityType, EntitySet> entitySetLookup = container.BaseEntitySets.OfType<EntitySet>().ToDictionary(e => e.ElementType);
|
||
|
|
||
|
// Create a lookup from parent entity to entity
|
||
|
ILookup<EntityType, EntityType> derivedTypesLookup = _context.MetadataWorkspace.GetItems<EntityType>(DataSpace.CSpace).ToLookup(e => (EntityType)e.BaseType);
|
||
|
|
||
|
// Keeps track of the current entity set being processed
|
||
|
EntitySet currentEntitySet = null;
|
||
|
|
||
|
// Do a DFS to get the inheritance hierarchy in order
|
||
|
// i.e. Consider the hierarchy
|
||
|
// null -> Person
|
||
|
// Person -> Employee, Contact
|
||
|
// Employee -> SalesPerson, Programmer
|
||
|
// We'll walk the children in a depth first order -> Person, Employee, SalesPerson, Programmer, Contact.
|
||
|
var objectStack = new Stack<EntityType>();
|
||
|
// Start will null (the root of the hierarchy)
|
||
|
objectStack.Push(null);
|
||
|
while (objectStack.Any()) {
|
||
|
EntityType entityType = objectStack.Pop();
|
||
|
if (entityType != null) {
|
||
|
// Update the entity set when we are at another root type (a type without a base type).
|
||
|
if (entityType.BaseType == null) {
|
||
|
currentEntitySet = entitySetLookup[entityType];
|
||
|
}
|
||
|
|
||
|
var table = CreateTableProvider(currentEntitySet, entityType);
|
||
|
tables.Add(table);
|
||
|
}
|
||
|
|
||
|
foreach (EntityType derivedEntityType in derivedTypesLookup[entityType]) {
|
||
|
// Push the derived entity types on the stack
|
||
|
objectStack.Push(derivedEntityType);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
_tables = tables.AsReadOnly();
|
||
|
}
|
||
|
|
||
|
public override object CreateContext() {
|
||
|
return ContextFactory();
|
||
|
}
|
||
|
|
||
|
public override ReadOnlyCollection<TableProvider> Tables {
|
||
|
get {
|
||
|
return _tables;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
internal Type GetClrType(EdmType entityType) {
|
||
|
var result = _entityTypeToClrType[entityType];
|
||
|
Debug.Assert(result != null, String.Format(CultureInfo.CurrentCulture, "Cannot map EdmType '{0}' to matching CLR Type", entityType));
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
internal Type GetClrType(EnumType enumType) {
|
||
|
var objectSpaceType = (EnumType)_context.MetadataWorkspace.GetObjectSpaceType(enumType);
|
||
|
return _objectSpaceItems.GetClrType(objectSpaceType);
|
||
|
}
|
||
|
|
||
|
private Type GetClrType(EntityType entityType) {
|
||
|
var objectSpaceType = (EntityType)_context.MetadataWorkspace.GetObjectSpaceType(entityType);
|
||
|
return _objectSpaceItems.GetClrType(objectSpaceType);
|
||
|
}
|
||
|
|
||
|
private TableProvider CreateTableProvider(EntitySet entitySet, EntityType entityType) {
|
||
|
// Get the parent clr type
|
||
|
Type parentClrType = null;
|
||
|
EntityType parentEntityType = entityType.BaseType as EntityType;
|
||
|
if (parentEntityType != null) {
|
||
|
parentClrType = GetClrType(parentEntityType);
|
||
|
}
|
||
|
|
||
|
Type rootClrType = GetClrType(entitySet.ElementType);
|
||
|
Type clrType = GetClrType(entityType);
|
||
|
|
||
|
_entityTypeToClrType[entityType] = clrType;
|
||
|
|
||
|
// Normally, use the entity set name as the table name
|
||
|
string tableName = entitySet.Name;
|
||
|
|
||
|
// But in inheritance scenarios where all types in the hierarchy share the same entity set,
|
||
|
// we need to use the type name instead for the table name.
|
||
|
if (parentClrType != null) {
|
||
|
tableName = entityType.Name;
|
||
|
}
|
||
|
|
||
|
EFTableProvider table = new EFTableProvider(this, entitySet, entityType, clrType, parentClrType, rootClrType, tableName);
|
||
|
TableEndLookup[entityType] = table;
|
||
|
|
||
|
return table;
|
||
|
}
|
||
|
}
|
||
|
}
|