You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			1830 lines
		
	
	
		
			76 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			1830 lines
		
	
	
		
			76 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|   | using System; | ||
|  | using System.Collections; | ||
|  | using System.Collections.Generic; | ||
|  | using System.Collections.ObjectModel; | ||
|  | using System.Linq.Expressions; | ||
|  | using System.Reflection; | ||
|  | using System.Text; | ||
|  | using System.Linq; | ||
|  | using System.Data.Linq.Provider; | ||
|  | using System.Data.Linq.Mapping; | ||
|  | using System.Data.Linq.SqlClient; | ||
|  | using System.Threading; | ||
|  | using System.Runtime.Versioning; | ||
|  | using LinqToSqlShared.Mapping; | ||
|  | using System.Runtime.CompilerServices; | ||
|  | 
 | ||
|  | namespace System.Data.Linq.Mapping { | ||
|  | 
 | ||
|  |     internal class MappedMetaModel : MetaModel { | ||
|  |         ReaderWriterLock @lock = new ReaderWriterLock(); | ||
|  |         MappingSource mappingSource; | ||
|  |         Type contextType; | ||
|  |         Type providerType; | ||
|  |         DatabaseMapping mapping; | ||
|  |         HashSet<Module> modules; | ||
|  |         Dictionary<string, Type> types; | ||
|  |         Dictionary<Type, MetaType> metaTypes; | ||
|  |         Dictionary<Type, MetaTable> metaTables; | ||
|  |         Dictionary<MetaPosition, MetaFunction> metaFunctions; | ||
|  |         bool fullyLoaded; | ||
|  | 
 | ||
|  |         [ResourceExposure(ResourceScope.Assembly | ResourceScope.Machine)] // mapping parameter contains various type references. | ||
|  |         [ResourceConsumption(ResourceScope.Assembly | ResourceScope.Machine)] // FindType method call. | ||
|  |         internal MappedMetaModel(MappingSource mappingSource, Type contextType, DatabaseMapping mapping) { | ||
|  |             this.mappingSource = mappingSource; | ||
|  |             this.contextType = contextType; | ||
|  |             this.mapping = mapping; | ||
|  |             this.modules = new HashSet<Module>(); | ||
|  |             this.modules.Add(this.contextType.Module); | ||
|  |             this.metaTypes = new Dictionary<Type, MetaType>(); | ||
|  |             this.metaTables = new Dictionary<Type, MetaTable>(); | ||
|  |             this.types = new Dictionary<string, Type>(); | ||
|  |             // Provider type | ||
|  |             if (this.providerType == null && !String.IsNullOrEmpty(this.mapping.Provider)) { | ||
|  |                 this.providerType = this.FindType(this.mapping.Provider, typeof(SqlProvider).Namespace); | ||
|  |                 if (this.providerType == null) { | ||
|  |                     throw Error.ProviderTypeNotFound(this.mapping.Provider); | ||
|  |                 } | ||
|  |             } | ||
|  |             else if (this.providerType == null) { | ||
|  |                 this.providerType = typeof(SqlProvider); | ||
|  |             } | ||
|  |             this.Init(); | ||
|  |         } | ||
|  |         #region Initialization | ||
|  |         private void Init() { | ||
|  |             if (!fullyLoaded) { | ||
|  |                 // The fullyLoaded state is required so that tools like | ||
|  |                 // CreateDatabase can get a full view of all tables. | ||
|  |                 @lock.AcquireWriterLock(Timeout.Infinite); | ||
|  |                 try { | ||
|  |                     if (!fullyLoaded) { | ||
|  |                         // Initialize static tables and functions. | ||
|  |                         this.InitStaticTables(); | ||
|  |                         this.InitFunctions(); | ||
|  |                         fullyLoaded = true; | ||
|  |                     } | ||
|  |                 } | ||
|  |                 finally { | ||
|  |                     @lock.ReleaseWriterLock(); | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         [ResourceExposure(ResourceScope.None)] // Exposure is via external mapping file/attributes. | ||
|  |         [ResourceConsumption(ResourceScope.Assembly | ResourceScope.Machine, ResourceScope.Assembly | ResourceScope.Machine)] // FindType method call. | ||
|  |         private void InitStaticTables() { | ||
|  |             this.InitStaticTableTypes(); | ||
|  |             foreach (TableMapping tableMapping in this.mapping.Tables) { | ||
|  |                 Type rowType = this.FindType(tableMapping.RowType.Name); | ||
|  |                 if (rowType == null) { | ||
|  |                     throw Error.CouldNotFindTypeFromMapping(tableMapping.RowType.Name); | ||
|  |                 } | ||
|  |                 Type rootType = this.GetRootType(rowType, tableMapping.RowType); | ||
|  |                 MetaTable table = new MappedTable(this, tableMapping, rootType); | ||
|  |                 foreach (MetaType mt in table.RowType.InheritanceTypes) { | ||
|  |                     this.metaTypes.Add(mt.Type, mt); | ||
|  |                     this.metaTables.Add(mt.Type, table); | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         private void InitStaticTableTypes() { | ||
|  |             for (Type type = this.contextType; type != typeof(DataContext); type = type.BaseType) { | ||
|  |                 FieldInfo[] fields = type.GetFields(BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance | BindingFlags.DeclaredOnly); | ||
|  |                 foreach (FieldInfo fi in fields) { | ||
|  |                     Type ft = fi.FieldType; | ||
|  |                     if (ft.IsGenericType && ft.GetGenericTypeDefinition() == typeof(Table<>)) { | ||
|  |                         Type rowType = ft.GetGenericArguments()[0]; | ||
|  |                         if (!this.types.ContainsKey(rowType.Name)) { | ||
|  |                             this.types.Add(rowType.FullName, rowType); | ||
|  |                             if (!this.types.ContainsKey(rowType.Name)) { | ||
|  |                                 this.types.Add(rowType.Name, rowType); | ||
|  |                             } | ||
|  |                             this.modules.Add(rowType.Module); | ||
|  |                         } | ||
|  |                     } | ||
|  |                 } | ||
|  |                 PropertyInfo[] props = type.GetProperties(BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance | BindingFlags.DeclaredOnly); | ||
|  |                 foreach (PropertyInfo pi in props) { | ||
|  |                     Type pt = pi.PropertyType; | ||
|  |                     if (pt.IsGenericType && pt.GetGenericTypeDefinition() == typeof(Table<>)) { | ||
|  |                         Type rowType = pt.GetGenericArguments()[0]; | ||
|  |                         if (!this.types.ContainsKey(rowType.Name)) { | ||
|  |                             this.types.Add(rowType.FullName, rowType); | ||
|  |                             if (!this.types.ContainsKey(rowType.Name)) { | ||
|  |                                 this.types.Add(rowType.Name, rowType); | ||
|  |                             } | ||
|  |                             this.modules.Add(rowType.Module); | ||
|  |                         } | ||
|  |                     } | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         [ResourceExposure(ResourceScope.None)] // mapping instance variable is set elsewhere. | ||
|  |         [ResourceConsumption(ResourceScope.Assembly | ResourceScope.Machine, ResourceScope.Assembly | ResourceScope.Machine)] // For GetMethods call. | ||
|  |         private void InitFunctions() { | ||
|  |             this.metaFunctions = new Dictionary<MetaPosition, MetaFunction>(); | ||
|  |             if (this.contextType != typeof(DataContext)) { | ||
|  |                 foreach (FunctionMapping fmap in this.mapping.Functions) { | ||
|  |                     MethodInfo method = this.GetMethod(fmap.MethodName); | ||
|  |                     if (method == null) { | ||
|  |                         throw Error.MethodCannotBeFound(fmap.MethodName); | ||
|  |                     } | ||
|  |                     MappedFunction func = new MappedFunction(this, fmap, method); | ||
|  |                     this.metaFunctions.Add(new MetaPosition(method), func); | ||
|  | 
 | ||
|  |                     // pre-set all known function result types into metaType map | ||
|  |                     foreach (MetaType rt in func.ResultRowTypes) { | ||
|  |                         foreach (MetaType it in rt.InheritanceTypes) { | ||
|  |                             if (!this.metaTypes.ContainsKey(it.Type)) { | ||
|  |                                 this.metaTypes.Add(it.Type, it); | ||
|  |                             } | ||
|  |                         } | ||
|  |                     } | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  |         #endregion | ||
|  | 
 | ||
|  |         public override MappingSource MappingSource { | ||
|  |             get { return this.mappingSource; } | ||
|  |         } | ||
|  | 
 | ||
|  |         public override Type ContextType { | ||
|  |             get { return this.contextType; } | ||
|  |         } | ||
|  | 
 | ||
|  |         public override string DatabaseName { | ||
|  |             get { return this.mapping.DatabaseName; } | ||
|  |         } | ||
|  | 
 | ||
|  |         public override Type ProviderType { | ||
|  |             get { return this.providerType; } | ||
|  |         } | ||
|  | 
 | ||
|  |         public override IEnumerable<MetaTable> GetTables() { | ||
|  |             return this.metaTables.Values.Where(x => x != null).Distinct(); | ||
|  |         } | ||
|  | 
 | ||
|  |         public override MetaTable GetTable(Type rowType) { | ||
|  |             if (rowType == null) { | ||
|  |                 throw Error.ArgumentNull("rowType"); | ||
|  |             } | ||
|  |             MetaTable table = null; | ||
|  |             this.metaTables.TryGetValue(rowType, out table); | ||
|  |             return table; | ||
|  |         } | ||
|  | 
 | ||
|  |         public override MetaType GetMetaType(Type type) { | ||
|  |             if (type == null) { | ||
|  |                 throw Error.ArgumentNull("type"); | ||
|  |             } | ||
|  |             MetaType mtype = null; | ||
|  |             @lock.AcquireReaderLock(Timeout.Infinite); | ||
|  |             try { | ||
|  |                 if (this.metaTypes.TryGetValue(type, out mtype)) { | ||
|  |                     return mtype; | ||
|  |                 } | ||
|  |             } | ||
|  |             finally { | ||
|  |                 @lock.ReleaseReaderLock(); | ||
|  |             } | ||
|  |             @lock.AcquireWriterLock(Timeout.Infinite); | ||
|  |             try { | ||
|  |                 if (!this.metaTypes.TryGetValue(type, out mtype)) { | ||
|  |                     // not known, so must be unmapped type | ||
|  |                     mtype = new UnmappedType(this, type); | ||
|  |                     this.metaTypes.Add(type, mtype); | ||
|  |                 } | ||
|  |             } | ||
|  |             finally { | ||
|  |                 @lock.ReleaseWriterLock(); | ||
|  |             } | ||
|  |             return mtype; | ||
|  |         } | ||
|  | 
 | ||
|  |         public override MetaFunction GetFunction(MethodInfo method) { | ||
|  |             if (method == null) { | ||
|  |                 throw new ArgumentNullException("method"); | ||
|  |             } | ||
|  |             MetaFunction func = null; | ||
|  |             this.metaFunctions.TryGetValue(new MetaPosition(method), out func); | ||
|  |             return func; | ||
|  |         } | ||
|  | 
 | ||
|  |         public override IEnumerable<MetaFunction> GetFunctions() { | ||
|  |             return this.metaFunctions.Values; | ||
|  |         } | ||
|  | 
 | ||
|  |         private Type GetRootType(Type type, TypeMapping rootMapping) { | ||
|  |             if (string.Compare(rootMapping.Name, type.Name, StringComparison.Ordinal) == 0 | ||
|  |                 || string.Compare(rootMapping.Name, type.FullName, StringComparison.Ordinal) == 0 | ||
|  |                 || string.Compare(rootMapping.Name, type.AssemblyQualifiedName, StringComparison.Ordinal) == 0) | ||
|  |                 return type; | ||
|  |             if (type.BaseType != typeof(object)) { | ||
|  |                 return this.GetRootType(type.BaseType, rootMapping); | ||
|  |             } | ||
|  |             throw Error.UnableToResolveRootForType(type); | ||
|  |         } | ||
|  | 
 | ||
|  |         [ResourceExposure(ResourceScope.Assembly | ResourceScope.Machine)] // name parameter will be found on a type. | ||
|  |         [ResourceConsumption(ResourceScope.Assembly | ResourceScope.Machine)] // FindType method call. | ||
|  |         private MethodInfo GetMethod(string name) { | ||
|  |             string typeName, methodName; | ||
|  |             this.GetTypeAndMethod(name, out typeName, out methodName); | ||
|  |             Type type = this.FindType(typeName); | ||
|  |             if (type != null) { | ||
|  |                 return type.GetMethod(methodName, BindingFlags.Instance|BindingFlags.Public|BindingFlags.NonPublic); | ||
|  |             } | ||
|  |             return null; | ||
|  |         } | ||
|  | 
 | ||
|  |         private void GetTypeAndMethod(string name, out string typeName, out string methodName) { | ||
|  |             int dotIndex = name.LastIndexOf(".", StringComparison.CurrentCulture); | ||
|  |             if (dotIndex > 0) { | ||
|  |                 typeName = name.Substring(0, dotIndex); | ||
|  |                 methodName = name.Substring(dotIndex + 1); | ||
|  |             } | ||
|  |             else { | ||
|  |                 typeName = this.contextType.FullName; | ||
|  |                 methodName = name; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         [ResourceExposure(ResourceScope.Assembly | ResourceScope.Machine)] // name parameter is a type name. | ||
|  |         [ResourceConsumption(ResourceScope.Assembly | ResourceScope.Machine)] // FindType method call. | ||
|  |         internal Type FindType(string name) { | ||
|  |             return this.FindType(name, this.contextType.Namespace); | ||
|  |         } | ||
|  | 
 | ||
|  |         [ResourceExposure(ResourceScope.Assembly | ResourceScope.Machine)] // name parameter is a type name. | ||
|  |         [ResourceConsumption(ResourceScope.Assembly | ResourceScope.Machine)] // SearchForType method call. | ||
|  |         internal Type FindType(string name, string defaultNamespace) { | ||
|  |             Type result = null; | ||
|  |             string name2 = null; | ||
|  |             @lock.AcquireReaderLock(Timeout.Infinite); | ||
|  |             try { | ||
|  |                 if (this.types.TryGetValue(name, out result)) { | ||
|  |                     return result; | ||
|  |                 } | ||
|  |                 name2 = name.Contains(".") ? null : defaultNamespace + "." + name; | ||
|  |                 if (name2 != null && this.types.TryGetValue(name2, out result)) { | ||
|  |                     return result; | ||
|  |                 } | ||
|  |             } | ||
|  |             finally { | ||
|  |                 @lock.ReleaseReaderLock(); | ||
|  |             } | ||
|  |             // don't block anyone while we search for the correct type | ||
|  |             Type foundResult = this.SearchForType(name); | ||
|  | 
 | ||
|  |             if (foundResult == null && name2 != null) { | ||
|  |                 foundResult = this.SearchForType(name2); | ||
|  |             } | ||
|  |             if (foundResult != null) { | ||
|  |                 @lock.AcquireWriterLock(Timeout.Infinite); | ||
|  |                 try { | ||
|  |                     if (this.types.TryGetValue(name, out result)) { | ||
|  |                         return result;  | ||
|  |                     } | ||
|  |                     this.types.Add(name, foundResult); | ||
|  |                     return foundResult; | ||
|  |                 } | ||
|  |                 finally { | ||
|  |                     @lock.ReleaseWriterLock(); | ||
|  |                 } | ||
|  |             } | ||
|  |             return null; | ||
|  |         } | ||
|  | 
 | ||
|  |         [ResourceExposure(ResourceScope.Assembly | ResourceScope.Machine)] // name parameter is a type name. | ||
|  |         [ResourceConsumption(ResourceScope.Assembly | ResourceScope.Machine)] // SearchForType method call. | ||
|  |         private Type SearchForType(string name) { | ||
|  |             // Try search for type using case sensitive. | ||
|  |             Type type = SearchForType(name, false); | ||
|  |             if (type != null) { | ||
|  |                 return type; | ||
|  |             } | ||
|  | 
 | ||
|  |             // Try search for type using case in-sensitive. | ||
|  |             return SearchForType(name, true); | ||
|  |         } | ||
|  | 
 | ||
|  |        [ResourceExposure(ResourceScope.Assembly | ResourceScope.Machine)] // name parameter is a type name. | ||
|  |        [ResourceConsumption(ResourceScope.Assembly | ResourceScope.Machine)] // Assembly.GetLoadedModules method call. | ||
|  |        private Type SearchForType(string name, bool ignoreCase) { | ||
|  |             // first try default system lookup | ||
|  |             Type type = Type.GetType(name, false, ignoreCase); | ||
|  |             if (type != null) { | ||
|  |                 return type; | ||
|  |             } | ||
|  | 
 | ||
|  |             // try all known modules (modules that other statically known types were found in) | ||
|  |             foreach (Module module in this.modules) { | ||
|  |                 type = module.GetType(name, false, ignoreCase); | ||
|  |                 if (type != null) { | ||
|  |                     return type; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             // try all loaded modules (is there a better way?) | ||
|  |             foreach (Assembly a in AppDomain.CurrentDomain.GetAssemblies()) { | ||
|  |                 foreach (Module module in a.GetLoadedModules()) { | ||
|  |                     type = module.GetType(name, false, ignoreCase); | ||
|  |                     if (type != null) { | ||
|  |                         return type; | ||
|  |                     } | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             return null; | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     internal sealed class MappedTable : MetaTable { | ||
|  |         MappedMetaModel model; | ||
|  |         TableMapping mapping; | ||
|  |         MetaType rowType; | ||
|  |         bool hasMethods; | ||
|  |         MethodInfo insertMethod; | ||
|  |         MethodInfo updateMethod; | ||
|  |         MethodInfo deleteMethod; | ||
|  | 
 | ||
|  |         [ResourceExposure(ResourceScope.Assembly | ResourceScope.Machine)] // Parameter contains various type references. | ||
|  |         [ResourceConsumption(ResourceScope.Assembly | ResourceScope.Machine)] // MappedRootType constructor call. | ||
|  |         internal MappedTable(MappedMetaModel model, TableMapping mapping, Type rowType) { | ||
|  |             this.model = model; | ||
|  |             this.mapping = mapping; | ||
|  |             this.rowType = new MappedRootType(model, this, mapping.RowType, rowType); | ||
|  |         } | ||
|  |         public override MetaModel Model { | ||
|  |             get { return this.model; } | ||
|  |         } | ||
|  |         public override string TableName { | ||
|  |             get { return this.mapping.TableName; } | ||
|  |         } | ||
|  |         public override MetaType RowType { | ||
|  |             get { return this.rowType; } | ||
|  |         } | ||
|  |         public override MethodInfo InsertMethod { | ||
|  |             get { | ||
|  |                 this.InitMethods(); | ||
|  |                 return this.insertMethod; | ||
|  |             } | ||
|  |         } | ||
|  |         public override MethodInfo UpdateMethod { | ||
|  |             get { | ||
|  |                 this.InitMethods(); | ||
|  |                 return this.updateMethod; | ||
|  |             } | ||
|  |         } | ||
|  |         public override MethodInfo DeleteMethod { | ||
|  |             get { | ||
|  |                 this.InitMethods(); | ||
|  |                 return this.deleteMethod; | ||
|  |             } | ||
|  |         } | ||
|  |         private void InitMethods() { | ||
|  |             if (!this.hasMethods) { | ||
|  |                 this.insertMethod = MethodFinder.FindMethod( | ||
|  |                     this.model.ContextType, | ||
|  |                     "Insert" + rowType.Name, | ||
|  |                     BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic, | ||
|  |                     new Type[] { rowType.Type } | ||
|  |                     ); | ||
|  |                 this.updateMethod = MethodFinder.FindMethod( | ||
|  |                     this.model.ContextType, | ||
|  |                     "Update" + rowType.Name, | ||
|  |                     BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic, | ||
|  |                     new Type[] { rowType.Type } | ||
|  |                     ); | ||
|  |                 this.deleteMethod = MethodFinder.FindMethod( | ||
|  |                     this.model.ContextType, | ||
|  |                     "Delete" + rowType.Name, | ||
|  |                     BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic, | ||
|  |                     new Type[] { rowType.Type } | ||
|  |                     ); | ||
|  |                 this.hasMethods = true; | ||
|  |             } | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     internal sealed class MappedRootType : MappedType { | ||
|  |         Dictionary<Type, MetaType> derivedTypes; | ||
|  |         Dictionary<object, MetaType> inheritanceCodes; | ||
|  |         ReadOnlyCollection<MetaType> inheritanceTypes; | ||
|  |         MetaType inheritanceDefault; | ||
|  |         bool hasInheritance; | ||
|  | 
 | ||
|  |         [ResourceExposure(ResourceScope.Assembly | ResourceScope.Machine)] // Various parameters can contain type names. | ||
|  |         [ResourceConsumption(ResourceScope.Assembly | ResourceScope.Machine)] // InitInheritedType method call. | ||
|  |         public MappedRootType(MappedMetaModel model, MappedTable table, TypeMapping typeMapping, Type type) | ||
|  |             : base(model, table, typeMapping, type, null) { | ||
|  |             if (typeMapping == null) | ||
|  |                 throw Error.ArgumentNull("typeMapping"); | ||
|  | 
 | ||
|  |             if (typeMapping.InheritanceCode != null || typeMapping.DerivedTypes.Count > 0) { | ||
|  |                 if (this.Discriminator == null) { | ||
|  |                     throw Error.NoDiscriminatorFound(type.Name); | ||
|  |                 } | ||
|  |                 this.hasInheritance = true; | ||
|  |                 if (!MappingSystem.IsSupportedDiscriminatorType(this.Discriminator.Type)) { | ||
|  |                     throw Error.DiscriminatorClrTypeNotSupported(this.Discriminator.DeclaringType.Name, this.Discriminator.Name, this.Discriminator.Type); | ||
|  |                 } | ||
|  |                 this.derivedTypes = new Dictionary<Type, MetaType>(); | ||
|  |                 this.inheritanceCodes = new Dictionary<object, MetaType>(); | ||
|  |                 this.InitInheritedType(typeMapping, this); | ||
|  |             } | ||
|  | 
 | ||
|  |             if (this.inheritanceDefault == null && (this.inheritanceCode != null || this.inheritanceCodes != null && this.inheritanceCodes.Count > 0)) | ||
|  |                 throw Error.InheritanceHierarchyDoesNotDefineDefault(type); | ||
|  | 
 | ||
|  |             if (this.derivedTypes != null) { | ||
|  |                 this.inheritanceTypes = this.derivedTypes.Values.ToList().AsReadOnly(); | ||
|  |             } | ||
|  |             else { | ||
|  |                 this.inheritanceTypes = new MetaType[] { this }.ToList().AsReadOnly(); | ||
|  |             } | ||
|  | 
 | ||
|  |             this.Validate(); | ||
|  |         } | ||
|  | 
 | ||
|  |         private void Validate() { | ||
|  |             Dictionary<object, string> memberToColumn = new Dictionary<object, string>(); | ||
|  |             foreach (MetaType type in this.InheritanceTypes) { | ||
|  |                 // NB: Table node in XML can have only one Type node -- enforced by XSD | ||
|  | 
 | ||
|  |                 foreach (MetaDataMember mem in type.PersistentDataMembers) { | ||
|  |                     if (mem.IsDeclaredBy(type)) { | ||
|  |                         if (mem.IsDiscriminator && !this.HasInheritance) { | ||
|  |                             throw Error.NonInheritanceClassHasDiscriminator(type); | ||
|  |                         } | ||
|  |                         if (!mem.IsAssociation) { | ||
|  |                             // validate that no database column is mapped twice | ||
|  |                             if (!string.IsNullOrEmpty(mem.MappedName)) { | ||
|  |                                 string column; | ||
|  |                                 object dn = InheritanceRules.DistinguishedMemberName(mem.Member); | ||
|  |                                 if (memberToColumn.TryGetValue(dn, out column)) { | ||
|  |                                     if (column != mem.MappedName) { | ||
|  |                                         throw Error.MemberMappedMoreThanOnce(mem.Member.Name); | ||
|  |                                     } | ||
|  |                                 } | ||
|  |                                 else { | ||
|  |                                     memberToColumn.Add(dn, mem.MappedName); | ||
|  |                                 } | ||
|  |                             } | ||
|  |                         } | ||
|  |                     } | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         [ResourceExposure(ResourceScope.Assembly | ResourceScope.Machine)] // typeMap parameter's Name property. | ||
|  |         [ResourceConsumption(ResourceScope.Assembly | ResourceScope.Machine)] // FindType method call. | ||
|  |         private MetaType InitDerivedTypes(TypeMapping typeMap) { | ||
|  |             Type type = ((MappedMetaModel)Model).FindType(typeMap.Name); | ||
|  |             if (type == null) | ||
|  |                 throw Error.CouldNotFindRuntimeTypeForMapping(typeMap.Name); | ||
|  |             MappedType rowType = new MappedType(this.Model, this.Table, typeMap, type, this); | ||
|  |             return this.InitInheritedType(typeMap, rowType); | ||
|  |         } | ||
|  | 
 | ||
|  |         [ResourceExposure(ResourceScope.Assembly | ResourceScope.Machine)] // typeMap parameter's Name property. | ||
|  |         [ResourceConsumption(ResourceScope.Assembly | ResourceScope.Machine)] // InitDerivedTypes method call. | ||
|  |         private MetaType InitInheritedType(TypeMapping typeMap, MappedType type) { | ||
|  |             this.derivedTypes.Add(type.Type, type); | ||
|  | 
 | ||
|  |             if (typeMap.InheritanceCode != null) { // Mapping with no inheritance code: For example, an unmapped intermediate class in a hierarchy. | ||
|  |                 if (this.Discriminator == null) | ||
|  |                     throw Error.NoDiscriminatorFound(type.Name); | ||
|  | 
 | ||
|  |                 if (type.Type.IsAbstract) | ||
|  |                     throw Error.AbstractClassAssignInheritanceDiscriminator(type.Type); | ||
|  | 
 | ||
|  |                 object keyValue = DBConvert.ChangeType(typeMap.InheritanceCode, this.Discriminator.Type); | ||
|  |                 foreach (object d in inheritanceCodes.Keys) { | ||
|  |                     // if the keys are equal, or if they are both strings containing only spaces | ||
|  |                     // they are considered equal | ||
|  |                     if ((keyValue.GetType() == typeof(string) && ((string)keyValue).Trim().Length == 0 && | ||
|  |                         d.GetType() == typeof(string) && ((string)d).Trim().Length == 0) || | ||
|  |                         object.Equals(d, keyValue)) { | ||
|  |                         throw Error.InheritanceCodeUsedForMultipleTypes(keyValue); | ||
|  |                     } | ||
|  |                 } | ||
|  |                 if (type.inheritanceCode != null) | ||
|  |                     throw Error.InheritanceTypeHasMultipleDiscriminators(type); | ||
|  |                 type.inheritanceCode = keyValue; | ||
|  |                 this.inheritanceCodes.Add(keyValue, type); | ||
|  |                 if (typeMap.IsInheritanceDefault) { | ||
|  |                     if (this.inheritanceDefault != null) | ||
|  |                         throw Error.InheritanceTypeHasMultipleDefaults(type); | ||
|  |                     this.inheritanceDefault = type; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             // init sub-inherited types | ||
|  |             foreach (TypeMapping tm in typeMap.DerivedTypes) { | ||
|  |                 this.InitDerivedTypes(tm); | ||
|  |             } | ||
|  | 
 | ||
|  |             return type; | ||
|  |         } | ||
|  | 
 | ||
|  |         public override bool HasInheritance { | ||
|  |             get { return this.hasInheritance; } | ||
|  |         } | ||
|  | 
 | ||
|  |         public override bool HasInheritanceCode { | ||
|  |             get { return this.InheritanceCode != null; } | ||
|  |         } | ||
|  | 
 | ||
|  |         public override ReadOnlyCollection<MetaType> InheritanceTypes { | ||
|  |             get { return this.inheritanceTypes; } | ||
|  |         } | ||
|  | 
 | ||
|  |         public override MetaType GetInheritanceType(Type type) { | ||
|  |             if (type == this.Type) | ||
|  |                 return this; | ||
|  |             MetaType metaType = null; | ||
|  |             if (this.derivedTypes != null) { | ||
|  |                 this.derivedTypes.TryGetValue(type, out metaType); | ||
|  |             } | ||
|  |             return metaType; | ||
|  |         } | ||
|  | 
 | ||
|  |         public override MetaType InheritanceDefault { | ||
|  |             get { return this.inheritanceDefault; } | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     internal class MappedType : MetaType { | ||
|  |         MetaModel model; | ||
|  |         MetaTable table; | ||
|  |         Type type; | ||
|  |         TypeMapping typeMapping; | ||
|  |         Dictionary<object, MetaDataMember> dataMemberMap; | ||
|  |         ReadOnlyCollection<MetaDataMember> dataMembers; | ||
|  |         ReadOnlyCollection<MetaDataMember> persistentDataMembers; | ||
|  |         ReadOnlyCollection<MetaDataMember> identities; | ||
|  |         MetaDataMember dbGeneratedIdentity; | ||
|  |         MetaDataMember version; | ||
|  |         MetaDataMember discriminator; | ||
|  |         MetaType inheritanceRoot; | ||
|  |         bool inheritanceBaseSet; | ||
|  |         MetaType inheritanceBase; | ||
|  |         internal object inheritanceCode; | ||
|  |         ReadOnlyCollection<MetaType> derivedTypes; | ||
|  |         ReadOnlyCollection<MetaAssociation> associations; | ||
|  |         bool hasMethods; | ||
|  |         bool hasAnyLoadMethod; | ||
|  |         bool hasAnyValidateMethod; | ||
|  |         MethodInfo onLoadedMethod; | ||
|  |         MethodInfo onValidateMethod; | ||
|  | 
 | ||
|  |         object locktarget = new object(); // Hold locks on private object rather than public MetaType. | ||
|  | 
 | ||
|  |         internal MappedType(MetaModel model, MetaTable table, TypeMapping typeMapping, Type type, MetaType inheritanceRoot) { | ||
|  |             this.model = model; | ||
|  |             this.table = table; | ||
|  |             this.typeMapping = typeMapping; | ||
|  |             this.type = type; | ||
|  |             this.inheritanceRoot = inheritanceRoot != null ? inheritanceRoot : this; | ||
|  |             this.InitDataMembers(); | ||
|  | 
 | ||
|  |             this.identities = this.dataMembers.Where(m => m.IsPrimaryKey).ToList().AsReadOnly(); | ||
|  |             this.persistentDataMembers = this.dataMembers.Where(m => m.IsPersistent).ToList().AsReadOnly(); | ||
|  |         } | ||
|  |         #region Initialization | ||
|  |         private void ValidatePrimaryKeyMember(MetaDataMember mm) { | ||
|  |             //if the type is a sub-type, no member in the type can be primary key | ||
|  |             if (mm.IsPrimaryKey && this.inheritanceRoot != this && mm.Member.DeclaringType == this.type) { | ||
|  |                 throw (Error.PrimaryKeyInSubTypeNotSupported(this.type.Name, mm.Name)); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         private void InitMethods() { | ||
|  |             if (!this.hasMethods) { | ||
|  |                 this.onLoadedMethod = MethodFinder.FindMethod( | ||
|  |                     this.Type, | ||
|  |                     "OnLoaded", | ||
|  |                     BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic, | ||
|  |                     Type.EmptyTypes, | ||
|  |                     false | ||
|  |                     ); | ||
|  |                 this.onValidateMethod = MethodFinder.FindMethod( | ||
|  |                     this.Type, | ||
|  |                     "OnValidate", | ||
|  |                     BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic, | ||
|  |                     new[] { typeof(ChangeAction) }, | ||
|  |                     false | ||
|  |                     ); | ||
|  | 
 | ||
|  |                 this.hasAnyLoadMethod = (this.onLoadedMethod != null) || (this.InheritanceBase != null && this.InheritanceBase.HasAnyLoadMethod); | ||
|  |                 this.hasAnyValidateMethod = (this.onValidateMethod != null) || (this.InheritanceBase != null && this.InheritanceBase.HasAnyValidateMethod); | ||
|  | 
 | ||
|  |                 this.hasMethods = true; | ||
|  |             } | ||
|  |         } | ||
|  |          | ||
|  |         private void InitDataMembers() { | ||
|  |             if (this.dataMembers == null) { | ||
|  |                 Dictionary<object, MetaDataMember> map = new Dictionary<object, MetaDataMember>(); | ||
|  |                 List<MetaDataMember> dMembers = new List<MetaDataMember>(); | ||
|  |                 int ordinal = 0; | ||
|  |                 BindingFlags flags = BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.FlattenHierarchy; | ||
|  | 
 | ||
|  |                 // Map of valid mapped names. | ||
|  |                 Dictionary<string, MemberMapping> names = new Dictionary<string, MemberMapping>(); | ||
|  |                 Type currentType = this.type; | ||
|  |                 for (TypeMapping tm = this.typeMapping; tm != null; tm = tm.BaseType) { | ||
|  |                     foreach (MemberMapping mmap in tm.Members) { | ||
|  |                         names[mmap.MemberName + ":" + currentType.Name] = mmap; | ||
|  |                     } | ||
|  |                     currentType = currentType.BaseType; | ||
|  |                 } | ||
|  | 
 | ||
|  |                 HashSet<string> namesSeen = new HashSet<string>(); // Keep track of which names from the mapping file have been seen. | ||
|  |                 FieldInfo[] fis = TypeSystem.GetAllFields(this.type, flags).ToArray(); | ||
|  |                 if (fis != null) { | ||
|  |                     foreach (FieldInfo fi in fis) { | ||
|  |                         MemberMapping mmap; | ||
|  |                         string name = fi.Name + ":" + fi.DeclaringType.Name; | ||
|  |                         if (names.TryGetValue(name, out mmap)) { | ||
|  |                             namesSeen.Add(name); | ||
|  |                             object dn = InheritanceRules.DistinguishedMemberName(fi); | ||
|  |                             MetaDataMember mm; | ||
|  |                             if (!map.TryGetValue(dn, out mm)) { | ||
|  |                                 mm = new MappedDataMember(this, fi, mmap, ordinal); | ||
|  |                                 map.Add(InheritanceRules.DistinguishedMemberName(mm.Member), mm); | ||
|  |                                 dMembers.Add(mm); | ||
|  |                                 this.InitSpecialMember(mm); | ||
|  |                             } | ||
|  |                             ValidatePrimaryKeyMember(mm); | ||
|  |                             ordinal++; | ||
|  |                         } | ||
|  |                     } | ||
|  |                 } | ||
|  | 
 | ||
|  |                 PropertyInfo[] pis = TypeSystem.GetAllProperties(this.type, flags).ToArray(); | ||
|  |                 if (pis != null) { | ||
|  |                     foreach (PropertyInfo pi in pis) { | ||
|  |                         MemberMapping mmap; | ||
|  |                         string name = pi.Name + ":" + pi.DeclaringType.Name; | ||
|  |                         if (names.TryGetValue(name, out mmap)) { | ||
|  |                             namesSeen.Add(name); | ||
|  |                             MetaDataMember mm; | ||
|  |                             object dn = InheritanceRules.DistinguishedMemberName(pi); | ||
|  |                             if (!map.TryGetValue(dn, out mm)) { | ||
|  |                                 mm = new MappedDataMember(this, pi, mmap, ordinal); | ||
|  |                                 map.Add(InheritanceRules.DistinguishedMemberName(mm.Member), mm); | ||
|  |                                 dMembers.Add(mm); | ||
|  |                                 this.InitSpecialMember(mm); | ||
|  |                             } | ||
|  |                             ValidatePrimaryKeyMember(mm); | ||
|  |                             ordinal++; | ||
|  |                         } | ||
|  |                     } | ||
|  |                 } | ||
|  | 
 | ||
|  |                 this.dataMembers = dMembers.AsReadOnly(); | ||
|  |                 this.dataMemberMap = map; | ||
|  | 
 | ||
|  |                 // Finally, make sure that all types in the mapping file were consumed. | ||
|  |                 foreach(string name in namesSeen) { | ||
|  |                     names.Remove(name); | ||
|  |                 } | ||
|  |                 foreach(var orphan in names) { | ||
|  |                     Type aboveRoot = inheritanceRoot.Type.BaseType; | ||
|  |                     while (aboveRoot!=null) { | ||
|  |                         foreach(MemberInfo mi in aboveRoot.GetMembers(BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic)) { | ||
|  |                             if(String.Compare(mi.Name, orphan.Value.MemberName, StringComparison.Ordinal)==0) { | ||
|  |                                 throw Error.MappedMemberHadNoCorrespondingMemberInType(orphan.Value.MemberName, type.Name); | ||
|  |                             } | ||
|  |                         } | ||
|  |                         aboveRoot = aboveRoot.BaseType; | ||
|  |                     } | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  |         private void InitSpecialMember(MetaDataMember mm) { | ||
|  |             // Can only have one auto gen member that is also an identity member, | ||
|  |             // except if that member is a computed column (since they are implicitly auto gen) | ||
|  |             if (mm.IsDbGenerated && mm.IsPrimaryKey && string.IsNullOrEmpty(mm.Expression)) { | ||
|  |                 if (this.dbGeneratedIdentity != null) { | ||
|  |                     throw Error.TwoMembersMarkedAsPrimaryKeyAndDBGenerated(mm.Member, this.dbGeneratedIdentity.Member); | ||
|  |                 } | ||
|  |                 this.dbGeneratedIdentity = mm; | ||
|  |             } | ||
|  |             if (mm.IsPrimaryKey && !MappingSystem.IsSupportedIdentityType(mm.Type)) | ||
|  |             { | ||
|  |                 throw Error.IdentityClrTypeNotSupported(mm.DeclaringType, mm.Name, mm.Type); | ||
|  |             } | ||
|  |             if (mm.IsVersion) { | ||
|  |                 if (this.version != null) { | ||
|  |                     throw Error.TwoMembersMarkedAsRowVersion(mm.Member, this.version.Member); | ||
|  |                 } | ||
|  |                 this.version = mm; | ||
|  |             } | ||
|  |             if (mm.IsDiscriminator) { | ||
|  |                 if (this.discriminator != null) { | ||
|  |                     if (!InheritanceRules.AreSameMember(this.discriminator.Member, mm.Member)) { | ||
|  |                         throw Error.TwoMembersMarkedAsInheritanceDiscriminator(mm.Member, this.discriminator.Member); | ||
|  |                     } | ||
|  |                 } | ||
|  |                 else { | ||
|  |                     this.discriminator = mm; | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  |         #endregion | ||
|  |         public override MetaModel Model { | ||
|  |             get { return this.model; } | ||
|  |         } | ||
|  |         public override MetaTable Table { | ||
|  |             get { return this.table; } | ||
|  |         } | ||
|  |         public override Type Type { | ||
|  |             get { return this.type; } | ||
|  |         } | ||
|  |         public override string Name { | ||
|  |             get { return this.type.Name; } | ||
|  |         } | ||
|  |         public override bool IsEntity { | ||
|  |             get { | ||
|  |                 if (this.table != null) { | ||
|  |                     return table.RowType.IdentityMembers.Count > 0; | ||
|  |                 } | ||
|  |                 return false; | ||
|  |             } | ||
|  | 
 | ||
|  |         } | ||
|  |         public override bool CanInstantiate { | ||
|  |             get { return !this.type.IsAbstract && (this == this.InheritanceRoot || this.HasInheritanceCode); } | ||
|  |         } | ||
|  |         public override MetaDataMember DBGeneratedIdentityMember { | ||
|  |             get { return this.dbGeneratedIdentity; } | ||
|  |         } | ||
|  |         public override MetaDataMember VersionMember { | ||
|  |             get { return this.version; } | ||
|  |         } | ||
|  |         public override MetaDataMember Discriminator { | ||
|  |             get { return this.discriminator; } | ||
|  |         } | ||
|  |         public override bool HasUpdateCheck { | ||
|  |             get { | ||
|  |                 foreach (MetaDataMember member in this.PersistentDataMembers) { | ||
|  |                     if (member.UpdateCheck != UpdateCheck.Never) { | ||
|  |                         return true; | ||
|  |                     } | ||
|  |                 } | ||
|  |                 return false; | ||
|  |             } | ||
|  |         } | ||
|  |         public override bool HasInheritance { | ||
|  |             get { return this.inheritanceRoot.HasInheritance; } | ||
|  |         } | ||
|  |         public override object InheritanceCode { | ||
|  |             get { return this.inheritanceCode; } | ||
|  |         } | ||
|  |         public override bool HasInheritanceCode { | ||
|  |             get { return this.InheritanceCode != null; } | ||
|  |         } | ||
|  |         public override bool IsInheritanceDefault { | ||
|  |             get { return this.InheritanceDefault == this; } | ||
|  |         } | ||
|  |         public override MetaType InheritanceDefault { | ||
|  |             get { | ||
|  |                 if (this.inheritanceRoot == this) | ||
|  |                     throw Error.CannotGetInheritanceDefaultFromNonInheritanceClass(); | ||
|  |                 return this.InheritanceRoot.InheritanceDefault; | ||
|  |             } | ||
|  |         } | ||
|  |         public override MetaType InheritanceRoot { | ||
|  |             get { return this.inheritanceRoot; } | ||
|  |         } | ||
|  |         public override MetaType InheritanceBase { | ||
|  |             get { | ||
|  |                 // LOCKING: Cannot initialize at construction | ||
|  |                 if (!this.inheritanceBaseSet && this.inheritanceBase == null) { | ||
|  |                     lock (this.locktarget) { | ||
|  |                         if (this.inheritanceBase == null) { | ||
|  |                             this.inheritanceBase = InheritanceBaseFinder.FindBase(this); | ||
|  |                             this.inheritanceBaseSet = true; | ||
|  |                         } | ||
|  |                     } | ||
|  |                 } | ||
|  |                 return this.inheritanceBase; | ||
|  |             } | ||
|  |         } | ||
|  |         public override ReadOnlyCollection<MetaType> InheritanceTypes { | ||
|  |             get { return this.inheritanceRoot.InheritanceTypes; } | ||
|  |         } | ||
|  |         public override ReadOnlyCollection<MetaType> DerivedTypes { | ||
|  |             get { | ||
|  |                 // LOCKING: Cannot initialize at construction because derived types | ||
|  |                 // won't exist yet. | ||
|  |                 if (this.derivedTypes == null) { | ||
|  |                     lock (this.locktarget) { | ||
|  |                         if (this.derivedTypes == null) { | ||
|  |                             List<MetaType> dTypes = new List<MetaType>(); | ||
|  |                             foreach (MetaType mt in this.InheritanceTypes) { | ||
|  |                                 if (mt.Type.BaseType == this.type) | ||
|  |                                     dTypes.Add(mt); | ||
|  |                             } | ||
|  |                             this.derivedTypes = dTypes.AsReadOnly(); | ||
|  |                         } | ||
|  |                     } | ||
|  |                 } | ||
|  |                 return this.derivedTypes; | ||
|  |             } | ||
|  |         } | ||
|  |         public override MetaType GetInheritanceType(Type inheritanceType) { | ||
|  |             foreach (MetaType mt in this.InheritanceTypes) | ||
|  |                 if (mt.Type == inheritanceType) | ||
|  |                     return mt; | ||
|  |             return null; | ||
|  |         } | ||
|  |         public override MetaType GetTypeForInheritanceCode(object key) { | ||
|  |             if (this.InheritanceRoot.Discriminator.Type == typeof(string)) { | ||
|  |                 string skey = (string)key; | ||
|  |                 foreach (MetaType mt in this.InheritanceRoot.InheritanceTypes) { | ||
|  |                     if (string.Compare((string)mt.InheritanceCode, skey, StringComparison.OrdinalIgnoreCase) == 0) | ||
|  |                         return mt; | ||
|  |                 } | ||
|  |             } | ||
|  |             else { | ||
|  |                 foreach (MetaType mt in this.InheritanceRoot.InheritanceTypes) { | ||
|  |                     if (object.Equals(mt.InheritanceCode, key)) | ||
|  |                         return mt; | ||
|  |                 } | ||
|  |             } | ||
|  |             return null; | ||
|  |         } | ||
|  |         public override ReadOnlyCollection<MetaDataMember> DataMembers { | ||
|  |             get { return this.dataMembers; } | ||
|  |         } | ||
|  |         public override ReadOnlyCollection<MetaDataMember> PersistentDataMembers { | ||
|  |             get { return this.persistentDataMembers; } | ||
|  |         } | ||
|  |         public override ReadOnlyCollection<MetaDataMember> IdentityMembers { | ||
|  |             get { return this.identities; } | ||
|  |         } | ||
|  |         public override ReadOnlyCollection<MetaAssociation> Associations { | ||
|  |             get { | ||
|  |                 // LOCKING: Associations are late-expanded so that cycles are broken. | ||
|  |                 if (this.associations == null) { | ||
|  |                     lock (this.locktarget) { | ||
|  |                         if (this.associations == null) { | ||
|  |                             this.associations = this.dataMembers.Where(m => m.IsAssociation).Select(m => m.Association).ToList().AsReadOnly(); | ||
|  |                         } | ||
|  |                     } | ||
|  |                 } | ||
|  |                 return this.associations; | ||
|  |             } | ||
|  |         } | ||
|  |         public override MetaDataMember GetDataMember(MemberInfo mi) { | ||
|  |             if (mi == null) | ||
|  |                 throw Error.ArgumentNull("mi"); | ||
|  |             MetaDataMember mm; | ||
|  |             if (this.dataMemberMap.TryGetValue(InheritanceRules.DistinguishedMemberName(mi), out mm)) { | ||
|  |                 return mm; | ||
|  |             } else { | ||
|  |                 if (mi.DeclaringType.IsInterface) { | ||
|  |                     throw Error.MappingOfInterfacesMemberIsNotSupported(mi.DeclaringType.Name, mi.Name); | ||
|  |                 } else { //the member is not mapped in the base class | ||
|  |                     throw Error.UnmappedClassMember(mi.DeclaringType.Name, mi.Name); | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         public override MethodInfo OnLoadedMethod { | ||
|  |             get { | ||
|  |                 this.InitMethods(); | ||
|  |                 return this.onLoadedMethod; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         public override MethodInfo OnValidateMethod { | ||
|  |             get { | ||
|  |                 this.InitMethods(); | ||
|  |                 return this.onValidateMethod; | ||
|  |             } | ||
|  |         } | ||
|  |         public override bool HasAnyValidateMethod { | ||
|  |             get { | ||
|  |                 this.InitMethods(); | ||
|  |                 return this.hasAnyValidateMethod; | ||
|  |             } | ||
|  |         } | ||
|  |         public override bool HasAnyLoadMethod { | ||
|  |             get { | ||
|  |                 this.InitMethods(); | ||
|  |                 return this.hasAnyLoadMethod; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         public override string ToString() { | ||
|  |             return this.Name; | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     internal sealed class MappedDataMember : MetaDataMember { | ||
|  |         MetaType declaringType; | ||
|  |         MemberInfo member; | ||
|  |         MemberInfo storageMember; | ||
|  |         int ordinal; | ||
|  |         Type type; | ||
|  |         bool hasAccessors; | ||
|  |         MetaAccessor accPublic; | ||
|  |         MetaAccessor accPrivate; | ||
|  |         MetaAccessor accDefValue; | ||
|  |         MetaAccessor accDefSource; | ||
|  |         MemberMapping memberMap; | ||
|  |         MappedAssociation assoc; | ||
|  |         bool isNullableType; | ||
|  |         bool isDeferred; | ||
|  |         bool isPrimaryKey; | ||
|  |         bool isVersion; | ||
|  |         bool isDBGenerated; | ||
|  |         bool isDiscriminator; | ||
|  |         bool canBeNull = true; | ||
|  |         string dbType; | ||
|  |         string expression; | ||
|  |         string mappedName; | ||
|  |         UpdateCheck updateCheck = UpdateCheck.Never; | ||
|  |         AutoSync autoSync = AutoSync.Never; | ||
|  |         object locktarget = new object(); // Hold locks on private object rather than public MetaType. | ||
|  |         bool hasLoadMethod; | ||
|  |         MethodInfo loadMethod; | ||
|  | 
 | ||
|  |         internal MappedDataMember(MetaType declaringType, MemberInfo mi, MemberMapping map, int ordinal) { | ||
|  |             this.declaringType = declaringType; | ||
|  |             this.member = mi; | ||
|  |             this.ordinal = ordinal; | ||
|  |             this.type = TypeSystem.GetMemberType(mi); | ||
|  |             this.isNullableType = TypeSystem.IsNullableType(this.type); | ||
|  |             this.memberMap = map; | ||
|  |             if (this.memberMap != null && this.memberMap.StorageMemberName != null) { | ||
|  |                 MemberInfo[] mis = mi.DeclaringType.GetMember(this.memberMap.StorageMemberName, BindingFlags.Instance | BindingFlags.NonPublic); | ||
|  |                 if (mis == null || mis.Length != 1) { | ||
|  |                     throw Error.BadStorageProperty(this.memberMap.StorageMemberName, mi.DeclaringType, mi.Name); | ||
|  |                 } | ||
|  |                 this.storageMember = mis[0]; | ||
|  |             } | ||
|  |             Type storageType = this.storageMember != null ? TypeSystem.GetMemberType(this.storageMember) : this.type; | ||
|  |             this.isDeferred = IsDeferredType(storageType); | ||
|  |             ColumnMapping cmap = map as ColumnMapping; | ||
|  |             if (cmap != null && cmap.IsDbGenerated && cmap.IsPrimaryKey) { | ||
|  |                 // auto-gen identities must be synced on insert | ||
|  |                 if ((cmap.AutoSync != AutoSync.Default) && (cmap.AutoSync != AutoSync.OnInsert)) { | ||
|  |                     throw Error.IncorrectAutoSyncSpecification(mi.Name); | ||
|  |                 } | ||
|  |             } | ||
|  |             if (cmap != null) { | ||
|  |                 this.isPrimaryKey = cmap.IsPrimaryKey; | ||
|  |                 this.isVersion = cmap.IsVersion; | ||
|  |                 this.isDBGenerated = cmap.IsDbGenerated || !string.IsNullOrEmpty(cmap.Expression) || this.isVersion; | ||
|  |                 this.isDiscriminator = cmap.IsDiscriminator; | ||
|  |                 this.canBeNull = cmap.CanBeNull == null ? this.isNullableType || !this.type.IsValueType : (bool)cmap.CanBeNull; | ||
|  |                 this.dbType = cmap.DbType; | ||
|  |                 this.expression = cmap.Expression; | ||
|  |                 this.updateCheck = cmap.UpdateCheck; | ||
|  |                 // auto-gen keys are always and only synced on insert | ||
|  |                 if (this.IsDbGenerated && this.IsPrimaryKey) { | ||
|  |                     this.autoSync = AutoSync.OnInsert; | ||
|  |                 } | ||
|  |                 else if (cmap.AutoSync != AutoSync.Default) { | ||
|  |                     // if the user has explicitly set it, use their value | ||
|  |                     this.autoSync = cmap.AutoSync; | ||
|  |                 } | ||
|  |                 else if (this.IsDbGenerated) { | ||
|  |                     // database generated members default to always | ||
|  |                     this.autoSync = AutoSync.Always; | ||
|  |                 } | ||
|  |             } | ||
|  |             this.mappedName = this.memberMap.DbName != null ? this.memberMap.DbName : this.member.Name; | ||
|  |         } | ||
|  |         private void InitAccessors() { | ||
|  |             if (!this.hasAccessors) { | ||
|  |                 lock (this.locktarget) { | ||
|  |                     if (!this.hasAccessors) { | ||
|  |                         if (this.storageMember != null) { | ||
|  |                             this.accPrivate = MakeMemberAccessor(this.member.ReflectedType, this.storageMember, null); | ||
|  |                             if (this.isDeferred) { | ||
|  |                                 MakeDeferredAccessors(this.member.ReflectedType, this.accPrivate, out this.accPrivate, out this.accDefValue, out this.accDefSource); | ||
|  |                             } | ||
|  |                             this.accPublic = MakeMemberAccessor(this.member.ReflectedType, this.member, this.accPrivate); | ||
|  |                         } | ||
|  |                         else { | ||
|  |                             this.accPublic = this.accPrivate = MakeMemberAccessor(this.member.ReflectedType, this.member, null); | ||
|  |                             if (this.isDeferred) { | ||
|  |                                 MakeDeferredAccessors(this.member.ReflectedType, this.accPrivate, out this.accPrivate, out this.accDefValue, out this.accDefSource); | ||
|  |                             } | ||
|  |                         } | ||
|  |                         this.hasAccessors = true; | ||
|  |                     } | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  |         public override MetaType DeclaringType { | ||
|  |             get { return this.declaringType; } | ||
|  |         } | ||
|  |         public override bool IsDeclaredBy(MetaType metaType) { | ||
|  |             if (metaType == null) { | ||
|  |                 throw Error.ArgumentNull("metaType"); | ||
|  |             } | ||
|  |             return metaType.Type == this.member.DeclaringType; | ||
|  |         } | ||
|  |         public override MemberInfo Member { | ||
|  |             get { return this.member; } | ||
|  |         } | ||
|  |         public override MemberInfo StorageMember { | ||
|  |             get { return this.storageMember; } | ||
|  |         } | ||
|  |         public override string Name { | ||
|  |             get { return this.member.Name; } | ||
|  |         } | ||
|  |         public override int Ordinal { | ||
|  |             get { return this.ordinal; } | ||
|  |         } | ||
|  |         public override Type Type { | ||
|  |             get { return this.type; } | ||
|  |         } | ||
|  |         public override MetaAccessor MemberAccessor { | ||
|  |             get { | ||
|  |                 this.InitAccessors(); | ||
|  |                 return this.accPublic; | ||
|  |             } | ||
|  |         } | ||
|  |         public override MetaAccessor StorageAccessor { | ||
|  |             get {  | ||
|  |                 this.InitAccessors(); | ||
|  |                 return this.accPrivate;  | ||
|  |             } | ||
|  |         } | ||
|  |         public override MetaAccessor DeferredValueAccessor { | ||
|  |             get { | ||
|  |                 this.InitAccessors(); | ||
|  |                 return this.accDefValue; | ||
|  |             } | ||
|  |         } | ||
|  |         public override MetaAccessor DeferredSourceAccessor { | ||
|  |             get { | ||
|  |                 this.InitAccessors(); | ||
|  |                 return this.accDefSource; | ||
|  |             } | ||
|  |         } | ||
|  |         public override bool IsDeferred { | ||
|  |             get { return this.isDeferred; } | ||
|  |         } | ||
|  |         public override bool IsPersistent { | ||
|  |             get { return this.memberMap != null; } | ||
|  |         } | ||
|  |         public override bool IsAssociation { | ||
|  |             get { return this.memberMap is AssociationMapping; } | ||
|  |         } | ||
|  |         public override bool IsPrimaryKey { | ||
|  |             get { return this.isPrimaryKey; } | ||
|  |         } | ||
|  |         /// <summary> | ||
|  |         /// Returns true if the member is explicitly marked as auto gen, or if the | ||
|  |         /// member is computed or generated by the database server. | ||
|  |         /// </summary> | ||
|  |         public override bool IsDbGenerated { | ||
|  |             get { return this.isDBGenerated; } | ||
|  |         } | ||
|  |         public override bool IsVersion { | ||
|  |             get { return this.isVersion; } | ||
|  |         } | ||
|  |         public override bool IsDiscriminator { | ||
|  |             get { return this.isDiscriminator; } | ||
|  |         } | ||
|  |         public override bool CanBeNull { | ||
|  |             get { return this.canBeNull; } | ||
|  |         } | ||
|  |         public override string DbType { | ||
|  |             get { return this.dbType; } | ||
|  |         } | ||
|  |         public override string Expression { | ||
|  |             get { return this.expression; } | ||
|  |         } | ||
|  |         public override string MappedName { | ||
|  |             get { return this.mappedName; } | ||
|  |         } | ||
|  |         public override UpdateCheck UpdateCheck { | ||
|  |             get { return this.updateCheck; } | ||
|  |         } | ||
|  |         public override AutoSync AutoSync { | ||
|  |             get { return this.autoSync; } | ||
|  |         } | ||
|  |         public override MetaAssociation Association { | ||
|  |             get { | ||
|  |                 if (this.IsAssociation) { | ||
|  |                     // LOCKING: This deferral isn't an optimization. It can't be done in the constructor | ||
|  |                     // because there may be loops in the association graph. | ||
|  |                     if (this.assoc == null) { | ||
|  |                         lock (this.locktarget) { | ||
|  |                             if (this.assoc == null) { | ||
|  |                                 this.assoc = new MappedAssociation(this, (AssociationMapping)this.memberMap); | ||
|  |                             } | ||
|  |                         } | ||
|  |                     } | ||
|  |                 } | ||
|  |                 return this.assoc; | ||
|  |             } | ||
|  |         } | ||
|  |         public override MethodInfo LoadMethod { | ||
|  |             get { | ||
|  |                 if (this.hasLoadMethod == false && this.IsDeferred) { | ||
|  |                     // defer searching for this access method until we really need to know | ||
|  |                     this.loadMethod = MethodFinder.FindMethod( | ||
|  |                         ((MappedMetaModel)this.declaringType.Model).ContextType, | ||
|  |                         "Load" + this.member.Name, | ||
|  |                         BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic, | ||
|  |                         new Type[] { this.DeclaringType.Type } | ||
|  |                         ); | ||
|  |                     this.hasLoadMethod = true; | ||
|  |                 } | ||
|  |                 return this.loadMethod; | ||
|  |             } | ||
|  |         } | ||
|  |         private bool IsDeferredType(Type clrType) { | ||
|  |             if (clrType == null || clrType == typeof(object)) { | ||
|  |                 return false; | ||
|  |             } | ||
|  |             if (clrType.IsGenericType) { | ||
|  |                 Type gtype = clrType.GetGenericTypeDefinition(); | ||
|  |                 return gtype == typeof(Link<>) || | ||
|  |                     typeof(EntitySet<>).IsAssignableFrom(gtype) || | ||
|  |                     typeof(EntityRef<>).IsAssignableFrom(gtype) || | ||
|  |                     IsDeferredType(clrType.BaseType); | ||
|  |             } | ||
|  |             return false; | ||
|  |         } | ||
|  |         private static MetaAccessor MakeMemberAccessor(Type accessorType, MemberInfo mi, MetaAccessor storage) { | ||
|  |             FieldInfo fi = mi as FieldInfo; | ||
|  |             MetaAccessor acc = null; | ||
|  |             if (fi != null) { | ||
|  |                 acc = FieldAccessor.Create(accessorType, fi); | ||
|  |             } | ||
|  |             else { | ||
|  |                 PropertyInfo pi = (PropertyInfo)mi; | ||
|  |                 acc = PropertyAccessor.Create(accessorType, pi, storage); | ||
|  |             } | ||
|  |             return acc; | ||
|  |         } | ||
|  |         [MethodImpl(MethodImplOptions.NoInlining | MethodImplOptions.NoOptimization)] | ||
|  |         private static void MakeDeferredAccessors( | ||
|  |             Type declaringType, MetaAccessor accessor, | ||
|  |             out MetaAccessor accessorValue, out MetaAccessor accessorDeferredValue, out MetaAccessor accessorDeferredSource | ||
|  |             ) { | ||
|  |             if (accessor.Type.IsGenericType) { | ||
|  |                 Type gtype = accessor.Type.GetGenericTypeDefinition(); | ||
|  |                 Type itemType = accessor.Type.GetGenericArguments()[0]; | ||
|  |                 if (gtype == typeof(Link<>)) { | ||
|  |                     accessorValue = CreateAccessor(typeof(LinkValueAccessor<,>).MakeGenericType(declaringType, itemType), accessor); | ||
|  |                     accessorDeferredValue = CreateAccessor(typeof(LinkDefValueAccessor<,>).MakeGenericType(declaringType, itemType), accessor); | ||
|  |                     accessorDeferredSource = CreateAccessor(typeof(LinkDefSourceAccessor<,>).MakeGenericType(declaringType, itemType), accessor); | ||
|  |                     return; | ||
|  |                 } | ||
|  |                 else if (typeof(EntityRef<>).IsAssignableFrom(gtype)) { | ||
|  |                     accessorValue = CreateAccessor(typeof(EntityRefValueAccessor<,>).MakeGenericType(declaringType, itemType), accessor); | ||
|  |                     accessorDeferredValue = CreateAccessor(typeof(EntityRefDefValueAccessor<,>).MakeGenericType(declaringType, itemType), accessor); | ||
|  |                     accessorDeferredSource = CreateAccessor(typeof(EntityRefDefSourceAccessor<,>).MakeGenericType(declaringType, itemType), accessor); | ||
|  |                     return; | ||
|  |                 } | ||
|  |                 else if (typeof(EntitySet<>).IsAssignableFrom(gtype)) { | ||
|  |                     accessorValue = CreateAccessor(typeof(EntitySetValueAccessor<,>).MakeGenericType(declaringType, itemType), accessor); | ||
|  |                     accessorDeferredValue = CreateAccessor(typeof(EntitySetDefValueAccessor<,>).MakeGenericType(declaringType, itemType), accessor); | ||
|  |                     accessorDeferredSource = CreateAccessor(typeof(EntitySetDefSourceAccessor<,>).MakeGenericType(declaringType, itemType), accessor); | ||
|  |                     return; | ||
|  |                 } | ||
|  |             } | ||
|  |             throw Error.UnhandledDeferredStorageType(accessor.Type); | ||
|  |         } | ||
|  |         private static MetaAccessor CreateAccessor(Type accessorType, params object[] args) { | ||
|  |             return (MetaAccessor)Activator.CreateInstance(accessorType, BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic, null, args, null); | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     internal class MappedAssociation : MetaAssociationImpl { | ||
|  |         MappedDataMember thisMember; | ||
|  |         MetaDataMember otherMember; | ||
|  |         MetaType otherType; | ||
|  |         ReadOnlyCollection<MetaDataMember> thisKey; | ||
|  |         ReadOnlyCollection<MetaDataMember> otherKey; | ||
|  |         bool isMany; | ||
|  |         bool isForeignKey; | ||
|  |         bool isNullable; | ||
|  |         bool thisKeyIsPrimaryKey; | ||
|  |         bool otherKeyIsPrimaryKey; | ||
|  |         AssociationMapping assocMap; | ||
|  | 
 | ||
|  |         internal MappedAssociation(MappedDataMember mm, AssociationMapping assocMap) { | ||
|  |             this.thisMember = mm; | ||
|  |             this.assocMap = assocMap; | ||
|  |             this.Init(); | ||
|  |             this.InitOther(); | ||
|  |             //validate the number of ThisKey columns is the same as the number of OtherKey columns | ||
|  |             if (this.thisKey.Count != this.otherKey.Count && this.thisKey.Count > 0 && this.otherKey.Count > 0) { | ||
|  |                 throw Error.MismatchedThisKeyOtherKey(thisMember.Name, thisMember.DeclaringType.Name); | ||
|  |             } | ||
|  |         } | ||
|  |         #region Initialization | ||
|  |         private void Init() { | ||
|  |             this.isMany = TypeSystem.IsSequenceType(this.thisMember.Type); | ||
|  |             this.thisKey = (this.assocMap.ThisKey != null) | ||
|  |                 ? MakeKeys(this.thisMember.DeclaringType, this.assocMap.ThisKey) | ||
|  |                 : this.thisMember.DeclaringType.IdentityMembers; | ||
|  |             // this association refers to the parent if thisKey is not our own identity | ||
|  |             this.thisKeyIsPrimaryKey = AreEqual(this.thisKey, this.thisMember.DeclaringType.IdentityMembers); | ||
|  |             this.isForeignKey = this.assocMap.IsForeignKey; | ||
|  | 
 | ||
|  |             // if any key members are not nullable, the association is not nullable | ||
|  |             this.isNullable = true; | ||
|  |             foreach (MetaDataMember mm in this.thisKey) { | ||
|  |                 if (mm == null) | ||
|  |                     throw Error.UnexpectedNull("MetaDataMember"); | ||
|  | 
 | ||
|  |                 if (!mm.CanBeNull) { | ||
|  |                     this.isNullable = false; | ||
|  |                     break; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             // validate DeleteOnNull specification | ||
|  |             if (assocMap.DeleteOnNull == true) { | ||
|  |                 if (!(isForeignKey && !isMany && !isNullable)) { | ||
|  |                     throw Error.InvalidDeleteOnNullSpecification(thisMember); | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  |         private void InitOther() { | ||
|  |             if (this.otherType == null) { | ||
|  |                 Type ot = this.isMany ? TypeSystem.GetElementType(this.thisMember.Type) : this.thisMember.Type; | ||
|  |                 this.otherType = this.thisMember.DeclaringType.Model.GetMetaType(ot); | ||
|  |                 System.Diagnostics.Debug.Assert(this.otherType.IsEntity); | ||
|  |                 this.otherKey = (assocMap.OtherKey != null) | ||
|  |                     ? MakeKeys(this.otherType, this.assocMap.OtherKey) | ||
|  |                     : this.otherType.IdentityMembers; | ||
|  |                 this.otherKeyIsPrimaryKey = AreEqual(this.otherKey, this.otherType.IdentityMembers); | ||
|  |                 foreach (MetaDataMember omm in this.otherType.DataMembers) { | ||
|  |                     if (omm.IsAssociation && omm != this.thisMember && omm.MappedName == this.thisMember.MappedName) { | ||
|  |                         this.otherMember = omm; | ||
|  |                         break; | ||
|  |                     } | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  |         #endregion | ||
|  |         public override MetaDataMember ThisMember { | ||
|  |             get { return this.thisMember; } | ||
|  |         } | ||
|  |         public override ReadOnlyCollection<MetaDataMember> ThisKey { | ||
|  |             get { return this.thisKey; } | ||
|  |         } | ||
|  |         public override MetaDataMember OtherMember { | ||
|  |             get { return this.otherMember; } | ||
|  |         } | ||
|  |         public override ReadOnlyCollection<MetaDataMember> OtherKey { | ||
|  |             get { return this.otherKey; } | ||
|  |         } | ||
|  |         public override MetaType OtherType { | ||
|  |             get { return this.otherType; } | ||
|  |         } | ||
|  |         public override bool IsMany { | ||
|  |             get { return this.isMany; } | ||
|  |         } | ||
|  |         public override bool IsForeignKey { | ||
|  |             get { return this.isForeignKey; } | ||
|  |         } | ||
|  |         public override bool IsUnique { | ||
|  |             get { return this.assocMap.IsUnique; } | ||
|  |         } | ||
|  |         public override bool IsNullable { | ||
|  |             get { return this.isNullable; } | ||
|  |         } | ||
|  |         public override bool ThisKeyIsPrimaryKey { | ||
|  |             get { return this.thisKeyIsPrimaryKey; } | ||
|  |         } | ||
|  |         public override bool OtherKeyIsPrimaryKey { | ||
|  |             get { return this.otherKeyIsPrimaryKey; } | ||
|  |         } | ||
|  |         public override string DeleteRule { | ||
|  |             get { | ||
|  |                 return this.assocMap.DeleteRule; | ||
|  |             } | ||
|  |         } | ||
|  |         public override bool DeleteOnNull { | ||
|  |             get { | ||
|  |                 return this.assocMap.DeleteOnNull; | ||
|  |             } | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     class MappedFunction : MetaFunction { | ||
|  |         MetaModel model; | ||
|  |         FunctionMapping map; | ||
|  |         MethodInfo method; | ||
|  |         ReadOnlyCollection<MetaParameter> parameters; | ||
|  |         MetaParameter returnParameter; | ||
|  |         ReadOnlyCollection<MetaType> rowTypes; | ||
|  |         static ReadOnlyCollection<MetaParameter> _emptyParameters = new List<MetaParameter>(0).AsReadOnly(); | ||
|  |         static ReadOnlyCollection<MetaType> _emptyTypes = new List<MetaType>(0).AsReadOnly(); | ||
|  | 
 | ||
|  |         [ResourceExposure(ResourceScope.Assembly | ResourceScope.Machine)] // map parameter contains type names. | ||
|  |         [ResourceConsumption(ResourceScope.Assembly | ResourceScope.Machine)] // FindType method call. | ||
|  |         internal MappedFunction(MappedMetaModel model, FunctionMapping map, MethodInfo method) { | ||
|  |             this.model = model; | ||
|  |             this.map = map; | ||
|  |             this.method = method; | ||
|  |             this.rowTypes = _emptyTypes; | ||
|  | 
 | ||
|  |             if (map.Types.Count == 0 && this.method.ReturnType == typeof(IMultipleResults)) { | ||
|  |                 throw Error.NoResultTypesDeclaredForFunction(method.Name); | ||
|  |             } | ||
|  |             else if (map.Types.Count > 1 && this.method.ReturnType != typeof(IMultipleResults)) { | ||
|  |                 throw Error.TooManyResultTypesDeclaredForFunction(method.Name); | ||
|  |             } | ||
|  |             else if (map.Types.Count == 1 && this.method.ReturnType != typeof(IMultipleResults)) { | ||
|  |                 Type elementType = TypeSystem.GetElementType(method.ReturnType); | ||
|  |                 this.rowTypes = new List<MetaType>(1) { this.GetMetaType(map.Types[0], elementType) }.AsReadOnly(); | ||
|  |             } | ||
|  |             else if (map.Types.Count > 0) { | ||
|  |                 List<MetaType> rowTypes = new List<MetaType>(); | ||
|  |                 foreach (TypeMapping rtm in map.Types) { | ||
|  |                     Type elementType = model.FindType(rtm.Name); | ||
|  |                     if (elementType == null) { | ||
|  |                         throw Error.CouldNotFindElementTypeInModel(rtm.Name); | ||
|  |                     } | ||
|  |                     MetaType mt = this.GetMetaType(rtm, elementType); | ||
|  |                     // Only add unique meta types | ||
|  |                     if (!rowTypes.Contains(mt)) { | ||
|  |                         rowTypes.Add(mt); | ||
|  |                     } | ||
|  |                 } | ||
|  |                 this.rowTypes = rowTypes.AsReadOnly(); | ||
|  |             } | ||
|  |             else if (map.FunReturn != null) { | ||
|  |                 this.returnParameter = new MappedReturnParameter(method.ReturnParameter, map.FunReturn); | ||
|  |             } | ||
|  | 
 | ||
|  |             // Parameters. | ||
|  |             ParameterInfo[] pis = this.method.GetParameters(); | ||
|  |             if (pis.Length > 0) { | ||
|  |                 List<MetaParameter> mps = new List<MetaParameter>(pis.Length); | ||
|  |                 if (this.map.Parameters.Count != pis.Length) { | ||
|  |                     throw Error.IncorrectNumberOfParametersMappedForMethod(this.map.MethodName); | ||
|  |                 } | ||
|  |                 for (int i = 0; i < pis.Length; i++) { | ||
|  |                     mps.Add(new MappedParameter(pis[i], this.map.Parameters[i])); | ||
|  |                 } | ||
|  |                 this.parameters = mps.AsReadOnly(); | ||
|  |             } | ||
|  |             else { | ||
|  |                 this.parameters = _emptyParameters; | ||
|  |             } | ||
|  |         } | ||
|  |         /// <summary> | ||
|  |         /// For the specified type, if it is a mapped type, use the Table | ||
|  |         /// metatype to get the correct inheritance metatype, | ||
|  |         /// otherwise create a new meta type. | ||
|  |         [ResourceExposure(ResourceScope.Assembly | ResourceScope.Machine)] // Parameter contains various type references. | ||
|  |         [ResourceConsumption(ResourceScope.Assembly | ResourceScope.Machine)] // MappedRootType constructor call.         | ||
|  |         private MetaType GetMetaType(TypeMapping tm, Type elementType) { | ||
|  |             MetaTable tbl = model.GetTable(elementType); | ||
|  |             if (tbl != null) { | ||
|  |                 return tbl.RowType.GetInheritanceType(elementType); | ||
|  |             } | ||
|  |             return new MappedRootType((MappedMetaModel)model, null, tm, elementType); | ||
|  |         } | ||
|  |         public override ReadOnlyCollection<MetaParameter> Parameters { | ||
|  |             get { return this.parameters; } | ||
|  |         } | ||
|  |         public override string MappedName { | ||
|  |             get { return this.map.Name; } | ||
|  |         } | ||
|  |         public override MethodInfo Method { | ||
|  |             get { return this.method; } | ||
|  |         } | ||
|  |         public override MetaModel Model { | ||
|  |             get { return this.model; } | ||
|  |         } | ||
|  |         public override string Name { | ||
|  |             get { return this.method.Name; } | ||
|  |         } | ||
|  |         public override bool IsComposable { | ||
|  |             get { return this.map.IsComposable; } | ||
|  |         } | ||
|  |         public override MetaParameter ReturnParameter { | ||
|  |             get { return this.returnParameter; } | ||
|  |         } | ||
|  |         public override bool HasMultipleResults { | ||
|  |             get { return this.method.ReturnType == typeof(IMultipleResults); } | ||
|  |         } | ||
|  |         public override ReadOnlyCollection<MetaType> ResultRowTypes { | ||
|  |             get { return this.rowTypes; } | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     internal sealed class MappedParameter : MetaParameter { | ||
|  |         private ParameterInfo parameterInfo; | ||
|  |         private ParameterMapping map; | ||
|  | 
 | ||
|  |         public MappedParameter(ParameterInfo parameterInfo, ParameterMapping map) { | ||
|  |             this.parameterInfo = parameterInfo; | ||
|  |             this.map = map; | ||
|  |         } | ||
|  |         public override ParameterInfo Parameter { | ||
|  |             get { return this.parameterInfo; } | ||
|  |         } | ||
|  |         public override string Name { | ||
|  |             get { return this.parameterInfo.Name; } | ||
|  |         } | ||
|  |         public override string MappedName { | ||
|  |             get { return this.map.Name; } | ||
|  |         } | ||
|  |         public override Type ParameterType { | ||
|  |             get { return this.parameterInfo.ParameterType; } | ||
|  |         } | ||
|  |         public override string DbType { | ||
|  |             get { return this.map.DbType; } | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     internal sealed class MappedReturnParameter : MetaParameter { | ||
|  |         private ParameterInfo parameterInfo; | ||
|  |         private ReturnMapping map; | ||
|  | 
 | ||
|  |         public MappedReturnParameter(ParameterInfo parameterInfo, ReturnMapping map) { | ||
|  |             this.parameterInfo = parameterInfo; | ||
|  |             this.map = map; | ||
|  |         } | ||
|  |         public override ParameterInfo Parameter { | ||
|  |             get { return this.parameterInfo; } | ||
|  |         } | ||
|  |         public override string Name { | ||
|  |             get { return null; } | ||
|  |         } | ||
|  |         public override string MappedName { | ||
|  |             get { return null; } | ||
|  |         } | ||
|  |         public override Type ParameterType { | ||
|  |             get { return this.parameterInfo.ParameterType; } | ||
|  |         } | ||
|  |         public override string DbType { | ||
|  |             get { return this.map.DbType; } | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     internal abstract class MetaAssociationImpl : MetaAssociation { | ||
|  | 
 | ||
|  |         private static char[] keySeparators = new char[] { ',' }; | ||
|  |         /// <summary> | ||
|  |         /// Given a MetaType and a set of key fields, return the set of MetaDataMembers | ||
|  |         /// corresponding to the key. | ||
|  |         /// </summary> | ||
|  |         protected static ReadOnlyCollection<MetaDataMember> MakeKeys(MetaType mtype, string keyFields) { | ||
|  |             string[] names = keyFields.Split(keySeparators); | ||
|  |             MetaDataMember[] members = new MetaDataMember[names.Length]; | ||
|  |             for (int i = 0; i < names.Length; i++) { | ||
|  |                 names[i] = names[i].Trim(); | ||
|  |                 MemberInfo[] rmis = mtype.Type.GetMember(names[i], BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic); | ||
|  |                 if (rmis == null || rmis.Length != 1) { | ||
|  |                     throw Error.BadKeyMember(names[i], keyFields, mtype.Name); | ||
|  |                 } | ||
|  |                 members[i] = mtype.GetDataMember(rmis[0]); | ||
|  |                 if (members[i] == null) { | ||
|  |                     throw Error.BadKeyMember(names[i], keyFields, mtype.Name); | ||
|  |                 } | ||
|  |             } | ||
|  |             return new List<MetaDataMember>(members).AsReadOnly(); | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// Compare two sets of keys for equality. | ||
|  |         /// </summary> | ||
|  |         protected static bool AreEqual(IEnumerable<MetaDataMember> key1, IEnumerable<MetaDataMember> key2) { | ||
|  |             using (IEnumerator<MetaDataMember> e1 = key1.GetEnumerator()) { | ||
|  |                 using (IEnumerator<MetaDataMember> e2 = key2.GetEnumerator()) { | ||
|  |                     bool m1, m2; | ||
|  |                     for (m1 = e1.MoveNext(), m2 = e2.MoveNext(); m1 && m2; m1 = e1.MoveNext(), m2 = e2.MoveNext()) { | ||
|  |                         if (e1.Current != e2.Current) | ||
|  |                             return false; | ||
|  |                     } | ||
|  |                     if (m1 != m2) | ||
|  |                         return false; | ||
|  |                 } | ||
|  |             } | ||
|  |             return true; | ||
|  |         } | ||
|  | 
 | ||
|  |         public override string ToString() { | ||
|  |             return string.Format(Globalization.CultureInfo.InvariantCulture, "{0} ->{1} {2}", ThisMember.DeclaringType.Name, IsMany ? "*" : "", OtherType.Name); | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     internal sealed class UnmappedType : MetaType { | ||
|  |         MetaModel model; | ||
|  |         Type type; | ||
|  |         Dictionary<object, MetaDataMember> dataMemberMap; | ||
|  |         ReadOnlyCollection<MetaDataMember> dataMembers; | ||
|  |         ReadOnlyCollection<MetaType> inheritanceTypes; | ||
|  |         object locktarget = new object(); // Hold locks on private object rather than public MetaType. | ||
|  | 
 | ||
|  |         private static ReadOnlyCollection<MetaType> _emptyTypes = new List<MetaType>().AsReadOnly(); | ||
|  |         private static ReadOnlyCollection<MetaDataMember> _emptyDataMembers = new List<MetaDataMember>().AsReadOnly(); | ||
|  |         private static ReadOnlyCollection<MetaAssociation> _emptyAssociations = new List<MetaAssociation>().AsReadOnly(); | ||
|  | 
 | ||
|  |         internal UnmappedType(MetaModel model, Type type) { | ||
|  |             this.model = model; | ||
|  |             this.type = type; | ||
|  |         } | ||
|  | 
 | ||
|  |         public override MetaModel Model { | ||
|  |             get { return this.model; } | ||
|  |         } | ||
|  |         public override MetaTable Table { | ||
|  |             get { return null; } | ||
|  |         } | ||
|  |         public override Type Type { | ||
|  |             get { return this.type; } | ||
|  |         } | ||
|  |         public override string Name { | ||
|  |             get { return this.type.Name; } | ||
|  |         } | ||
|  |         public override bool IsEntity { | ||
|  |             get { return false; } | ||
|  |         } | ||
|  |         public override bool CanInstantiate { | ||
|  |             get { return !this.type.IsAbstract; } | ||
|  |         } | ||
|  |         public override MetaDataMember DBGeneratedIdentityMember { | ||
|  |             get { return null; } | ||
|  |         } | ||
|  |         public override MetaDataMember VersionMember { | ||
|  |             get { return null; } | ||
|  |         } | ||
|  |         public override MetaDataMember Discriminator { | ||
|  |             get { return null; } | ||
|  |         } | ||
|  |         public override bool HasUpdateCheck { | ||
|  |             get { return false; } | ||
|  |         } | ||
|  |         public override ReadOnlyCollection<MetaType> InheritanceTypes { | ||
|  |             get { | ||
|  |                 if (this.inheritanceTypes == null) { | ||
|  |                     lock (this.locktarget) { | ||
|  |                         if (this.inheritanceTypes == null) { | ||
|  |                             this.inheritanceTypes = new MetaType[] { this }.ToList().AsReadOnly(); | ||
|  |                         } | ||
|  |                     } | ||
|  |                 } | ||
|  |                 return this.inheritanceTypes; | ||
|  |             } | ||
|  |         } | ||
|  |         public override MetaType GetInheritanceType(Type inheritanceType) { | ||
|  |             if (inheritanceType == this.type) | ||
|  |                 return this; | ||
|  |             return null; | ||
|  |         } | ||
|  |         public override ReadOnlyCollection<MetaType> DerivedTypes { | ||
|  |             get { return _emptyTypes; } | ||
|  |         } | ||
|  |         public override MetaType GetTypeForInheritanceCode(object key) { | ||
|  |             return null; | ||
|  |         } | ||
|  |         public override bool HasInheritance { | ||
|  |             get { return false; } | ||
|  |         } | ||
|  |         public override bool HasInheritanceCode { | ||
|  |             get { return false; } | ||
|  |         } | ||
|  |         public override object InheritanceCode { | ||
|  |             get { return null; } | ||
|  |         } | ||
|  |         public override MetaType InheritanceRoot { | ||
|  |             get { return this; } | ||
|  |         } | ||
|  |         public override MetaType InheritanceBase { | ||
|  |             get { return null; } | ||
|  |         } | ||
|  |         public override MetaType InheritanceDefault { | ||
|  |             get { return null; } | ||
|  |         } | ||
|  |         public override bool IsInheritanceDefault { | ||
|  |             get { return false; } | ||
|  |         } | ||
|  |         public override ReadOnlyCollection<MetaDataMember> DataMembers { | ||
|  |             get { | ||
|  |                 this.InitDataMembers(); | ||
|  |                 return this.dataMembers; | ||
|  |             } | ||
|  |         } | ||
|  |         public override ReadOnlyCollection<MetaDataMember> PersistentDataMembers { | ||
|  |             get { return _emptyDataMembers; } | ||
|  |         } | ||
|  |         public override ReadOnlyCollection<MetaDataMember> IdentityMembers { | ||
|  |             get { | ||
|  |                 this.InitDataMembers(); | ||
|  |                 return this.dataMembers; | ||
|  |             } | ||
|  |         } | ||
|  |         public override ReadOnlyCollection<MetaAssociation> Associations { | ||
|  |             get { return _emptyAssociations; } | ||
|  |         } | ||
|  |         public override MetaDataMember GetDataMember(MemberInfo mi) { | ||
|  |             if (mi == null) | ||
|  |                 throw Error.ArgumentNull("mi"); | ||
|  |             this.InitDataMembers(); | ||
|  |             if (this.dataMemberMap == null) { | ||
|  |                 lock (this.locktarget) { | ||
|  |                     if (this.dataMemberMap == null) { | ||
|  |                         Dictionary<object, MetaDataMember> map = new Dictionary<object, MetaDataMember>(); | ||
|  |                         foreach (MetaDataMember mm in this.dataMembers) { | ||
|  |                             map.Add(InheritanceRules.DistinguishedMemberName(mm.Member), mm); | ||
|  |                         } | ||
|  |                         this.dataMemberMap = map; | ||
|  |                     } | ||
|  |                 } | ||
|  |             } | ||
|  |             object dn = InheritanceRules.DistinguishedMemberName(mi); | ||
|  |             MetaDataMember mdm; | ||
|  |             this.dataMemberMap.TryGetValue(dn, out mdm); | ||
|  |             return mdm; | ||
|  |         } | ||
|  | 
 | ||
|  |         private void InitDataMembers() { | ||
|  |             if (this.dataMembers == null) { | ||
|  |                 lock (this.locktarget) { | ||
|  |                     if (this.dataMembers == null) { | ||
|  |                         List<MetaDataMember> dMembers = new List<MetaDataMember>(); | ||
|  |                         int ordinal = 0; | ||
|  |                         BindingFlags flags = BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.FlattenHierarchy; | ||
|  |                         foreach (FieldInfo fi in this.type.GetFields(flags)) { | ||
|  |                             MetaDataMember mm = new UnmappedDataMember(this, fi, ordinal); | ||
|  |                             dMembers.Add(mm); | ||
|  |                             ordinal++; | ||
|  |                         } | ||
|  |                         foreach (PropertyInfo pi in this.type.GetProperties(flags)) { | ||
|  |                             MetaDataMember mm = new UnmappedDataMember(this, pi, ordinal); | ||
|  |                             dMembers.Add(mm); | ||
|  |                             ordinal++; | ||
|  |                         } | ||
|  |                         this.dataMembers = dMembers.AsReadOnly(); | ||
|  |                     } | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         public override string ToString() { | ||
|  |             return this.Name; | ||
|  |         } | ||
|  | 
 | ||
|  |         public override MethodInfo OnLoadedMethod { | ||
|  |             get { return null; } | ||
|  |         } | ||
|  | 
 | ||
|  |         public override MethodInfo OnValidateMethod { | ||
|  |             get { return null; } | ||
|  |         } | ||
|  |         public override bool HasAnyValidateMethod { | ||
|  |             get { | ||
|  |                 return false; | ||
|  |             } | ||
|  |         } | ||
|  |         public override bool HasAnyLoadMethod { | ||
|  |             get { | ||
|  |                 return false; | ||
|  |             } | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     internal sealed class UnmappedDataMember : MetaDataMember { | ||
|  |         MetaType declaringType; | ||
|  |         MemberInfo member; | ||
|  |         int ordinal; | ||
|  |         Type type; | ||
|  |         MetaAccessor accPublic; | ||
|  |         object lockTarget = new object(); | ||
|  | 
 | ||
|  |         internal UnmappedDataMember(MetaType declaringType, MemberInfo mi, int ordinal) { | ||
|  |             this.declaringType = declaringType; | ||
|  |             this.member = mi; | ||
|  |             this.ordinal = ordinal; | ||
|  |             this.type = TypeSystem.GetMemberType(mi); | ||
|  |         } | ||
|  |         private void InitAccessors() { | ||
|  |             if (this.accPublic == null) { | ||
|  |                 lock (this.lockTarget) { | ||
|  |                     if (this.accPublic == null) { | ||
|  |                         this.accPublic = MakeMemberAccessor(this.member.ReflectedType, this.member); | ||
|  |                     } | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  |         public override MetaType DeclaringType { | ||
|  |             get { return this.declaringType; } | ||
|  |         } | ||
|  |         public override bool IsDeclaredBy(MetaType metaType) { | ||
|  |             if (metaType == null) { | ||
|  |                 throw Error.ArgumentNull("metaType"); | ||
|  |             } | ||
|  |             return metaType.Type == this.member.DeclaringType; | ||
|  |         } | ||
|  |         public override MemberInfo Member { | ||
|  |             get { return this.member; } | ||
|  |         } | ||
|  |         public override MemberInfo StorageMember { | ||
|  |             get { return this.member; } | ||
|  |         } | ||
|  |         public override string Name { | ||
|  |             get { return this.member.Name; } | ||
|  |         } | ||
|  |         public override int Ordinal { | ||
|  |             get { return this.ordinal; } | ||
|  |         } | ||
|  |         public override Type Type { | ||
|  |             get { return this.type; } | ||
|  |         } | ||
|  |         public override MetaAccessor MemberAccessor { | ||
|  |             get { | ||
|  |                 this.InitAccessors(); | ||
|  |                 return this.accPublic; | ||
|  |             } | ||
|  |         } | ||
|  |         public override MetaAccessor StorageAccessor { | ||
|  |             get { | ||
|  |                 this.InitAccessors(); | ||
|  |                 return this.accPublic; | ||
|  |             } | ||
|  |         } | ||
|  |         public override MetaAccessor DeferredValueAccessor { | ||
|  |             get { return null; } | ||
|  |         } | ||
|  |         public override MetaAccessor DeferredSourceAccessor { | ||
|  |             get { return null; } | ||
|  |         } | ||
|  |         public override bool IsDeferred { | ||
|  |             get { return false; } | ||
|  |         } | ||
|  |         public override bool IsPersistent { | ||
|  |             get { return false; } | ||
|  |         } | ||
|  |         public override bool IsAssociation { | ||
|  |             get { return false; } | ||
|  |         } | ||
|  |         public override bool IsPrimaryKey { | ||
|  |             get { return false; } | ||
|  |         } | ||
|  |         public override bool IsDbGenerated { | ||
|  |             get { return false; } | ||
|  |         } | ||
|  |         public override bool IsVersion { | ||
|  |             get { return false; } | ||
|  |         } | ||
|  |         public override bool IsDiscriminator { | ||
|  |             get { return false; } | ||
|  |         } | ||
|  |         public override bool CanBeNull { | ||
|  |             get { return !this.type.IsValueType || TypeSystem.IsNullableType(this.type); } | ||
|  |         } | ||
|  |         public override string DbType { | ||
|  |             get { return null; } | ||
|  |         } | ||
|  |         public override string Expression { | ||
|  |             get { return null; } | ||
|  |         } | ||
|  |         public override string MappedName { | ||
|  |             get { return this.member.Name; } | ||
|  |         } | ||
|  |         public override UpdateCheck UpdateCheck { | ||
|  |             get { return UpdateCheck.Never; } | ||
|  |         } | ||
|  |         public override AutoSync AutoSync { | ||
|  |             get { return AutoSync.Never; } | ||
|  |         } | ||
|  |         public override MetaAssociation Association { | ||
|  |             get { return null; } | ||
|  |         } | ||
|  |         public override MethodInfo LoadMethod { | ||
|  |             get { return null; } | ||
|  |         } | ||
|  |         private static MetaAccessor MakeMemberAccessor(Type accessorType, MemberInfo mi) { | ||
|  |             FieldInfo fi = mi as FieldInfo; | ||
|  |             MetaAccessor acc = null; | ||
|  |             if (fi != null) { | ||
|  |                 acc = FieldAccessor.Create(accessorType, fi); | ||
|  |             } | ||
|  |             else { | ||
|  |                 PropertyInfo pi = (PropertyInfo)mi; | ||
|  |                 acc = PropertyAccessor.Create(accessorType, pi, null); | ||
|  |             } | ||
|  |             return acc; | ||
|  |         } | ||
|  |     } | ||
|  | } |