view Source/DataAccess/DataAccessor.cs @ 1:8f65451dc28f

Исправлена проблема с фабрикой и выборкой нескольких объектов в linq выражении
author cin
date Fri, 28 Mar 2014 01:04:56 +0400
parents f990fcb411a9
children
line wrap: on
line source

using System;
using System.Collections;
using System.Collections.Generic;
using System.Data;
using System.Data.SqlTypes;
using System.Diagnostics;
using System.IO;
using System.Reflection;
using System.Xml;
#if !SILVERLIGHT
using System.Xml.Linq;
#endif

namespace BLToolkit.DataAccess
{
	using Aspects;
	using Common;
	using Data;
	using Data.DataProvider;
	using Mapping;
	using Patterns;
	using Properties;
	using Reflection;
	using TypeBuilder;

	[DataAccessor, DebuggerStepThrough]
	public abstract class DataAccessor : DataAccessorBase
	{
		#region Constructors

		protected DataAccessor()
		{
		}

		protected DataAccessor(DbManager dbManager)
			: base(dbManager)
		{
		}

		protected DataAccessor(DbManager dbManager, bool dispose)
			: base(dbManager, dispose)
		{
		}

		#endregion

		#region CreateInstance

		public static DataAccessor CreateInstance(Type type)
		{
			return (DataAccessor)Activator.CreateInstance(TypeFactory.GetType(type));
		}

		public static DataAccessor CreateInstance(Type type, InitContext context)
		{
			return (DataAccessor)Activator.CreateInstance(TypeFactory.GetType(type), context);
		}

		public static DataAccessor CreateInstance(Type type, DbManager dbManager)
		{
			return CreateInstance(type, dbManager, false);
		}

		public static DataAccessor CreateInstance(
			Type type,
			InitContext context,
			DbManager dbManager)
		{
			return CreateInstance(type, context, dbManager, false);
		}

		public static DataAccessor CreateInstance(Type type, DbManager dbManager, bool dispose)
		{
			var da = CreateInstance(type);

			da.SetDbManager(dbManager, dispose);

			return da;
		}

		public static DataAccessor CreateInstance(
			Type type,
			InitContext context,
			DbManager dbManager,
			bool dispose)
		{
			var da = CreateInstance(type, context);

			da.SetDbManager(dbManager, dispose);

			return da;
		}

		public static T CreateInstance<T>() where T : DataAccessor
		{
			return TypeFactory.CreateInstance<T>();
		}

		public static T CreateInstance<T>(DbManager dbManager)
			where T : DataAccessor
		{
			return CreateInstance<T>(dbManager, false);
		}

		public static T CreateInstance<T>(DbManager dbManager, bool dispose)
			where T : DataAccessor
		{
			var da = TypeFactory.CreateInstance<T>();

			da.SetDbManager(dbManager, dispose);

			return da;
		}

		#endregion

		#region Protected Members

		#region Parameters

		[NoInterception]
		protected virtual string GetQueryParameterName(DbManager db, string paramName)
		{
			return (string)db.DataProvider.Convert(paramName, ConvertType.NameToQueryParameter);
		}

		[NoInterception]
		protected virtual string GetSpParameterName(DbManager db, string paramName)
		{
			return (string)db.DataProvider.Convert(paramName, db.GetConvertTypeToParameter());
		}

		[NoInterception]
		protected virtual IDbDataParameter[] PrepareParameters(DbManager db, object[] parameters)
		{
			return db.PrepareParameters(parameters);
		}

		[NoInterception]
		protected virtual IDbDataParameter GetParameter(DbManager db, string paramName)
		{
			var p = db.Parameter(paramName);

			if (p == null)
			{
				// This usually means that the parameter name is incorrect.
				//
				throw new DataAccessException(string.Format(
					Resources.DataAccessot_ParameterNotFound, paramName));
			}

			// Input parameter mapping make no sence.
			//
			Debug.WriteLineIf(p.Direction == ParameterDirection.Input,
				string.Format("'{0}.{1}' is an input parameter.",
					db.Command.CommandText, paramName));

			return p;
		}

		[NoInterception]
		protected virtual IDbDataParameter[] CreateParameters(
			DbManager                 db,
			object                    obj,
			string[]                  outputParameters,
			string[]                  inputOutputParameters,
			string[]                  ignoreParameters,
			params IDbDataParameter[] commandParameters)
		{
			return db.CreateParameters(obj, outputParameters,
				inputOutputParameters, ignoreParameters, commandParameters);
		}

