sabato 24 marzo 2012

IEnumerable AsDataReader & SqlBulkCopy

 

Volete trasferire il contenuto della vostra collezione Enumerabile su una tabella SQL o caricare un DataTable a partire da essa?

Dopo alcune ricerche mi sono soffermato sul seguente articolo :

//===============================================================================
// Copyright  2008 Microsoft Corporation.  All rights reserved.
// THIS CODE AND INFORMATION IS PROVIDED "AS IS" WITHOUT WARRANTY
// OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING BUT NOT
// LIMITED TO THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
// FITNESS FOR A PARTICULAR PURPOSE.
//===============================================================================
//
//This code lives here: http://code.msdn.microsoft.com/LinqEntityDataReader
//Please visit for comments, issues and updates
//
//Version 1.0.0.1 Added GetSchemaTable support for Loading DataTables from EntityDataReader
//Version 1.0.0.2 Added support for Entity Framework types, including Foreign Key columns
//Version 1.0.0.3 In DataReader.GetValue, now using dynamic methods for common scalar types instead of reflection with PropertyInfo.GetValue()
//Version 1.0.0.4 Added support for simple IEnumerable<T> where T is a scalar to support, eg, passing List<int> to a TVP
//Version 1.0.0.5 Simplified the Attribute code, added dynamic method support for all scalar types
//Version 1.0.0.6 Replaced the switch block for ValueAccessor with a Lambda Expression
//Version 1.0.0.6 Fixed a bug with nullable foreign keys on EF entities
//Version 1.0.0.6 Extensive refactoring, introduced EntityDataReaderOptions, changed constructors
//Version 1.0.0.6 Introduced option to flatten related entities.  Now you can have the EntityDataReader flatten an object graph.
//                This is especially useful for enabling you to project all the scalars of a related enty by just projecting the entity.
//                eg. a projection like new { salesOrder.ID, salesOrder.Customer, salesOrder.Product }                 
//                will create a DbDataReader with Id, Customer.*, Product.*
//Version 1.0.0.7 For anonymous types the order of columns is now controlled by the anonymous type's constructor arguments, for which
//                Reflection tracks the ordinal position.  This ordering is applied to any type where the constructor args match the properties
//                1-1 on name and type.  Reflection over properties does not preserve the declaration order of the properties, and for Table Valued Parameters
//                SqlClient maps the DbDataReader columns to the TVP columns by ordinal position, not by name.  This relies on the behavior of
//                the C# compiler that Anonymous types have a constructor with constructor arguments that match the object initializer on both name
//                and position.



namespace Microsoft.Samples.EntityDataReader
{
  using System;
  using System.Collections.Generic;
  using System.Linq;
  using System.Data;
  using System.Reflection;
  using System.Data.Common;
  using System.Data.Objects.DataClasses;
  using System.Data.Metadata.Edm;
  using System.Data.Objects;
  using System.Linq.Expressions;


  /// <summary>
  /// The EntityDataReader wraps a collection of CLR objects in a DbDataReader.  
  /// Only "scalar" properties are projected, with the exception that Entity Framework
  /// EntityObjects that have references to other EntityObjects will have key values for
  /// the related entity projected.
  ///
  /// This is useful for doing high-speed data loads with SqlBulkCopy, and copying collections
  /// of entities ot a DataTable for use with SQL Server Table-Valued parameters, or for interop
  /// with older ADO.NET applciations.
  ///
  /// For explicit control over the fields projected by the DataReader, just wrap your collection
  /// of entities in a anonymous type projection before wrapping it in an EntityDataReader.
  ///
  /// Instead of
  /// IEnumerable<Order> orders;
  /// ...
  /// IDataReader dr = orders.AsDataReader();
  ///
  /// do
  /// IEnumerable<Order> orders;
  /// ...
  /// var q = from o in orders
  ///         select new
  ///         {
  ///            ID=o.ID,
  ///            ShipDate=o.ShipDate,
  ///            ProductName=o.Product.Name,
  ///            ...
  ///         }
  /// IDataReader dr = q.AsDataReader();
  ///
  /// </summary>
  /// <typeparam name="T"></typeparam>
  public sealed class EntityDataReader<T> : DbDataReader, IDataReader
  {

    readonly IEnumerator<T> enumerator;
    readonly EntityDataReaderOptions options;

    T current;
    bool closed = false;

    static List<Attribute> scalarAttributes;
    static List<Attribute> scalarAttributesPlusRelatedObjectScalarAttributes;
    static List<Attribute> scalarAttributesPlusRelatedObjectKeyAttributes;

    readonly List<Attribute> attributes;

    #region Attribute inner type
    private class Attribute
    {
      //PropertyInfo propertyInfo;
      public readonly Type Type;
      public readonly string FullName;
      public readonly string Name;
      public readonly bool IsRelatedAttribute;

      readonly Func<T, object> ValueAccessor;

