using Microsoft.EntityFrameworkCore; using Microsoft.EntityFrameworkCore.Infrastructure; using Microsoft.EntityFrameworkCore.Metadata; using System; using System.Collections.Generic; using System.ComponentModel.DataAnnotations.Schema; using System.Data; using System.Data.Common; using System.Linq; using System.Linq.Expressions; using System.Reflection; using System.Threading.Tasks; using Znyc.Recruitment.Admin.Commons.Attributes; using Znyc.Recruitment.Admin.Commons.Core.DataManager; using Znyc.Recruitment.Admin.Commons.Entitys; using Znyc.Recruitment.Admin.Commons.Enums; using Znyc.Recruitment.Admin.Commons.Helpers; using Znyc.Recruitment.Admin.Commons.IDbContext; using Znyc.Recruitment.Admin.Commons.Pages; namespace Znyc.Recruitment.Admin.Commons.DbContextCore { /// /// DbContext上下文的实现 /// public class BaseDbContext : DbContext, IDbContextCore { /// /// protected BaseDbContext() { } /// /// /// public BaseDbContext(DbConnectionOptions dbConnectionOptions) { this.dbConnectionOptions = dbConnectionOptions; } /// /// 配置,初始化数据库引擎 /// protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder) { if (dbConnectionOptions == null) { dbConnectionOptions = DBServerProvider.GeDbConnectionOptions(); } string defaultSqlConnectionString = dbConnectionOptions.ConnectionString; DatabaseType dbType = dbConnectionOptions.DatabaseType; if (dbType == DatabaseType.SqlServer) { optionsBuilder.UseSqlServer(defaultSqlConnectionString); } else if (dbType == DatabaseType.MySql) { optionsBuilder.UseMySql(defaultSqlConnectionString, new MySqlServerVersion(new Version(5, 7, 1))); } else if (dbType == DatabaseType.Oracle) { optionsBuilder.UseOracle(defaultSqlConnectionString); } else if (dbType == DatabaseType.SQLite) { optionsBuilder.UseSqlite(defaultSqlConnectionString); } else if (dbType == DatabaseType.Npgsql) { optionsBuilder.UseNpgsql(defaultSqlConnectionString); } else { throw new NotSupportedException("The database is not supported"); } //使用查询跟踪行为 optionsBuilder.UseQueryTrackingBehavior(QueryTrackingBehavior.NoTracking); base.OnConfiguring(optionsBuilder); } /// /// 模型创建 /// /// protected override void OnModelCreating(ModelBuilder modelBuilder) { MappingEntityTypes(modelBuilder); base.OnModelCreating(modelBuilder); } /// /// /// private void MappingEntityTypes(ModelBuilder modelBuilder) { IList assemblies = RuntimeHelper.GetAllZnycAssemblies(); foreach (Assembly assembly in assemblies) { IEnumerable entityTypes = assembly.GetTypes() .Where(type => !string.IsNullOrWhiteSpace(type.Namespace)) .Where(type => type.IsClass) .Where(type => type.BaseType != null) .Where(type => typeof(Entity).IsAssignableFrom(type) || type.IsSubclassOf(typeof(BaseViewModel))); foreach (Type entityType in entityTypes) { if (modelBuilder.Model.FindEntityType(entityType) != null || entityType.Name == "Entity" || entityType.Name == "BaseEntity`1") { continue; } TableAttribute table = entityType.GetCustomAttributes().FirstOrDefault(); modelBuilder.Model.AddEntityType(entityType).SetTableName(table.Name); IMutableEntityType ientityTypes = modelBuilder.Model.FindEntityType(entityType); ShardingTableAttribute attr = entityType.GetCustomAttributes().FirstOrDefault(); if (attr != null && entityType != null) { modelBuilder.Model.FindEntityType(entityType) .SetTableName($"{entityType.Name}{attr.Splitter}{DateTime.Now.ToString(attr.Suffix)}"); } if (typeof(IDeleteAudited).IsAssignableFrom(typeof(Entity))) { modelBuilder.Entity().HasQueryFilter(m => ((IDeleteAudited)m).IsDeleted == false); } if (IsMultiTenant) { if (typeof(IMustHaveTenant).IsAssignableFrom(typeof(Entity))) { modelBuilder.Entity().HasQueryFilter(m => ((IMustHaveTenant)m).TenantId == ""); } } } } } #region 基础参数 /// /// 数据库配置名称,可在子类指定不同的配置名称,用于访问不同的数据库 /// protected string dbConfigName = ""; /// /// 是否开启多租户 /// protected bool isMultiTenant = false; /// /// protected DbConnectionOptions dbConnectionOptions; /// /// 数据库访问对象的外键约束 /// public bool IsMultiTenant => isMultiTenant; /// /// /// public DatabaseFacade GetDatabase() { return Database; } #endregion 基础参数 #region 接口实现 /// /// 新增实体 /// /// /// /// public new virtual int Add(T entity) where T : class { base.Add(entity); return SaveChanges(); } /// /// 新增实体 /// /// /// /// public virtual async Task AddAsync(T entity) where T : class { await base.AddAsync(entity); return await SaveChangesAsync(); } /// /// 批量新增 /// /// /// /// public virtual int AddRange(ICollection entities) where T : class { base.AddRange(entities); return SaveChanges(); } /// /// 批量新增 /// /// /// /// public virtual async Task AddRangeAsync(ICollection entities) where T : class { await base.AddRangeAsync(entities); return await SaveChangesAsync(); } /// /// 统计数量Count() /// /// /// 查询条件 /// public virtual int Count(Expression> where = null) where T : class { return where == null ? GetDbSet().Count() : GetDbSet().Count(where); } /// /// 统计数量Count() /// /// /// 查询条件 /// public virtual async Task CountAsync(Expression> where = null) where T : class { return await (where == null ? GetDbSet().CountAsync() : GetDbSet().CountAsync(where)); } /// /// 物理删除数据 /// /// /// 主键类型 /// 主键 /// public virtual int Delete(TKey key) where T : Entity { T entity = Find(key); Remove(entity); return SaveChanges(); } /// /// 执行Sql,返回影响记录行数 /// /// /// /// public virtual int ExecuteSqlWithNonQuery(string sql, params object[] parameters) { return Database.ExecuteSqlRaw(sql, parameters); } /// /// 执行Sql,返回影响记录行数 /// /// /// /// public virtual async Task ExecuteSqlWithNonQueryAsync(string sql, params object[] parameters) { return await Database.ExecuteSqlRawAsync(sql, parameters); } /// /// 编辑更新保存实体 /// /// /// /// public virtual int Edit(T entity) where T : class { base.Update(entity); base.Entry(entity).State = EntityState.Modified; return SaveChanges(); } /// /// 批量更新保存实体 /// 以添加状态开始跟踪给定的实体和任何其他尚未被跟踪的可访问实体 /// /// /// /// public virtual int EditRange(ICollection entities) where T : class { GetDbSet().AttachRange(entities.ToArray()); return SaveChanges(); } /// /// 是否存在,存在返回true,不存在返回false /// /// /// 查询条件 /// public virtual bool Exist(Expression> where = null) where T : class { return where == null ? GetDbSet().Any() : GetDbSet().Any(where); } /// /// 是否存在,存在返回true,不存在返回false /// /// /// /// public virtual async Task ExistAsync(Expression> where = null) where T : class { return await Task.FromResult(Exist(where)); } /// /// 根据条件进行查询数据 /// /// /// /// 查询条件 /// public virtual IQueryable FilterWithInclude(Func, IQueryable> include, Expression> where) where T : class { IQueryable result = GetDbSet().AsQueryable(); if (where != null) { result = GetDbSet().Where(@where); } if (include != null) { result = include(result); } return result; } /// /// 根据主键查询实体 /// /// /// /// public virtual T Find(object key) where T : class { return base.Find(key); } /// /// 根据主键查询实体 /// /// /// /// public virtual T Find(string key) where T : class { return base.Find(key); } /// /// 根据主键查询实体 /// /// /// /// /// public virtual T Find(TKey key) where T : Entity { return base.Find(key); } /// /// 根据主键查询实体 /// /// /// /// public virtual async Task FindAsync(object key) where T : class { return await base.FindAsync(key); } /// /// 根据主键查询实体 /// /// /// /// /// public virtual async Task FindAsync(TKey key) where T : Entity { return await base.FindAsync(key); } /// /// 根据条件查询实体,返回实体集合 /// /// /// 查询条件 /// 是否启用模型跟踪,默认为false不跟踪 /// public virtual IQueryable Get(Expression> where = null, bool asNoTracking = false) where T : class { IQueryable query = GetDbSet().AsQueryable(); if (where != null) { query = query.Where(@where); } if (asNoTracking) { query = query.AsNoTracking(); } return query; } /// /// 获取所有实体类型 /// /// public virtual List GetAllEntityTypes() { return Model.GetEntityTypes().ToList(); } /// /// /// /// public virtual DbSet GetDbSet() where T : class { if (Model.FindEntityType(typeof(T)) != null) { return Set(); } throw new Exception( $"类型{typeof(T).Name}未在数据库上下文中注册,请先在DbContextOption设置ModelAssemblyName以将所有实体类型注册到数据库上下文中。"); } /// /// 根据条件查询一个实体, /// 检验序列中是否包含有元素。引用类型的默认值default(T)为null,表示在序列中没有找到元素。 /// /// /// 查询条件 /// public virtual T GetSingleOrDefault(Expression> where = null) where T : class { return where == null ? GetDbSet().SingleOrDefault() : GetDbSet().SingleOrDefault(where); } /// /// 根据条件查询一个实体, /// 检验序列中是否包含有元素。引用类型的默认值default(T)为null,表示在序列中没有找到元素。 /// /// /// 查询条件 /// public virtual async Task GetSingleOrDefaultAsync(Expression> where = null) where T : class { return await (where == null ? GetDbSet().SingleOrDefaultAsync() : GetDbSet().SingleOrDefaultAsync(where)); } /// /// 更新指定字段的值 /// /// /// 数据实体 /// 指定字段 /// public virtual int Update(T model, params string[] updateColumns) where T : class { if (updateColumns != null && updateColumns.Length > 0) { if (Entry(model).State == EntityState.Added || Entry(model).State == EntityState.Detached) { GetDbSet().Attach(model); } foreach (string propertyName in updateColumns) { Entry(model).Property(propertyName).IsModified = true; } } else { Entry(model).State = EntityState.Modified; } return SaveChanges(); } /// /// 批量插入 /// /// /// 数据实体集合 /// 数据库表名称 public virtual void BulkInsert(IList entities, string destinationTableName = null) where T : class { if (!Database.IsSqlServer() && !Database.IsMySql()) { throw new NotSupportedException("This method only supports for SQL Server or MySql."); } } /// /// Sql查询,并返回实体 /// /// 查询对象实体 /// 返回/输出实体 /// sql语句 /// SQL参数 /// public virtual List SqlQuery(string sql, params object[] parameters) where T : class { return GetDbSet().FromSqlRaw(sql, parameters).Cast().ToList(); } /// /// Sql查询,并返回实体 /// /// 查询对象实体 /// 返回/输出实体 /// sql语句 /// SQL参数 /// public virtual async Task> SqlQueryAsync(string sql, params object[] parameters) where T : class where TView : class { return await GetDbSet().FromSqlRaw(sql, parameters).Cast().ToListAsync(); } /// /// 分页查询,SQL语句查询,返回指定输出对象集合 /// /// 查询对象实体 /// 返回/输出实体 /// sql语句 /// 排序条件 /// 当前页 /// 每页显示数量 /// /// public virtual PageResult SqlQueryByPagination(string sql, string[] orderBys, int pageIndex, int pageSize, Action eachAction = null) where T : class where TView : class { throw new NotImplementedException(); } /// /// 分页查询,SQL语句查询,返回数据实体集合 /// /// 查询对象实体 /// sql语句 /// 排序条件 /// 当前页 /// 每页显示数量 /// 查询SQL参数 /// public virtual PageResult SqlQueryByPagination(string sql, string[] orderBys, int pageIndex, int pageSize, params DbParameter[] parameters) where T : class, new() { throw new NotImplementedException(); } #region 显式编译的查询,提高查询性能 /// /// 根据主键查询返回一个实体,该方法是显式编译的查询 /// /// /// 主键类型 /// 主键值 /// public T GetByCompileQuery(TKey id) where T : Entity { return EF.CompileQuery((DbContext context, TKey id) => context.Set().Find(id))(this, id); } /// /// 根据主键查询返回一个实体,该方法是显式编译的查询 /// /// /// 主键类型 /// 主键值 /// public Task GetByCompileQueryAsync(TKey id) where T : Entity { return EF.CompileAsyncQuery((DbContext context, TKey id) => context.Set().Find(id))(this, id); } /// /// 根据条件查询返回实体集合,该方法是显式编译的查询 /// /// /// 查询条件 /// public IList GetByCompileQuery(Expression> filter) where T : class { if (filter == null) { filter = m => true; } return EF.CompileQuery((DbContext context) => context.Set().AsNoTracking().Where(filter).ToList())(this); } /// /// 根据条件查询返回实体集合,该方法是显式编译的查询 /// /// /// 查询条件 /// public Task> GetByCompileQueryAsync(Expression> filter) where T : class { if (filter == null) { filter = m => true; } return EF.CompileAsyncQuery((DbContext context) => context.Set().AsNoTracking().Where(filter).ToList())(this); } /// /// 根据条件查询一个实体,该方法是显式编译的查询 /// 检验序列中是否包含有元素。引用类型的默认值default(T)为null,表示在序列中没有找到元素。 /// /// /// 查询条件 /// public T FirstOrDefaultByCompileQuery(Expression> filter) where T : class { if (filter == null) { filter = m => true; } return EF.CompileQuery((DbContext context) => context.Set().AsNoTracking().FirstOrDefault(filter))(this); } /// /// 根据条件查询一个实体,该方法是显式编译的查询 /// 检验序列中是否包含有元素。引用类型的默认值default(T)为null,表示在序列中没有找到元素。 /// /// /// 查询条件 /// public Task FirstOrDefaultByCompileQueryAsync(Expression> filter) where T : class { if (filter == null) { filter = m => true; } return EF.CompileAsyncQuery((DbContext context) => context.Set().AsNoTracking().FirstOrDefault(filter))(this); } /// /// 根据条件查询一个实体,启用模型跟踪,该方法是显式编译的查询 /// 检验序列中是否包含有元素。引用类型的默认值default(T)为null,表示在序列中没有找到元素。 /// /// /// 查询条件 /// public T FirstOrDefaultWithTrackingByCompileQuery(Expression> filter) where T : class { if (filter == null) { filter = m => true; } return EF.CompileQuery((DbContext context) => context.Set().FirstOrDefault(filter))(this); } /// /// 根据条件查询一个实体,启用模型跟踪,该方法是显式编译的查询 /// 检验序列中是否包含有元素。引用类型的默认值default(T)为null,表示在序列中没有找到元素。 /// /// /// 查询条件 /// public Task FirstOrDefaultWithTrackingByCompileQueryAsync(Expression> filter) where T : class { if (filter == null) { filter = m => true; } return EF.CompileAsyncQuery((DbContext context) => context.Set().FirstOrDefault(filter))(this); } /// /// 统计数量Count(),该方法是显式编译的查询 /// /// /// 查询条件 /// public int CountByCompileQuery(Expression> filter) where T : class { if (filter == null) { filter = m => true; } return EF.CompileQuery((DbContext context) => context.Set().Count(filter))(this); } /// /// 统计数量Count(),该方法是显式编译的查询 /// /// /// 查询条件 /// public Task CountByCompileQueryAsync(Expression> filter) where T : class { if (filter == null) { filter = m => true; } return EF.CompileAsyncQuery((DbContext context) => context.Set().Count(filter))(this); } /// /// 根据sql语句返回DataTable数据 /// /// Sql语句 /// 执行超时时间,默认30毫秒 /// DbParameter[]参数 /// public virtual DataTable GetDataTable(string sql, int cmdTimeout = 30, params DbParameter[] parameters) { throw new NotImplementedException(); } /// /// 根据sql语句返回List数据 /// /// Sql语句 /// 执行超时时间,默认30毫秒 /// DbParameter[] 参数 /// public virtual List GetDataTables(string sql, int cmdTimeout = 30, params DbParameter[] parameters) { throw new NotImplementedException(); } #endregion 显式编译的查询,提高查询性能 #endregion 接口实现 } }