		[NoInterception]
		protected virtual IDbDataParameter[] CreateParameters(
			DbManager                 db,
			DataRow                   dataRow,
			string[]                  outputParameters,
			string[]                  inputOutputParameters,
			string[]                  ignoreParameters,
			params IDbDataParameter[] commandParameters)
		{
			return db.CreateParameters(dataRow, outputParameters,
				inputOutputParameters, ignoreParameters, commandParameters);
		}

		[NoInterception]
		protected virtual string PrepareSqlQuery(DbManager db, int queryID, int uniqueID, string sqlQuery)
		{
			return sqlQuery;
		}

		#endregion

		#region ExecuteDictionary

		protected void ExecuteDictionary(
			DbManager             db,
			IDictionary           dictionary,
			Type                  objectType,
			Type                  keyType,
			string                methodName)
		{
			var isIndex = TypeHelper.IsSameOrParent(typeof(CompoundValue), keyType);
			var mms     = new SqlQuery(Extensions).GetKeyFieldList(db, objectType);

			if (mms.Length == 0)
				throw new DataAccessException(string.Format(
					Resources.DataAccessor_UnknownIndex,
					GetType().Name, methodName));

			if (mms.Length > 1 && keyType != typeof(object) && !isIndex)
				throw new DataAccessException(string.Format(
					Resources.DataAccessor_InvalidKeyType,
					GetType().Name, methodName));

			if (isIndex || mms.Length > 1)
			{
				var fields = new string[mms.Length];

				for (var i = 0; i < mms.Length; i++)
					fields[i] = mms[i].MemberName;

				db.ExecuteDictionary(dictionary, new MapIndex(fields), objectType, null);
			}
			else
			{
				db.ExecuteDictionary(dictionary, mms[0].MemberName, objectType, null);
			}
		}

		protected void ExecuteDictionary<TValue>(
			DbManager                          db,
			IDictionary<CompoundValue, TValue> dictionary,
			Type                               objectType,
			string                             methodName)
		{
			var mms = new SqlQuery(Extensions).GetKeyFieldList(db, objectType);

			if (mms.Length == 0)
				throw new DataAccessException(string.Format(
					Resources.DataAccessor_UnknownIndex,
					GetType().Name, methodName));

			var fields = new string[mms.Length];

			for (var i = 0; i < mms.Length; i++)
				fields[i] = mms[i].MemberName;

			db.ExecuteDictionary(dictionary, new MapIndex(fields), objectType, null);
		}

		protected void ExecuteDictionary<TKey, TValue>(
			DbManager                 db,
			IDictionary<TKey, TValue> dictionary,
			Type                      objectType,
			string                    methodName)
		{
			var mms = new SqlQuery(Extensions).GetKeyFieldList(db, objectType);

			if (mms.Length == 0)
				throw new DataAccessException(string.Format(
					Resources.DataAccessor_UnknownIndex,
					GetType().Name, methodName));

			if (mms.Length != 1)
				throw new DataAccessException(string.Format(
					Resources.DataAccessor_IndexIsComplex,
					GetType().Name, methodName));

			db.ExecuteDictionary(dictionary, mms[0].MemberName, objectType, null);
		}

		protected void ExecuteScalarDictionary(
			DbManager             db,
			IDictionary           dictionary,
			Type                  objectType,
			Type                  keyType,
			string                methodName,
			NameOrIndexParameter  scalarField,
			Type                  elementType)
		{
			var isIndex = TypeHelper.IsSameOrParent(typeof(CompoundValue), keyType);
			var mms     = new SqlQuery(Extensions).GetKeyFieldList(db, objectType);

			if (mms.Length == 0)
				throw new DataAccessException(string.Format(
					Resources.DataAccessor_UnknownIndex,
					GetType().Name, methodName));

			if (mms.Length > 1 && keyType != typeof(object) && !isIndex)
				throw new DataAccessException(string.Format(
					Resources.DataAccessor_InvalidKeyType,
					GetType().Name, methodName));

			if (isIndex || mms.Length > 1)
			{
				var fields = new string[mms.Length];

				for (var i = 0; i < mms.Length; i++)
					fields[i] = mms[i].Name;

				db.ExecuteScalarDictionary(dictionary, new MapIndex(fields), scalarField, elementType);
			}
			else
			{
				db.ExecuteScalarDictionary(
					dictionary,
					mms[0].Name,
					keyType,
					scalarField,
					elementType);
			}
		}