      /// <summary>
      /// Uses Lamda expressions to create a Func<T,object> that invokes the given property getter.
      /// The property value will be extracted and cast to type TProperty
      /// </summary>
      /// <typeparam name="TObject">The type of the object declaring the property.</typeparam>
      /// <typeparam name="TProperty">The type to cast the property value to</typeparam>
      /// <param name="pi">PropertyInfo pointing to the property to wrap</param>
      /// <returns></returns>
      public static Func<TObject, TProperty> MakePropertyAccessor<TObject,TProperty>(PropertyInfo pi)
      {
        ParameterExpression objParam = Expression.Parameter(typeof(TObject), "obj");
        MemberExpression typedAccessor = Expression.PropertyOrField(objParam, pi.Name);
        UnaryExpression castToObject = Expression.Convert(typedAccessor, typeof(object));
        LambdaExpression lambdaExpr = Expression.Lambda<Func<TObject, TProperty>>(castToObject, objParam);

        return (Func<TObject, TProperty>)lambdaExpr.Compile();
      }


      public static Func<TObject, TProperty> MakeRelatedPropertyAccessor<TObject, TProperty>(PropertyInfo pi, PropertyInfo pi2)
      {

        Func<TObject, object> getRelatedObject;
        {
          // expression like:
          //    return (object)t.SomeProp;
          ParameterExpression typedParam = Expression.Parameter(typeof(T), "t");
          MemberExpression typedAccessor = Expression.PropertyOrField(typedParam, pi.Name);
          UnaryExpression castToObject = Expression.Convert(typedAccessor, typeof(object));
          LambdaExpression lambdaExpr = Expression.Lambda<Func<TObject, object>>(castToObject, typedParam);
          getRelatedObject = (Func<TObject, object>)lambdaExpr.Compile();
        }


        Func<object, TProperty> getRelatedObjectProperty;
        {
                 
          // expression like:
          //    return (object)((PropType)o).RelatedProperty;
          ParameterExpression objParam = Expression.Parameter(typeof(object), "o");
          UnaryExpression typedParam = Expression.Convert(objParam,pi.PropertyType);
          MemberExpression typedAccessor = Expression.PropertyOrField(typedParam, pi2.Name);
          UnaryExpression castToObject = Expression.Convert(typedAccessor, typeof(TProperty));
          LambdaExpression lambdaExpr = Expression.Lambda<Func<object, TProperty>>(castToObject, objParam);
          getRelatedObjectProperty = (Func<object, TProperty>)lambdaExpr.Compile();
        }

        Func<TObject, TProperty> f = (TObject t) =>
        {
          object o = getRelatedObject(t);
          if (o == null) return default(TProperty);
          return getRelatedObjectProperty(o);
        };

        return f;
      }

      public Attribute(PropertyInfo pi)
      {
        this.FullName = pi.DeclaringType.Name + "_" + pi.Name;
        this.Name = pi.Name;
        Type = pi.PropertyType;
        IsRelatedAttribute = false;

        ValueAccessor = MakePropertyAccessor<T,object>(pi);
      }

      public Attribute(string fullName, string name, Type type, Func<T, object> getValue, bool isRelatedAttribute)
      {
        this.FullName = fullName;
        this.Name = name;
        this.Type = type;
        this.ValueAccessor = getValue;
        this.IsRelatedAttribute = isRelatedAttribute;
      }
      
      public object GetValue(T target)
      {
        return ValueAccessor(target);
      }
    }
    #endregion

    #region "Scalar Types"

    static bool IsScalarType(Type t)
    {
      return scalarTypes.Contains(t);
    }
    static readonly HashSet<Type> scalarTypes = LoadScalarTypes();
    static HashSet<Type> LoadScalarTypes()
    {
      HashSet<Type> set = new HashSet<Type>()
                              {
                                //reference types
                                typeof(String),
                                typeof(Byte[]),
                                //value types
                                typeof(Byte),
                                typeof(Int16),
                                typeof(Int32),
                                typeof(Int64),
                                typeof(Single),
                                typeof(Double),
                                typeof(Decimal),
                                typeof(DateTime),
                                typeof(Guid),
                                typeof(Boolean),
                                typeof(TimeSpan),
                                //nullable value types
                                typeof(Byte?),
                                typeof(Int16?),
                                typeof(Int32?),
                                typeof(Int64?),
                                typeof(Single?),
                                typeof(Double?),
                                typeof(Decimal?),
                                typeof(DateTime?),
                                typeof(Guid?),
                                typeof(Boolean?),
                                typeof(TimeSpan?)
                              };
      

      return set;

    }
    #endregion

    #region Constructors

    public EntityDataReader(IEnumerable<T> col)
      : this(col,EntityDataReaderOptions.Default,null) { }

    public EntityDataReader(IEnumerable<T> col, EntityDataReaderOptions options)
      : this(col, options, null) { }

    public EntityDataReader(IEnumerable<T> col, EntityDataReaderOptions options, ObjectContext objectContext)
    {
      this.enumerator = col.GetEnumerator();
      this.options = options;

      if (options.RecreateForeignKeysForEntityFrameworkEntities && objectContext == null)
      {
        throw new ArgumentException("If RecreateForeignKeysForEntityFrameworkEntities=true then objectContext is required");
      }

      //done without a lock, so we risk running twice
      if (scalarAttributes == null)
      {
         scalarAttributes = DiscoverScalarAttributes(typeof(T));
      }
      if (options.FlattenRelatedObjects && scalarAttributesPlusRelatedObjectScalarAttributes == null)
      {
        var atts = DiscoverRelatedObjectScalarAttributes(typeof(T));
        scalarAttributesPlusRelatedObjectScalarAttributes = atts.Concat(scalarAttributes).ToList();
      }
      if (options.RecreateForeignKeysForEntityFrameworkEntities && scalarAttributesPlusRelatedObjectKeyAttributes == null)
      {
        var atts = DiscoverRelatedObjectKeyAttributes(typeof(T), objectContext);
        scalarAttributesPlusRelatedObjectKeyAttributes = atts.Concat(scalarAttributes).ToList();
      }

      if (options.FlattenRelatedObjects)
      {
        attributes = scalarAttributesPlusRelatedObjectScalarAttributes;
      }
      else if (objectContext != null)
      {
        attributes = scalarAttributesPlusRelatedObjectKeyAttributes;
      }
      else
      {
        attributes = scalarAttributes;
      }


    }


