using Dapper; using System; using System.Collections.Generic; using System.Data; using System.Linq; using System.Reflection; using System.Text; using System.Threading.Tasks; namespace Dapper.Contrib.Extensions { /// /// 修改mysql新增数据时返回影响记录数 /// public static partial class SqlMapperExtensions { /// /// Returns a single entity by a single id from table "Ts" asynchronously using Task. T must be of interface type. /// Id must be marked with [Key] attribute. /// Created entity is tracked/intercepted for changes and used by the Update() extension. /// /// Interface type to create and populate /// Open SqlConnection /// Id of the entity to get, must be marked with [Key] attribute /// The transaction to run under, null (the default) if none /// Number of seconds before command execution timeout /// Entity of T public static async Task GetAsync(this IDbConnection connection, dynamic id, IDbTransaction transaction = null, int? commandTimeout = null) where T : class { Type type = typeof(T); if (!GetQueries.TryGetValue(type.TypeHandle, out string sql)) { PropertyInfo key = GetSingleKey(nameof(GetAsync)); string name = GetTableName(type); sql = $"SELECT * FROM {name} WHERE {key.Name} = @keyName"; GetQueries[type.TypeHandle] = sql; } DynamicParameters dynParms = new DynamicParameters(); dynParms.Add("@keyName", id); if (!type.IsInterface) { return (await connection.QueryAsync(sql, dynParms, transaction, commandTimeout) .ConfigureAwait(false)).FirstOrDefault(); } IDictionary res = (await connection.QueryAsync(sql, dynParms).ConfigureAwait(false)).FirstOrDefault() as IDictionary; if (res == null) { return null; } T obj = ProxyGenerator.GetInterfaceProxy(); foreach (PropertyInfo property in TypePropertiesCache(type)) { object val = res[property.Name]; if (val == null) { continue; } if (property.PropertyType.IsGenericType && property.PropertyType.GetGenericTypeDefinition() == typeof(Nullable<>)) { Type genericType = Nullable.GetUnderlyingType(property.PropertyType); if (genericType != null) { property.SetValue(obj, Convert.ChangeType(val, genericType), null); } } else { property.SetValue(obj, Convert.ChangeType(val, property.PropertyType), null); } } ((IProxy)obj).IsDirty = false; //reset change tracking and return return obj; } /// /// Returns a list of entites from table "Ts". /// Id of T must be marked with [Key] attribute. /// Entities created from interfaces are tracked/intercepted for changes and used by the Update() extension /// for optimal performance. /// /// Interface or type to create and populate /// Open SqlConnection /// The transaction to run under, null (the default) if none /// Number of seconds before command execution timeout /// Entity of T public static Task> GetAllAsync(this IDbConnection connection, IDbTransaction transaction = null, int? commandTimeout = null) where T : class { Type type = typeof(T); Type cacheType = typeof(List); if (!GetQueries.TryGetValue(cacheType.TypeHandle, out string sql)) { GetSingleKey(nameof(GetAll)); string name = GetTableName(type); sql = "SELECT * FROM " + name; GetQueries[cacheType.TypeHandle] = sql; } if (!type.IsInterface) { return connection.QueryAsync(sql, null, transaction, commandTimeout); } return GetAllAsyncImpl(connection, transaction, commandTimeout, sql, type); } private static async Task> GetAllAsyncImpl(IDbConnection connection, IDbTransaction transaction, int? commandTimeout, string sql, Type type) where T : class { IEnumerable result = await connection.QueryAsync(sql).ConfigureAwait(false); List list = new List(); foreach (IDictionary res in result) { T obj = ProxyGenerator.GetInterfaceProxy(); foreach (PropertyInfo property in TypePropertiesCache(type)) { object val = res[property.Name]; if (val == null) { continue; } if (property.PropertyType.IsGenericType && property.PropertyType.GetGenericTypeDefinition() == typeof(Nullable<>)) { Type genericType = Nullable.GetUnderlyingType(property.PropertyType); if (genericType != null) { property.SetValue(obj, Convert.ChangeType(val, genericType), null); } } else { property.SetValue(obj, Convert.ChangeType(val, property.PropertyType), null); } } ((IProxy)obj).IsDirty = false; //reset change tracking and return list.Add(obj); } return list; } /// /// Inserts an entity into table "Ts" asynchronously using Task and returns identity id. /// /// The type being inserted. /// Open SqlConnection /// Entity to insert /// The transaction to run under, null (the default) if none /// Number of seconds before command execution timeout /// The specific ISqlAdapter to use, auto-detected based on connection if null /// Identity of inserted entity public static async Task InsertAsync(this IDbConnection connection, T entityToInsert, IDbTransaction transaction = null, int? commandTimeout = null, ISqlAdapter sqlAdapter = null) where T : class { Type type = typeof(T); sqlAdapter = sqlAdapter ?? GetFormatter(connection); bool isList = false; if (type.IsArray) { isList = true; type = type.GetElementType(); } else if (type.IsGenericType) { TypeInfo typeInfo = type.GetTypeInfo(); bool implementsGenericIEnumerableOrIsGenericIEnumerable = typeInfo.ImplementedInterfaces.Any(ti => ti.IsGenericType && ti.GetGenericTypeDefinition() == typeof(IEnumerable<>)) || typeInfo.GetGenericTypeDefinition() == typeof(IEnumerable<>); if (implementsGenericIEnumerableOrIsGenericIEnumerable) { isList = true; type = type.GetGenericArguments()[0]; } } string name = GetTableName(type); StringBuilder sbColumnList = new StringBuilder(null); List allProperties = TypePropertiesCache(type); List keyProperties = KeyPropertiesCache(type).ToList(); List computedProperties = ComputedPropertiesCache(type); List allPropertiesExceptKeyAndComputed = allProperties.ToList(); //.Except(keyProperties.Union(computedProperties)).ToList(); for (int i = 0; i < allPropertiesExceptKeyAndComputed.Count; i++) { PropertyInfo property = allPropertiesExceptKeyAndComputed[i]; sqlAdapter.AppendColumnName(sbColumnList, property.Name); if (i < allPropertiesExceptKeyAndComputed.Count - 1) { sbColumnList.Append(", "); } } StringBuilder sbParameterList = new StringBuilder(null); for (int i = 0; i < allPropertiesExceptKeyAndComputed.Count; i++) { PropertyInfo property = allPropertiesExceptKeyAndComputed[i]; sbParameterList.AppendFormat("@{0}", property.Name); if (i < allPropertiesExceptKeyAndComputed.Count - 1) { sbParameterList.Append(", "); } } int returnVal = 0; bool wasClosed = connection.State == ConnectionState.Closed; if (wasClosed) { connection.Open(); } if (!isList) //single entity { returnVal = await sqlAdapter.InsertAsync(connection, transaction, commandTimeout, name, sbColumnList.ToString(), sbParameterList.ToString(), keyProperties, entityToInsert); } else { //insert list of entities string cmd = $"insert into {name} ({sbColumnList}) values ({sbParameterList})"; returnVal = await connection.ExecuteAsync(cmd, entityToInsert, transaction, commandTimeout); } if (wasClosed) { connection.Close(); } return returnVal; } /// /// Inserts an entity into table "Ts" asynchronously using Task and returns identity id. /// /// The type being inserted. /// Open SqlConnection /// Entity to insert /// The transaction to run under, null (the default) if none /// Number of seconds before command execution timeout /// The specific ISqlAdapter to use, auto-detected based on connection if null /// Identity of inserted entity public static async Task InsertReturnPrimaryKeyAsync(this IDbConnection connection, T entityToInsert, IDbTransaction transaction = null, int? commandTimeout = null, ISqlAdapter sqlAdapter = null) where T : class { Type type = typeof(T); sqlAdapter = sqlAdapter ?? GetFormatter(connection); if (type.IsArray) { type = type.GetElementType(); } else if (type.IsGenericType) { TypeInfo typeInfo = type.GetTypeInfo(); bool implementsGenericIEnumerableOrIsGenericIEnumerable = typeInfo.ImplementedInterfaces.Any(ti => ti.IsGenericType && ti.GetGenericTypeDefinition() == typeof(IEnumerable<>)) || typeInfo.GetGenericTypeDefinition() == typeof(IEnumerable<>); if (implementsGenericIEnumerableOrIsGenericIEnumerable) { type = type.GetGenericArguments()[0]; } } string name = GetTableName(type); StringBuilder sbColumnList = new StringBuilder(null); List allProperties = TypePropertiesCache(type); List keyProperties = KeyPropertiesCache(type).ToList(); List computedProperties = ComputedPropertiesCache(type); List allPropertiesExceptKeyAndComputed = allProperties.ToList(); //.Except(keyProperties.Union(computedProperties)).ToList(); for (int i = 0; i < allPropertiesExceptKeyAndComputed.Count; i++) { PropertyInfo property = allPropertiesExceptKeyAndComputed[i]; sqlAdapter.AppendColumnName(sbColumnList, property.Name); if (i < allPropertiesExceptKeyAndComputed.Count - 1) { sbColumnList.Append(", "); } } StringBuilder sbParameterList = new StringBuilder(null); for (int i = 0; i < allPropertiesExceptKeyAndComputed.Count; i++) { PropertyInfo property = allPropertiesExceptKeyAndComputed[i]; sbParameterList.AppendFormat("@{0}", property.Name); if (i < allPropertiesExceptKeyAndComputed.Count - 1) { sbParameterList.Append(", "); } } long returnVal = 0; bool wasClosed = connection.State == ConnectionState.Closed; if (wasClosed) { connection.Open(); } //if (!isList) //single entity //{ // returnVal = await sqlAdapter.(connection, transaction, commandTimeout, name, // sbColumnList.ToString(), // sbParameterList.ToString(), keyProperties, entityToInsert); //} //else //{ //insert list of entities string cmd = $"insert into {name} ({sbColumnList}) values ({sbParameterList});SELECT LAST_INSERT_ID();"; returnVal = await connection.ExecuteScalarAsync(cmd, entityToInsert, transaction, commandTimeout); // } if (wasClosed) { connection.Close(); } return returnVal; } /// /// Updates entity in table "Ts" asynchronously using Task, checks if the entity is modified if the entity is tracked /// by the Get() extension. /// /// Type to be updated /// Open SqlConnection /// Entity to be updated /// The transaction to run under, null (the default) if none /// Number of seconds before command execution timeout /// true if updated, false if not found or not modified (tracked entities) public static async Task UpdateAsync(this IDbConnection connection, T entityToUpdate, IDbTransaction transaction = null, int? commandTimeout = null) where T : class { if (entityToUpdate is IProxy proxy && !proxy.IsDirty) { return false; } Type type = typeof(T); if (type.IsArray) { type = type.GetElementType(); } else if (type.IsGenericType) { TypeInfo typeInfo = type.GetTypeInfo(); bool implementsGenericIEnumerableOrIsGenericIEnumerable = typeInfo.ImplementedInterfaces.Any(ti => ti.IsGenericType && ti.GetGenericTypeDefinition() == typeof(IEnumerable<>)) || typeInfo.GetGenericTypeDefinition() == typeof(IEnumerable<>); if (implementsGenericIEnumerableOrIsGenericIEnumerable) { type = type.GetGenericArguments()[0]; } } List keyProperties = KeyPropertiesCache(type).ToList(); List explicitKeyProperties = ExplicitKeyPropertiesCache(type); if (keyProperties.Count == 0 && explicitKeyProperties.Count == 0) { throw new ArgumentException("Entity must have at least one [Key] or [ExplicitKey] property"); } string name = GetTableName(type); StringBuilder sb = new StringBuilder(); sb.AppendFormat("update {0} set ", name); List allProperties = TypePropertiesCache(type); keyProperties.AddRange(explicitKeyProperties); List computedProperties = ComputedPropertiesCache(type); List nonIdProps = allProperties.Except(keyProperties.Union(computedProperties)).ToList(); ISqlAdapter adapter = GetFormatter(connection); for (int i = 0; i < nonIdProps.Count; i++) { PropertyInfo property = nonIdProps[i]; adapter.AppendColumnNameEqualsValue(sb, property.Name); if (i < nonIdProps.Count - 1) { sb.Append(", "); } } sb.Append(" where "); for (int i = 0; i < keyProperties.Count; i++) { PropertyInfo property = keyProperties[i]; adapter.AppendColumnNameEqualsValue(sb, property.Name); if (i < keyProperties.Count - 1) { sb.Append(" and "); } } int updated = await connection .ExecuteAsync(sb.ToString(), entityToUpdate, commandTimeout: commandTimeout, transaction: transaction) .ConfigureAwait(false); return updated > 0; } /// /// Delete entity in table "Ts" asynchronously using Task. /// /// Type of entity /// Open SqlConnection /// Entity to delete /// The transaction to run under, null (the default) if none /// Number of seconds before command execution timeout /// true if deleted, false if not found public static async Task DeleteAsync(this IDbConnection connection, T entityToDelete, IDbTransaction transaction = null, int? commandTimeout = null) where T : class { if (entityToDelete == null) { throw new ArgumentException("Cannot Delete null Object", nameof(entityToDelete)); } Type type = typeof(T); if (type.IsArray) { type = type.GetElementType(); } else if (type.IsGenericType) { TypeInfo typeInfo = type.GetTypeInfo(); bool implementsGenericIEnumerableOrIsGenericIEnumerable = typeInfo.ImplementedInterfaces.Any(ti => ti.IsGenericType && ti.GetGenericTypeDefinition() == typeof(IEnumerable<>)) || typeInfo.GetGenericTypeDefinition() == typeof(IEnumerable<>); if (implementsGenericIEnumerableOrIsGenericIEnumerable) { type = type.GetGenericArguments()[0]; } } List keyProperties = KeyPropertiesCache(type); List explicitKeyProperties = ExplicitKeyPropertiesCache(type); if (keyProperties.Count == 0 && explicitKeyProperties.Count == 0) { throw new ArgumentException("Entity must have at least one [Key] or [ExplicitKey] property"); } string name = GetTableName(type); keyProperties.AddRange(explicitKeyProperties); StringBuilder sb = new StringBuilder(); sb.AppendFormat("DELETE FROM {0} WHERE ", name); ISqlAdapter adapter = GetFormatter(connection); for (int i = 0; i < keyProperties.Count; i++) { PropertyInfo property = keyProperties[i]; adapter.AppendColumnNameEqualsValue(sb, property.Name); if (i < keyProperties.Count - 1) { sb.Append(" AND "); } } int deleted = await connection.ExecuteAsync(sb.ToString(), entityToDelete, transaction, commandTimeout) .ConfigureAwait(false); return deleted > 0; } /// /// Delete all entities in the table related to the type T asynchronously using Task. /// /// Type of entity /// Open SqlConnection /// The transaction to run under, null (the default) if none /// Number of seconds before command execution timeout /// true if deleted, false if none found public static async Task DeleteAllAsync(this IDbConnection connection, IDbTransaction transaction = null, int? commandTimeout = null) where T : class { Type type = typeof(T); string statement = "DELETE FROM " + GetTableName(type); int deleted = await connection.ExecuteAsync(statement, null, transaction, commandTimeout) .ConfigureAwait(false); return deleted > 0; } } } public partial interface ISqlAdapter { /// /// Inserts into the database, returning the Id of the row created. /// /// The connection to use. /// The transaction to use. /// The command timeout to use. /// The table to insert into. /// The columns to set with this insert. /// The parameters to set for this insert. /// The key columns in this table. /// The entity to insert. /// The Id of the row created. Task InsertAsync(IDbConnection connection, IDbTransaction transaction, int? commandTimeout, string tableName, string columnList, string parameterList, IEnumerable keyProperties, object entityToInsert); } public partial class SqlServerAdapter { /// /// Inserts into the database, returning the Id of the row created. /// /// The connection to use. /// The transaction to use. /// The command timeout to use. /// The table to insert into. /// The columns to set with this insert. /// The parameters to set for this insert. /// The key columns in this table. /// The entity to insert. /// The Id of the row created. public async Task InsertAsync(IDbConnection connection, IDbTransaction transaction, int? commandTimeout, string tableName, string columnList, string parameterList, IEnumerable keyProperties, object entityToInsert) { string cmd = $"INSERT INTO {tableName} ({columnList}) values ({parameterList}); select @@ROWCOUNT num"; SqlMapper.GridReader multi = await connection.QueryMultipleAsync(cmd, entityToInsert, transaction, commandTimeout) .ConfigureAwait(false); dynamic first = multi.Read().FirstOrDefault(); if (first == null || first.num == null) { return 0; } int num = (int)first.num; return num; } } public partial class SqlCeServerAdapter { /// /// Inserts into the database, returning the Id of the row created. /// /// The connection to use. /// The transaction to use. /// The command timeout to use. /// The table to insert into. /// The columns to set with this insert. /// The parameters to set for this insert. /// The key columns in this table. /// The entity to insert. /// The Id of the row created. public async Task InsertAsync(IDbConnection connection, IDbTransaction transaction, int? commandTimeout, string tableName, string columnList, string parameterList, IEnumerable keyProperties, object entityToInsert) { string cmd = $"INSERT INTO {tableName} ({columnList}) values ({parameterList}); select @@ROWCOUNT num"; SqlMapper.GridReader multi = await connection.QueryMultipleAsync(cmd, entityToInsert, transaction, commandTimeout) .ConfigureAwait(false); dynamic first = multi.Read().FirstOrDefault(); if (first == null || first.num == null) { return 0; } int num = (int)first.num; return num; } } public partial class MySqlAdapter { /// /// Inserts into the database, returning the Id of the row created. /// /// The connection to use. /// The transaction to use. /// The command timeout to use. /// The table to insert into. /// The columns to set with this insert. /// The parameters to set for this insert. /// The key columns in this table. /// The entity to insert. /// The Id of the row created. public async Task InsertAsync(IDbConnection connection, IDbTransaction transaction, int? commandTimeout, string tableName, string columnList, string parameterList, IEnumerable keyProperties, object entityToInsert) { string cmd = $"INSERT INTO {tableName} ({columnList}) VALUES ({parameterList});select ROW_COUNT() num"; SqlMapper.GridReader multi = await connection.QueryMultipleAsync(cmd, entityToInsert, transaction, commandTimeout) .ConfigureAwait(false); dynamic first = multi.Read().FirstOrDefault(); if (first == null || first.num == null) { return 0; } int num = (int)first.num; return num; //await connection.ExecuteAsync(cmd, entityToInsert, transaction, commandTimeout).ConfigureAwait(false); //var r = await connection.QueryAsync("SELECT LAST_INSERT_ID() id", transaction: transaction, commandTimeout: commandTimeout).ConfigureAwait(false); //var id = r.First().id; //if (id == null) return 0; //var pi = keyProperties as PropertyInfo[] ?? keyProperties.ToArray(); //if (pi.Length == 0) return Convert.ToInt32(id); //var idp = pi[0]; //idp.SetValue(entityToInsert, Convert.ChangeType(id, idp.PropertyType), null); //return Convert.ToInt32(id); } } public partial class PostgresAdapter { /// /// Inserts into the database, returning the Id of the row created. /// /// The connection to use. /// The transaction to use. /// The command timeout to use. /// The table to insert into. /// The columns to set with this insert. /// The parameters to set for this insert. /// The key columns in this table. /// The entity to insert. /// The Id of the row created. public async Task InsertAsync(IDbConnection connection, IDbTransaction transaction, int? commandTimeout, string tableName, string columnList, string parameterList, IEnumerable keyProperties, object entityToInsert) { StringBuilder sb = new StringBuilder(); sb.AppendFormat("INSERT INTO {0} ({1}) VALUES ({2})", tableName, columnList, parameterList); // If no primary key then safe to assume a join table with not too much data to return PropertyInfo[] propertyInfos = keyProperties as PropertyInfo[] ?? keyProperties.ToArray(); if (propertyInfos.Length == 0) { sb.Append(" RETURNING *"); } else { sb.Append(" RETURNING "); bool first = true; foreach (PropertyInfo property in propertyInfos) { if (!first) { sb.Append(", "); } first = false; sb.Append(property.Name); } } IEnumerable results = await connection.QueryAsync(sb.ToString(), entityToInsert, transaction, commandTimeout) .ConfigureAwait(false); // Return the key by assinging the corresponding property in the object - by product is that it supports compound primary keys int id = 0; foreach (PropertyInfo p in propertyInfos) { object value = ((IDictionary)results.First())[p.Name.ToLower()]; p.SetValue(entityToInsert, value, null); if (id == 0) { id = Convert.ToInt32(value); } } return id; } } public partial class SQLiteAdapter { /// /// Inserts into the database, returning the Id of the row created. /// /// The connection to use. /// The transaction to use. /// The command timeout to use. /// The table to insert into. /// The columns to set with this insert. /// The parameters to set for this insert. /// The key columns in this table. /// The entity to insert. /// The Id of the row created. public async Task InsertAsync(IDbConnection connection, IDbTransaction transaction, int? commandTimeout, string tableName, string columnList, string parameterList, IEnumerable keyProperties, object entityToInsert) { string cmd = $"INSERT INTO {tableName} ({columnList}) VALUES ({parameterList}); SELECT last_insert_rowid() id"; SqlMapper.GridReader multi = await connection.QueryMultipleAsync(cmd, entityToInsert, transaction, commandTimeout) .ConfigureAwait(false); int id = (int)multi.Read().First().id; PropertyInfo[] pi = keyProperties as PropertyInfo[] ?? keyProperties.ToArray(); if (pi.Length == 0) { return id; } PropertyInfo idp = pi[0]; idp.SetValue(entityToInsert, Convert.ChangeType(id, idp.PropertyType), null); return id; } } public partial class FbAdapter { /// /// Inserts into the database, returning the Id of the row created. /// /// The connection to use. /// The transaction to use. /// The command timeout to use. /// The table to insert into. /// The columns to set with this insert. /// The parameters to set for this insert. /// The key columns in this table. /// The entity to insert. /// The Id of the row created. public async Task InsertAsync(IDbConnection connection, IDbTransaction transaction, int? commandTimeout, string tableName, string columnList, string parameterList, IEnumerable keyProperties, object entityToInsert) { string cmd = $"insert into {tableName} ({columnList}) values ({parameterList})"; await connection.ExecuteAsync(cmd, entityToInsert, transaction, commandTimeout).ConfigureAwait(false); PropertyInfo[] propertyInfos = keyProperties as PropertyInfo[] ?? keyProperties.ToArray(); string keyName = propertyInfos[0].Name; IEnumerable r = await connection.QueryAsync($"SELECT FIRST 1 {keyName} ID FROM {tableName} ORDER BY {keyName} DESC", transaction: transaction, commandTimeout: commandTimeout).ConfigureAwait(false); dynamic id = r.First().ID; if (id == null) { return 0; } if (propertyInfos.Length == 0) { return Convert.ToInt32(id); } PropertyInfo idp = propertyInfos[0]; idp.SetValue(entityToInsert, Convert.ChangeType(id, idp.PropertyType), null); return Convert.ToInt32(id); } }