		#endregion

		#region ExecuteEnumerable

		protected IEnumerable<T> ExecuteEnumerable<T>(DbManager db, Type objectType, bool disposeDbManager)
		{
			try
			{
				using (var rd = db.ExecuteReader())
				{
					if (rd.Read())
					{
						var dest   = MappingSchema.GetObjectMapper(objectType);
						var source = MappingSchema.CreateDataReaderMapper(rd);

						var ctx = new InitContext
						{
							MappingSchema = MappingSchema,
							ObjectMapper  = dest,
							DataSource    = source,
							SourceObject  = rd
						};

						var index   = MappingSchema.GetIndex(source, dest);
						var mappers = ctx.MappingSchema.GetValueMappers(source, dest, index);

						do
						{
							var destObject = (T)dest.CreateInstance(ctx);

							if (ctx.StopMapping)
								yield return destObject;

							var smDest = destObject as ISupportMapping;

							if (smDest != null)
							{
								smDest.BeginMapping(ctx);

								if (ctx.StopMapping)
									yield return destObject;
							}

							MappingSchema.MapInternal(source, rd, dest, destObject, index, mappers);

							if (smDest != null)
								smDest.EndMapping(ctx);

							yield return destObject;
						} while (rd.Read());
					}
				}
			}
			finally
			{
				if (disposeDbManager)
					db.Dispose();
			}
		}

		protected IEnumerable ExecuteEnumerable(DbManager db, Type objectType, bool disposeDbManager)
		{
			var ms = db.MappingSchema;

			if (disposeDbManager)
			{
				using (db)
				using (var rd = db.ExecuteReader())
					while (rd.Read())
						yield return ms.MapDataReaderToObject(rd, objectType);
			}
			else
			{
				using (var rd = db.ExecuteReader())
					while (rd.Read())
						yield return ms.MapDataReaderToObject(rd, objectType);
			}
		}

		#endregion

		#region Convert

		#region Primitive Types

		[CLSCompliant(false)]
		[NoInterception]
		protected virtual SByte ConvertToSByte(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToSByte(value);
		}

		[NoInterception]
		protected virtual Int16 ConvertToInt16(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToInt16(value);
		}

		[NoInterception]
		protected virtual Int32 ConvertToInt32(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToInt32(value);
		}

		[NoInterception]
		protected virtual Int64 ConvertToInt64(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToInt64(value);
		}

		[NoInterception]
		protected virtual Byte ConvertToByte(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToByte(value);
		}

		[CLSCompliant(false)]
		[NoInterception]
		protected virtual UInt16 ConvertToUInt16(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToUInt16(value);
		}

		[CLSCompliant(false)]
		[NoInterception]
		protected virtual UInt32 ConvertToUInt32(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToUInt32(value);
		}

		[CLSCompliant(false)]
		[NoInterception]
		protected virtual UInt64 ConvertToUInt64(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToUInt64(value);
		}

		[NoInterception]
		protected virtual Char ConvertToChar(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToChar(value);
		}

		[NoInterception]
		protected virtual Single ConvertToSingle(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToSingle(value);
		}

		[NoInterception]
		protected virtual Double ConvertToDouble(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToDouble(value);
		}

		[NoInterception]
		protected virtual Boolean ConvertToBoolean(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToBoolean(value);
		}

		#endregion

		#region Simple Types

		[NoInterception]
		protected virtual String ConvertToString(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToString(value);
		}

		[NoInterception]
		protected virtual DateTime ConvertToDateTime(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToDateTime(value);
		}

		[NoInterception]
		protected virtual DateTimeOffset ConvertToDateTimeOffset(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToDateTimeOffset(value);
		}

		[NoInterception]
		protected virtual System.Data.Linq.Binary ConvertToLinqBinary(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToLinqBinary(value);
		}

		[NoInterception]
		protected virtual Decimal ConvertToDecimal(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToDecimal(value);
		}

		[NoInterception]
		protected virtual Guid ConvertToGuid(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToGuid(value);
		}