    static List<Attribute> DiscoverScalarAttributes(Type thisType)
    {     

      //Not a collection of entities, just an IEnumerable<String> or other scalar type.
      //So add just a single Attribute that returns the object itself
      if (IsScalarType(thisType))
      {
        return new List<Attribute> { new Attribute("Value","Value", thisType, t => t,false) };
      }


      //find all the scalar properties
      var allProperties = (from p in thisType.GetProperties()
                           where IsScalarType(p.PropertyType)
                           select p).ToList();

      //Look for a constructor with arguments that match the properties on name and type
      //(name modulo case, which varies between constructor args and properties in coding convention)
      //If such an "ordering constructor" exists, return the properties ordered by the corresponding
      //constructor args ordinal position.  
      //An important instance of an ordering constructor, is that C# anonymous types all have one.  So
      //this enables a simple convention to specify the order of columns projected by the EntityDataReader
      //by simply building the EntityDataReader from an anonymous type projection.
      //If such a constructor is found, replace allProperties with a collection of properties sorted by constructor order.
      foreach (var completeConstructor in from ci in thisType.GetConstructors()
                                         where ci.GetParameters().Count() == allProperties.Count()
                                         select ci)
      {
        var q = (from cp in completeConstructor.GetParameters()
                 join p in allProperties
                   on new { n = cp.Name.ToLower(), t = cp.ParameterType } equals new { n = p.Name.ToLower(), t = p.PropertyType }
                 select new { cp, p }).ToList();

        if (q.Count() == allProperties.Count()) //all constructor parameters matched by name and type to properties
        {
          //sort all properties by constructor ordinal position
          allProperties = ( from o in q
                            orderby o.cp.Position
                            select o.p ).ToList();
          break; //stop looking for an ordering consturctor
        }
                  
        
      }

      return allProperties.Select(p => new Attribute(p)).ToList();

    }
    static List<Attribute> DiscoverRelatedObjectKeyAttributes(Type thisType, ObjectContext objectContext)
    {

      var attributeList = new SortedList<string,Attribute>();

      
      //recreate foreign key column values
      //by adding Attributes for any key values of referenced entities
      //that aren't already exposed as scalar properties
      var mw = objectContext.MetadataWorkspace;
      var entityTypesByName = mw.GetItems<EntityType>(DataSpace.OSpace).ToLookup(e => e.FullName);

      //find the EntityType metadata for T
      EntityType thisEntity = entityTypesByName[thisType.FullName].First();
      var thisEntityKeys = thisEntity.KeyMembers.ToDictionary(k => k.Name);

      //TODOx use the NavigationProperties instead of the ENtityRelations -- too complicated
      //TODO fix the attribute naming.  Probably requires marking each attribtue as direct or related.
      

      var erProps = thisType.GetProperties()
                            .Where(p => typeof(EntityReference)
                            .IsAssignableFrom(p.PropertyType)).ToList();


      //For each EntityRelation property add add the keys of the related Entity
      foreach (var pi in erProps)
      {
        //Find the name of the CLR Type at the other end of the reference because we need to get its key attributes.
        //the property type is EntityReference<T>, we need T.
        string relatedEntityCLRTypeName = pi.PropertyType.GetGenericArguments().First().FullName;

        //Find the EntityType at the other end of the relationship because we need to get its key attributes.
        EntityType relatedEntityEFType = entityTypesByName[relatedEntityCLRTypeName].FirstOrDefault();
        if (relatedEntityEFType == null)
        {
          throw new InvalidOperationException("Cannot find EntityType for EntityReference Property " + pi.Name);
        }

        //Add attributes for each key value of the related entity.  These are the properties that
        //would probably appear in the storage object.  The names will be the same as they are on the
        //related entity, except prefixed with the related entity name,
        //and with a check to make sure that we're not introducing a duplicate.
        // so if you have
        //  if OrderItem.OrderID -> Order.ID   then the column will be Order_ID
        //  if OrderItem.OrderID -> Order.OrderID   then the column will be Order_OrderID
        foreach (var key in relatedEntityEFType.KeyMembers)
        {
          string targetKeyAttributeName = key.Name;

          //TODO it would be better to get the NavigationProperty and find the ToEndMember name
          //but the NavigationProperty doesn't have good way to get the EntityReference
          //or the related entity key.
          string referenceName;
          if (pi.Name.EndsWith("Reference",StringComparison.Ordinal))
          {
            referenceName = pi.Name.Substring(0, pi.Name.Length - "Reference".Length);
          }
          else  //there's no rule that the EntityReference named like this so, if not just use the target type
          {
            referenceName = pi.PropertyType.Name;

            //if there are multiple relations to the same target type, just uniqify them with an index
            int ix = erProps.Where(p => p.PropertyType == pi.PropertyType).ToList().IndexOf(pi);
            if (ix > 0)
            {
              referenceName = referenceName + ix.ToString(System.Globalization.CultureInfo.InvariantCulture);
            }
            
          }
          string fullName = referenceName + "_" + key.Name;
          

          //bind out local variables for the valueAccessor closure.
          Type kType = Type.GetType(key.TypeUsage.EdmType.FullName);
          PropertyInfo entityReferenceProperty = pi;

          Func<T, object> valueAccessor = o =>
          {
            EntityReference er = (EntityReference)entityReferenceProperty.GetValue(o, null);

            //for nullable foregn keys, just return null
            if (er.EntityKey == null)
            {
              return null;
            }
            object val = er.EntityKey.EntityKeyValues.First(k => k.Key == targetKeyAttributeName).Value;
            return val;
          };
          string name = key.Name;
          
          attributeList.Add(name, new Attribute(fullName, name, kType, valueAccessor, true));
        }


      }

      return attributeList.Values.ToList();

    }
    static List<Attribute> DiscoverRelatedObjectScalarAttributes(Type thisType)
    {
      
      var atts = new List<Attribute>();

      //get the related objects which aren't scalars, not EntityReference objects and not collections
      var relatedObjectProperties =
                        (from p in thisType.GetProperties()
                         where !IsScalarType(p.PropertyType)
                            && !typeof(System.Collections.IEnumerable).IsAssignableFrom(p.PropertyType)
                            && !typeof(EntityReference).IsAssignableFrom(p.PropertyType)
                            && !typeof(EntityKey).IsAssignableFrom(p.PropertyType)
                         select p).ToList();

      foreach (var rop in relatedObjectProperties)
      {
       var type = rop.PropertyType;
        //get the scalar properties for the related type
       var scalars = type.GetProperties().Where(p => IsScalarType(p.PropertyType)).ToList();

       foreach (var sp in scalars)
       {
         string attName = rop.Name + "_" + sp.Name;
         //create a value accessor which takes an instance of T, and returns the related object scalar
         var valueAccessor = Attribute.MakeRelatedPropertyAccessor<T,object>(rop, sp);
         string name = attName;
         Attribute att = new Attribute(rop.Name, attName,sp.PropertyType,valueAccessor,true);
         atts.Add(att);
       }

      }
      return atts;

    }



    #endregion

    #region Utility Methods
    static Type nullable_T = typeof(System.Nullable<int>).GetGenericTypeDefinition();
    static bool IsNullable(Type t)
    {
      return (t.IsGenericType
          && t.GetGenericTypeDefinition() == nullable_T);
    }
    static Type StripNullableType(Type t)
    {
      return t.GetGenericArguments()[0];
    }
    #endregion
    
    #region GetSchemaTable


    const string shemaTableSchema = @"<?xml version=""1.0"" standalone=""yes""?>
<xs:schema id=""NewDataSet"" xmlns="""" xmlns:xs=""http://www.w3.org/2001/XMLSchema"" xmlns:msdata=""urn:schemas-microsoft-com:xml-msdata"">
  <xs:element name=""NewDataSet"" msdata:IsDataSet=""true"" msdata:MainDataTable=""SchemaTable"" msdata:Locale="""">
    <xs:complexType>
      <xs:choice minOccurs=""0"" maxOccurs=""unbounded"">
        <xs:element name=""SchemaTable"" msdata:Locale="""" msdata:MinimumCapacity=""1"">
          <xs:complexType>
            <xs:sequence>
              <xs:element name=""ColumnName"" msdata:ReadOnly=""true"" type=""xs:string"" minOccurs=""0"" />
              <xs:element name=""ColumnOrdinal"" msdata:ReadOnly=""true"" type=""xs:int"" default=""0"" minOccurs=""0"" />
              <xs:element name=""ColumnSize"" msdata:ReadOnly=""true"" type=""xs:int"" minOccurs=""0"" />
              <xs:element name=""NumericPrecision"" msdata:ReadOnly=""true"" type=""xs:short"" minOccurs=""0"" />
              <xs:element name=""NumericScale"" msdata:ReadOnly=""true"" type=""xs:short"" minOccurs=""0"" />
              <xs:element name=""IsUnique"" msdata:ReadOnly=""true"" type=""xs:boolean"" minOccurs=""0"" />
              <xs:element name=""IsKey"" msdata:ReadOnly=""true"" type=""xs:boolean"" minOccurs=""0"" />
              <xs:element name=""BaseServerName"" msdata:ReadOnly=""true"" type=""xs:string"" minOccurs=""0"" />
              <xs:element name=""BaseCatalogName"" msdata:ReadOnly=""true"" type=""xs:string"" minOccurs=""0"" />
              <xs:element name=""BaseColumnName"" msdata:ReadOnly=""true"" type=""xs:string"" minOccurs=""0"" />
              <xs:element name=""BaseSchemaName"" msdata:ReadOnly=""true"" type=""xs:string"" minOccurs=""0"" />
              <xs:element name=""BaseTableName"" msdata:ReadOnly=""true"" type=""xs:string"" minOccurs=""0"" />
              <xs:element name=""DataType"" msdata:DataType=""System.Type, mscorlib, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089"" msdata:ReadOnly=""true"" type=""xs:string"" minOccurs=""0"" />
              <xs:element name=""AllowDBNull"" msdata:ReadOnly=""true"" type=""xs:boolean"" minOccurs=""0"" />
              <xs:element name=""ProviderType"" msdata:ReadOnly=""true"" type=""xs:int"" minOccurs=""0"" />
              <xs:element name=""IsAliased"" msdata:ReadOnly=""true"" type=""xs:boolean"" minOccurs=""0"" />
              <xs:element name=""IsExpression"" msdata:ReadOnly=""true"" type=""xs:boolean"" minOccurs=""0"" />
              <xs:element name=""IsIdentity"" msdata:ReadOnly=""true"" type=""xs:boolean"" minOccurs=""0"" />
              <xs:element name=""IsAutoIncrement"" msdata:ReadOnly=""true"" type=""xs:boolean"" minOccurs=""0"" />
              <xs:element name=""IsRowVersion"" msdata:ReadOnly=""true"" type=""xs:boolean"" minOccurs=""0"" />
              <xs:element name=""IsHidden"" msdata:ReadOnly=""true"" type=""xs:boolean"" minOccurs=""0"" />
              <xs:element name=""IsLong"" msdata:ReadOnly=""true"" type=""xs:boolean"" default=""false"" minOccurs=""0"" />
              <xs:element name=""IsReadOnly"" msdata:ReadOnly=""true"" type=""xs:boolean"" minOccurs=""0"" />
              <xs:element name=""ProviderSpecificDataType"" msdata:DataType=""System.Type, mscorlib, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089"" msdata:ReadOnly=""true"" type=""xs:string"" minOccurs=""0"" />
              <xs:element name=""DataTypeName"" msdata:ReadOnly=""true"" type=""xs:string"" minOccurs=""0"" />
              <xs:element name=""XmlSchemaCollectionDatabase"" msdata:ReadOnly=""true"" type=""xs:string"" minOccurs=""0"" />
              <xs:element name=""XmlSchemaCollectionOwningSchema"" msdata:ReadOnly=""true"" type=""xs:string"" minOccurs=""0"" />
              <xs:element name=""XmlSchemaCollectionName"" msdata:ReadOnly=""true"" type=""xs:string"" minOccurs=""0"" />
              <xs:element name=""UdtAssemblyQualifiedName"" msdata:ReadOnly=""true"" type=""xs:string"" minOccurs=""0"" />
              <xs:element name=""NonVersionedProviderType"" msdata:ReadOnly=""true"" type=""xs:int"" minOccurs=""0"" />
            </xs:sequence>
          </xs:complexType>
        </xs:element>
      </xs:choice>
    </xs:complexType>
  </xs:element>
</xs:schema>";
    public override DataTable GetSchemaTable()
    {
      DataSet s = new DataSet();
      s.Locale = System.Globalization.CultureInfo.CurrentCulture;
      s.ReadXmlSchema(new System.IO.StringReader(shemaTableSchema));
      DataTable t = s.Tables[0];
      for (int i = 0; i < this.FieldCount; i++)
      {
        DataRow row = t.NewRow();
        row["ColumnName"] = this.GetName(i);
        row["ColumnOrdinal"] = i;

        Type type = this.GetFieldType(i);
        if (type.IsGenericType
          && type.GetGenericTypeDefinition() == typeof(System.Nullable<int>).GetGenericTypeDefinition())
        {
          type = type.GetGenericArguments()[0];
        }
        row["DataType"] = this.GetFieldType(i);
        row["DataTypeName"] = this.GetDataTypeName(i);
        row["ColumnSize"] = -1;
        t.Rows.Add(row);
      }
      return t;

    }
    #endregion

    #region IDataReader Members

    public override void Close()
    {
      closed = true;
    }

    public override int Depth
    {
      get { return 1; }
    }


    public override bool IsClosed
    {
      get { return closed; }
    }

    public override bool NextResult()
    {
      return false;
    }

    int entitiesRead = 0;
    public override bool Read()
    {
       bool rv =  enumerator.MoveNext();
       if (rv)
       {
         current = enumerator.Current;
         entitiesRead += 1;
       }
       return rv;
    }

    public override int RecordsAffected
    {
      get { return -1; }
    }

    #endregion

    #region IDisposable Members

    protected override void Dispose(bool disposing)
    {
      Close();
      base.Dispose(disposing);
    }

    #endregion

    #region IDataRecord Members

    public override int FieldCount
    {
      get
      {
        return attributes.Count;
      }
    }

    TField GetValue<TField>(int i)
    {
      TField val = (TField)attributes[i].GetValue(current);
      return val;
    }
    public override bool GetBoolean(int i)
    {
      return GetValue<bool>(i);
    }

    public override byte GetByte(int i)
    {
      return GetValue<byte>(i);
    }

    public override long GetBytes(int i, long fieldOffset, byte[] buffer, int bufferoffset, int length)
    {

      var buf = GetValue<byte[]>(i);
      int bytes = Math.Min(length, buf.Length - (int)fieldOffset);
      Buffer.BlockCopy(buf, (int)fieldOffset, buffer, bufferoffset, bytes);
      return bytes;

    }

    public override char GetChar(int i)
    {
      return GetValue<char>(i);
    }

    public override long GetChars(int i, long fieldoffset, char[] buffer, int bufferoffset, int length)
    {
      //throw new NotImplementedException();
      string s = GetValue<string>(i);
      int chars   = Math.Min(length, s.Length-(int)fieldoffset );
      s.CopyTo((int)fieldoffset, buffer, bufferoffset, chars);

      return chars;
    }

    //public override DbDataReader GetData(int i)
    //{
    //  throw new NotImplementedException();
    //}

    public override string GetDataTypeName(int i)
    {
      return attributes[i].Type.Name;
    }

    public override DateTime GetDateTime(int i)
    {
      return GetValue<DateTime>(i);
    }

    public override decimal GetDecimal(int i)
    {
      return GetValue<decimal>(i);
    }

    public override double GetDouble(int i)
    {
      return GetValue<double>(i);
    }

    public override Type GetFieldType(int i)
    {
      Type t = attributes[i].Type;
      if (!options.ExposeNullableTypes && IsNullable(t))
      {
        return StripNullableType(t);
      }
      return t;
    }

    public override float GetFloat(int i)
    {
      return GetValue<float>(i);
    }

    public override Guid GetGuid(int i)
    {
      return GetValue<Guid>(i);
    }

    public override short GetInt16(int i)
    {
      return GetValue<short>(i);
    }

    public override int GetInt32(int i)
    {
      return GetValue<int>(i);
    }

    public override long GetInt64(int i)
    {
      return GetValue<long>(i);
    }

    public override string GetName(int i)
    {
      Attribute a = attributes[i];
      if (a.IsRelatedAttribute && options.PrefixRelatedObjectColumns)
      {
        return a.FullName;
      }
      return a.Name;
    }

    public override int GetOrdinal(string name)
    {
      for (int i = 0; i < attributes.Count; i++)
      {
        var a = attributes[i];

         if (!a.IsRelatedAttribute && a.Name == name)
         {
           return i;
         }

         if (options.PrefixRelatedObjectColumns && a.IsRelatedAttribute && a.FullName == name )
         {
          return i;
         }

         if (!options.PrefixRelatedObjectColumns && a.IsRelatedAttribute && a.Name == name)
         {
           return i;
         }


      }
      return -1;
    }

    public override string GetString(int i)
    {
      return GetValue<string>(i);
    }



    public override int GetValues(object[] values)
    {
      for (int i = 0; i < attributes.Count; i++)
      {
        values[i] = GetValue(i);
      }
      return attributes.Count;
    }


    
    public override object GetValue(int i)
    {
      object o = GetValue<object>(i);
      if (!options.ExposeNullableTypes && o == null)
      {
        return DBNull.Value;
      }
      return o;
    }

    public override bool IsDBNull(int i)
    {
      object o = GetValue<object>(i);
      return (o == null);
    }

    public override object this[string name]
    {
      get { return GetValue(GetOrdinal(name)); }
    }

    public override object this[int i]
    {
      get { return GetValue(i); }
    }

    #endregion

    #region DbDataReader Members



    public override System.Collections.IEnumerator GetEnumerator()
    {
      return this.enumerator;
    }

    public override bool HasRows
    {
      get { throw new NotSupportedException(); }
    }
    #endregion

  }

  public class EntityDataReaderOptions
  {
    public static EntityDataReaderOptions Default
    {
      get { return new EntityDataReaderOptions(true, false, true, false); }
    }

    public EntityDataReaderOptions(
      bool exposeNullableTypes,
      bool flattenRelatedObjects,
      bool prefixRelatedObjectColumns,
      bool recreateForeignKeysForEntityFrameworkEntities)
    {
      this.ExposeNullableTypes = exposeNullableTypes;
      this.FlattenRelatedObjects = flattenRelatedObjects;
      this.PrefixRelatedObjectColumns = prefixRelatedObjectColumns;
      this.RecreateForeignKeysForEntityFrameworkEntities = recreateForeignKeysForEntityFrameworkEntities;
    }

    /// <summary>
    /// If true nullable value types are returned directly by the DataReader.
    /// If false, the DataReader will expose non-nullable value types and return DbNull.Value
    /// for null values.  
    /// When loading a DataTable this option must be set to True, since DataTable does not support
    /// nullable types.
    /// </summary>
    public bool ExposeNullableTypes { get; set; }

    /// <summary>
    /// If True then the DataReader will project scalar properties from related objects in addition
    /// to scalar properties from the main object.  This is especially useful for custom projecttions like
    ///         var q = from od in db.SalesOrderDetail
    ///         select new
    ///         {
    ///           od,
    ///           ProductID=od.Product.ProductID,
    ///           ProductName=od.Product.Name
    ///         };
    /// Related objects assignable to EntityKey, EntityRelation, and IEnumerable are excluded.
    ///
    /// If False, then only scalar properties from teh main object will be projected.         
    /// </summary>
    public bool FlattenRelatedObjects { get; set; }

    /// <summary>
    /// If True columns projected from related objects will have column names prefixed by the
    /// name of the relating property.  This appies to either from setting FlattenRelatedObjects to True,
    /// or RecreateForeignKeysForEntityFrameworkEntities to True.
    ///
    /// If False columns will be created for related properties that are not prefixed.  This can lead
    /// to column name collision.
    /// </summary>
    public bool PrefixRelatedObjectColumns { get; set; }

    /// <summary>
    /// If True the DataReader will create columns for the key properties of related Entities.
    /// You must pass an ObjectContext and have retrieved the entity with change tracking for this to work.
    /// </summary>
    public bool RecreateForeignKeysForEntityFrameworkEntities { get; set; }

  }

  public static class EntityDataReaderExtensions
  {

    /// <summary>
    /// Wraps the IEnumerable in a DbDataReader, having one column for each "scalar" property of the type T.  
    /// The collection will be enumerated as the client calls IDataReader.Read().
    /// </summary>
    /// <typeparam name="T"></typeparam>
    /// <param name="collection"></param>
    /// <returns></returns>
    public static IDataReader AsDataReader<T>(this IEnumerable<T> collection)
    {

      //For anonymous type projections default to flattening related objects and not prefixing columns
      //The reason being that if the programmer has taken control of the projection, the default should
      //be to expose everying in the projection and not mess with the names.
      if (typeof(T).IsDefined(typeof(System.Runtime.CompilerServices.CompilerGeneratedAttribute), false))
      {
        var options = EntityDataReaderOptions.Default;
        options.FlattenRelatedObjects = true;
        options.PrefixRelatedObjectColumns = false;
        return new EntityDataReader<T>(collection,options);
      }
      return new EntityDataReader<T>(collection);
    }

    /// <summary>
    /// Wraps the IEnumerable in a DbDataReader, having one column for each "scalar" property of the type T.  
    /// The collection will be enumerated as the client calls IDataReader.Read().
    /// </summary>
    /// <typeparam name="T"></typeparam>
    /// <param name="collection"></param>
    /// <returns></returns>
    public static IDataReader AsDataReader<T>(this IEnumerable<T> collection, bool exposeNullableColumns, bool flattenRelatedObjects)
    {
      EntityDataReaderOptions options = new EntityDataReaderOptions(exposeNullableColumns, flattenRelatedObjects, true, false);
      
      return new EntityDataReader<T>(collection,options,null);
    }


    /// <summary>
    /// Enumerates the collection and copies the data into a DataTable.
    /// </summary>
    /// <typeparam name="T">The element type of the collection.</typeparam>
    /// <param name="collection">The collection to copy to a DataTable</param>
    /// <returns>A DataTable containing the scalar projection of the collection.</returns>
    public static DataTable ToDataTable<T>(this IEnumerable<T> collection)
    {
      DataTable t = new DataTable();
      t.Locale = System.Globalization.CultureInfo.CurrentCulture;
      t.TableName = typeof(T).Name;
      EntityDataReaderOptions options = EntityDataReaderOptions.Default;
      options.ExposeNullableTypes = false;
      EntityDataReader<T> dr = new EntityDataReader<T>(collection, options);
      t.Load(dr);
      return t;
    }

    /// <summary>
    /// Wraps the collection in a DataReader, but also includes columns for the key attributes of related Entities.
    /// </summary>
    /// <typeparam name="T">The element type of the collection.</typeparam>
    /// <param name="collection">A collection to wrap in a DataReader</param>
    /// <param name="cx">The Entity Framework ObjectContext, used for metadata access</param>
    /// <returns>A DbDataReader wrapping the collection.</returns>
    public static IDataReader AsDataReader<T>(this IEnumerable<T> collection, ObjectContext context) where T: EntityObject
    {
      EntityDataReaderOptions options = EntityDataReaderOptions.Default;
      options.RecreateForeignKeysForEntityFrameworkEntities = true;
      return new EntityDataReader<T>(collection,options,context);
    }

   

    /// <summary>
    /// Wraps the collection in a DataReader, but also includes columns for the key attributes of related Entities.
    /// </summary>
    /// <typeparam name="T">The element type of the collectin.</typeparam>
    /// <param name="collection">A collection to wrap in a DataReader</param>
    /// <param name="detachObjects">Option to detach each object in the collection from the ObjectContext.  This can reduce memory usage for queries returning large numbers of objects.</param>
    /// <param name="prefixReladObjectColumns">If True, qualify the related object keys, if False don't</param>
    /// <returns>A DbDataReader wrapping the collection.</returns>
    /// <summary>
    public static IDataReader AsDataReader<T>(this IEnumerable<T> collection, ObjectContext context, bool detachObjects, bool prefixRelatedObjectColumns) where T : EntityObject
    {
      EntityDataReaderOptions options = EntityDataReaderOptions.Default;
      options.RecreateForeignKeysForEntityFrameworkEntities = true;
      options.PrefixRelatedObjectColumns = prefixRelatedObjectColumns;

      if (detachObjects)
      {
        return new EntityDataReader<T>(collection.DetachAllFrom(context),options,context);
      }
      return new EntityDataReader<T>(collection,options,context);
    }
    static IEnumerable<T> DetachAllFrom<T>(this IEnumerable<T> col, ObjectContext cx)
    {
      foreach (var t in col)
      {
        cx.Detach(t);
        yield return t;
      }
    }

    /// <summary>
    /// Enumerates the collection and copies the data into a DataTable, but also includes columns for the key attributes of related Entities.
    /// </summary>
    /// <typeparam name="T">The element type of the collection.</typeparam>
    /// <param name="collection">The collection to copy to a DataTable</param>
    /// <param name="cx">The Entity Framework ObjectContext, used for metadata access</param>
    /// <returns>A DataTable containing the scalar projection of the collection.</returns>
    public static DataTable ToDataTable<T>(this IEnumerable<T> collection, ObjectContext context) where T: EntityObject
    {
      DataTable t = new DataTable();
      t.Locale = System.Globalization.CultureInfo.CurrentCulture;
      t.TableName = typeof(T).Name;

      EntityDataReaderOptions options = EntityDataReaderOptions.Default;
      options.RecreateForeignKeysForEntityFrameworkEntities = true;

      EntityDataReader<T> dr = new EntityDataReader<T>(collection, options, context);
      t.Load(dr);
      return t;
    }




  }
}

 

Bello vero?

Con questa estensione possiamo facilmente trasformare una IEnumerable in un IDataReader e usarlo nelle più comuni operazioni di lettura dati.

L’utilizzo è banale:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using Microsoft.Samples.EntityDataReader;

namespace CSharpLight_CSH
{
    class user
    {
        public int ID_User { get; set; }
        public int ID_Role { get; set; }
        public string Name { get; set; }
        public string Surname { get; set; }
    }

    class role
    {
        public int ID_Role { get; set; }
        public string Name { get; set; }
        public string Description { get; set; }
    }

    class Program
    {
        static void Main(string[] args)
        {
            List<role> rs = new List<role>();
            rs.Add(new role() { ID_Role = 1, Name = "Default", Description = "" });
            rs.Add(new role() { ID_Role = 2, Name = "J.Dev", Description = "" });
            rs.Add(new role() { ID_Role = 3, Name = "S.Dev", Description = "" });

            List<user> us = new List<user>();
            us.Add(new user() { ID_User = 1, ID_Role = 1, Name = "Fabio", Surname = "Arosio" });
            us.Add(new user() { ID_User = 1, ID_Role = 2, Name = "Claudio", Surname = "Arosio" });
            us.Add(new user() { ID_User = 1, ID_Role = 1, Name = "Lorenzo", Surname = "Arosio" });

            var us_rs = from u in us
                        where u.ID_Role == 1
                        select new
                        {
                            User_Name = u.Name,
                            User_Surname = u.Surname,
                            Role = rs.FirstOrDefault(r => r.ID_Role == u.ID_Role).Name
                        };

            //Ecco la bulkcopy
            System.Data.SqlClient.SqlBulkCopy SqlBulk = new System.Data.SqlClient.SqlBulkCopy("server=DBSERVER;database=DB;uid=USERID;pwd=PWD");
            SqlBulk.DestinationTableName = "TableInsertInto";
            SqlBulk.WriteToServer(us_rs.AsDataReader());

            //Oppure un Datatable
            System.Data.DataTable dt = new System.Data.DataTable("TestDT");
            dt.Load(us_rs.AsDataReader());

            //Semplice Vero?

            Console.ReadKey();
        }
    }
}

venerdì 23 marzo 2012

SpeedUp your Linq Queries with AsParallel

 

Diminuire il tempo di esecuzione di queries LINQ con l’abilitazione del parallelismo è semplice:

Public Class User
    Public ID_User As Integer
    Public ID_Role As Integer
    Public Name As String
    Public Surname As String
End Class
Public Class Role
    Public ID_Role As Integer
    Public Name As String
    Public Description As String
End Class

Sub Main()
    Dim sw As New Stopwatch
    Dim rs As New List(Of Role) From {New Role() With {.ID_Role = 1, .Name = "Default", .Description = ""}, _
                                      New Role() With {.ID_Role = 2, .Name = "J.Dev", .Description = ""}, _
                                      New Role() With {.ID_Role = 3, .Name = "S.Dev", .Description = ""}}

    Dim us As New List(Of User) From {New User() With {.ID_User = 1, .ID_Role = 1, .Name = "Fabio", .Surname = "Arosio"}, _
                                      New User() With {.ID_User = 1, .ID_Role = 2, .Name = "Claudio", .Surname = "Arosio"}, _
                                      New User() With {.ID_User = 1, .ID_Role = 1, .Name = "Lorenzo", .Surname = "Arosio"}}

    'Uso di una sola CPU
    sw.Start()
    Dim us_rs = (From p In us Where p.ID_Role = 1 _
                Select New With {.User_Name = p.Name, _
                                 .User_Surname = p.Surname, _
                                 .Role = (From q In rs Where q.ID_Role = p.ID_Role Select q.Name).FirstOrDefault _
                                }).ToList
    sw.Stop()
    Console.WriteLine("Single CPU Query - {0}", sw.Elapsed.TotalMilliseconds)


    'Multi CPU (AsParallel)
    sw = New Stopwatch
    sw.Start()
    Dim Parallel_us_rs = (From p In us Where p.ID_Role = 1 _
               Select New With {.User_Name = p.Name, _
                                .User_Surname = p.Surname, _
                                .Role = (From q In rs Where q.ID_Role = p.ID_Role Select q.Name).FirstOrDefault _
                               }).AsParallel.ToList
    sw.Stop()
    Console.WriteLine("Multi CPU Query - {0}", sw.Elapsed.TotalMilliseconds)


    '  PrintAnonymousT(us_rs)
    Console.ReadLine()
End Sub

image