		[NoInterception]
		protected virtual Stream ConvertToStream(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToStream(value);
		}

		[NoInterception]
		protected virtual XmlReader ConvertToXmlReader(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToXmlReader(value);
		}

		[NoInterception]
		protected virtual XmlDocument ConvertToXmlDocument(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToXmlDocument(value);
		}

#if !SILVERLIGHT
		[NoInterception]
        protected virtual XElement ConvertToXElement(DbManager db, object value, object parameter)
        {
            return db.MappingSchema.ConvertToXElement(value);
        }
#endif

        [NoInterception]
		protected virtual Byte[] ConvertToByteArray(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToByteArray(value);
		}

		[NoInterception]
		protected virtual Char[] ConvertToCharArray(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToCharArray(value);
		}

		#endregion

		#region SqlTypes

		[NoInterception]
		protected virtual SqlByte ConvertToSqlByte(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToSqlByte(value);
		}

		[NoInterception]
		protected virtual SqlInt16 ConvertToSqlInt16(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToSqlInt16(value);
		}

		[NoInterception]
		protected virtual SqlInt32 ConvertToSqlInt32(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToSqlInt32(value);
		}

		[NoInterception]
		protected virtual SqlInt64 ConvertToSqlInt64(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToSqlInt64(value);
		}

		[NoInterception]
		protected virtual SqlSingle ConvertToSqlSingle(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToSqlSingle(value);
		}

		[NoInterception]
		protected virtual SqlBoolean ConvertToSqlBoolean(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToSqlBoolean(value);
		}

		[NoInterception]
		protected virtual SqlDouble ConvertToSqlDouble(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToSqlDouble(value);
		}

		[NoInterception]
		protected virtual SqlDateTime ConvertToSqlDateTime(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToSqlDateTime(value);
		}

		[NoInterception]
		protected virtual SqlDecimal ConvertToSqlDecimal(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToSqlDecimal(value);
		}

		[NoInterception]
		protected virtual SqlMoney ConvertToSqlMoney(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToSqlMoney(value);
		}

		[NoInterception]
		protected virtual SqlGuid ConvertToSqlGuid(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToSqlGuid(value);
		}

		[NoInterception]
		protected virtual SqlString ConvertToSqlString(DbManager db, object value, object parameter)
		{
			return db.MappingSchema.ConvertToSqlString(value);
		}

		#endregion

		#region General case

		[NoInterception]
		protected virtual object ConvertChangeType(
			DbManager db,
			object    value,
			Type      conversionType,
			object    parameter)
		{
			return db.MappingSchema.ConvertChangeType(value, conversionType);
		}

		#endregion
		
		#endregion

		#region IsNull

		/// <summary>
		/// Reserved for internal BLToolkit use.
		/// </summary>
		public interface INullableInternal
		{
			bool IsNull { [MustImplement(false, false)] get; }
		}

		[NoInterception]
		protected virtual bool IsNull(
			DbManager db,
			object    value,
			object    parameter)
		{
			// Speed up for scalar and nullable types.
			//
			switch (System.Convert.GetTypeCode(value))
			{
				// null, DBNull.Value, Nullable<T> without a value.
				//
				case TypeCode.Empty:
				case TypeCode.DBNull:
					return true;

				case TypeCode.Object:
					break;

				// int, byte, string, DateTime and other primitives except Guid.
				// Also Nullable<T> with a value.
				//
				default:
					return false;
			}

			// Speed up for SqlTypes.
			//
			var nullable = value as INullable;
			if (nullable != null)
				return nullable.IsNull;

			// All other types which have 'IsNull' property but does not implement 'INullable' interface.
			// For example: 'Oracle.DataAccess.Types.OracleDecimal'.
			//
			// For types without 'IsNull' property the return value is always false.
			//
			var nullableInternal = (INullableInternal)DuckTyping.Implement(typeof(INullableInternal), value);

			return nullableInternal.IsNull;
		}

		#endregion

		protected virtual SqlQueryAttribute GetSqlQueryAttribute(MethodInfo methodInfo)
		{
			var attrs = methodInfo.GetCustomAttributes(typeof(SqlQueryAttribute), true);
			return (SqlQueryAttribute)attrs[0];
		}

		#endregion
	}
}