mirror of
				https://github.com/nsnail/FreeSql.git
				synced 2025-11-04 09:15:27 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			197 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			197 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
 | 
						||
using FreeSql.DataAnnotations;
 | 
						||
using System;
 | 
						||
using System.Collections;
 | 
						||
using System.Collections.Concurrent;
 | 
						||
using System.Collections.Generic;
 | 
						||
using System.Data.Common;
 | 
						||
using System.Diagnostics;
 | 
						||
using System.Linq;
 | 
						||
using System.Linq.Expressions;
 | 
						||
using System.Threading;
 | 
						||
 | 
						||
namespace FreeSql
 | 
						||
{
 | 
						||
    /// <summary>
 | 
						||
    /// 包括 CreateTime/UpdateTime/IsDeleted 的实体基类
 | 
						||
    /// </summary>
 | 
						||
    [Table(DisableSyncStructure = true)]
 | 
						||
    public abstract class BaseEntity
 | 
						||
    {
 | 
						||
        internal static IFreeSql _ormPriv;
 | 
						||
        /// <summary>
 | 
						||
        /// 全局 IFreeSql orm 对象
 | 
						||
        /// </summary>
 | 
						||
        public static IFreeSql Orm => _ormPriv ?? throw new Exception(@"使用前请初始化 BaseEntity.Initialization(new FreeSqlBuilder()
 | 
						||
.UseAutoSyncStructure(true)
 | 
						||
.UseConnectionString(DataType.Sqlite, ""data source=test.db;max pool size=5"")
 | 
						||
.Build());");
 | 
						||
        internal static Func<IUnitOfWork> _resolveUow;
 | 
						||
 | 
						||
        /// <summary>
 | 
						||
        /// 初始化BaseEntity
 | 
						||
        /// BaseEntity.Initialization(new FreeSqlBuilder()
 | 
						||
        /// <para></para>
 | 
						||
        /// .UseAutoSyncStructure(true)
 | 
						||
        /// <para></para>
 | 
						||
        /// .UseConnectionString(DataType.Sqlite, "data source=test.db;max pool size=5")
 | 
						||
        /// <para></para>
 | 
						||
        /// .Build());
 | 
						||
        /// </summary>
 | 
						||
        /// <param name="fsql">IFreeSql orm 对象</param>
 | 
						||
        /// <param name="resolveUow">工作单元(事务)委托,如果不使用事务请传 null<para></para>解释:由于AsyncLocal平台兼容不好,所以交给外部管理</param>
 | 
						||
        public static void Initialization(IFreeSql fsql, Func<IUnitOfWork> resolveUow)
 | 
						||
        {
 | 
						||
            _ormPriv = fsql;
 | 
						||
            _ormPriv.Aop.CurdBefore += (s, e) => Trace.WriteLine($"\r\n线程{Thread.CurrentThread.ManagedThreadId}: {e.Sql}\r\n");
 | 
						||
            if (_configEntityQueues.Any())
 | 
						||
            {
 | 
						||
                lock (_configEntityLock)
 | 
						||
                {
 | 
						||
                    while (_configEntityQueues.TryDequeue(out var cei))
 | 
						||
                        _ormPriv.CodeFirst.ConfigEntity(cei.EntityType, cei.Fluent);
 | 
						||
                }
 | 
						||
            }
 | 
						||
            _resolveUow = resolveUow;
 | 
						||
        }
 | 
						||
        
 | 
						||
        class ConfigEntityInfo
 | 
						||
        {
 | 
						||
            public Type EntityType;
 | 
						||
            public Action<TableFluent> Fluent;
 | 
						||
        }
 | 
						||
        static ConcurrentQueue<ConfigEntityInfo> _configEntityQueues = new ConcurrentQueue<ConfigEntityInfo>();
 | 
						||
        static object _configEntityLock = new object();
 | 
						||
        internal static void ConfigEntity(Type entityType, Action<TableFluent> fluent)
 | 
						||
        {
 | 
						||
            lock (_configEntityLock)
 | 
						||
            {
 | 
						||
                if (_ormPriv == null)
 | 
						||
                    _configEntityQueues.Enqueue(new ConfigEntityInfo { EntityType = entityType, Fluent = fluent });
 | 
						||
                else
 | 
						||
                    _ormPriv.CodeFirst.ConfigEntity(entityType, fluent);
 | 
						||
            }
 | 
						||
        }
 | 
						||
 | 
						||
        /// <summary>
 | 
						||
        /// 创建时间
 | 
						||
        /// </summary>
 | 
						||
        [Column(Position = -4)]
 | 
						||
        public virtual DateTime CreateTime { get; set; } = DateTime.Now;
 | 
						||
        /// <summary>
 | 
						||
        /// 更新时间
 | 
						||
        /// </summary>
 | 
						||
        [Column(Position = -3)]
 | 
						||
        public virtual DateTime UpdateTime { get; set; }
 | 
						||
        /// <summary>
 | 
						||
        /// 逻辑删除
 | 
						||
        /// </summary>
 | 
						||
        [Column(Position = -2)]
 | 
						||
        public virtual bool IsDeleted { get; set; }
 | 
						||
        /// <summary>
 | 
						||
        /// 排序
 | 
						||
        /// </summary>
 | 
						||
        [Column(Position = -1)]
 | 
						||
        public virtual int Sort { get; set; }
 | 
						||
    }
 | 
						||
 | 
						||
    [Table(DisableSyncStructure = true)]
 | 
						||
    public abstract class BaseEntityReadOnly<TEntity> : BaseEntity where TEntity : class
 | 
						||
    {
 | 
						||
        /// <summary>
 | 
						||
        /// 查询数据
 | 
						||
        /// </summary>
 | 
						||
        /// <returns></returns>
 | 
						||
        public static ISelect<TEntity> Select
 | 
						||
        {
 | 
						||
            get
 | 
						||
            {
 | 
						||
                var select = Orm.Select<TEntity>()
 | 
						||
                    .TrackToList(TrackToList) //自动为每个元素 Attach
 | 
						||
                    .WithTransaction(_resolveUow?.Invoke()?.GetOrBeginTransaction(false));
 | 
						||
                return select.WhereCascade(a => (a as BaseEntity).IsDeleted == false);
 | 
						||
            }
 | 
						||
        }
 | 
						||
 | 
						||
        static void TrackToList(object list)
 | 
						||
        {
 | 
						||
            if (list == null) return;
 | 
						||
            var ls = list as IList<TEntity>;
 | 
						||
            if (ls == null)
 | 
						||
            {
 | 
						||
                var ie = list as IEnumerable;
 | 
						||
                if (ie == null) return;
 | 
						||
                var isFirst = true;
 | 
						||
                IBaseRepository<TEntity> berepo = null;
 | 
						||
                foreach (var item in ie)
 | 
						||
                {
 | 
						||
                    if (item == null) return;
 | 
						||
                    if (isFirst)
 | 
						||
                    {
 | 
						||
                        isFirst = false;
 | 
						||
                        var itemType = item.GetType();
 | 
						||
                        if (itemType == typeof(object)) return;
 | 
						||
                        if (itemType.FullName.Contains("FreeSqlLazyEntity__")) itemType = itemType.BaseType;
 | 
						||
                        if (Orm.CodeFirst.GetTableByEntity(itemType)?.Primarys.Any() != true) return;
 | 
						||
                        if (item is BaseEntity<TEntity> == false) return;
 | 
						||
                    }
 | 
						||
                    var beitem = item as BaseEntity<TEntity>;
 | 
						||
                    if (beitem != null)
 | 
						||
                    {
 | 
						||
                        if (berepo == null) berepo = Orm.GetRepository<TEntity>();
 | 
						||
                        beitem.Repository = berepo;
 | 
						||
                        beitem.Attach();
 | 
						||
                    }
 | 
						||
                }
 | 
						||
                return;
 | 
						||
            }
 | 
						||
            if (ls.Any() == false) return;
 | 
						||
            if (ls.FirstOrDefault() is BaseEntity<TEntity> == false) return;
 | 
						||
            if (Orm.CodeFirst.GetTableByEntity(typeof(TEntity))?.Primarys.Any() != true) return;
 | 
						||
            IBaseRepository<TEntity> repo = null;
 | 
						||
            foreach (var item in ls)
 | 
						||
            {
 | 
						||
                var beitem = item as BaseEntity<TEntity>;
 | 
						||
                if (beitem != null)
 | 
						||
                {
 | 
						||
                    if (repo == null) repo = Orm.GetRepository<TEntity>();
 | 
						||
                    beitem.Repository = repo;
 | 
						||
                    beitem.Attach();
 | 
						||
                }
 | 
						||
            }
 | 
						||
        }
 | 
						||
 | 
						||
        /// <summary>
 | 
						||
        /// 查询条件,Where(a => a.Id > 10),支持导航对象查询,Where(a => a.Author.Email == "2881099@qq.com")
 | 
						||
        /// </summary>
 | 
						||
        /// <param name="exp">lambda表达式</param>
 | 
						||
        /// <returns></returns>
 | 
						||
        public static ISelect<TEntity> Where(Expression<Func<TEntity, bool>> exp) => Select.Where(exp);
 | 
						||
        /// <summary>
 | 
						||
        /// 查询条件,Where(true, a => a.Id > 10),支导航对象查询,Where(true, a => a.Author.Email == "2881099@qq.com")
 | 
						||
        /// </summary>
 | 
						||
        /// <param name="condition">true 时生效</param>
 | 
						||
        /// <param name="exp">lambda表达式</param>
 | 
						||
        /// <returns></returns>
 | 
						||
        public static ISelect<TEntity> WhereIf(bool condition, Expression<Func<TEntity, bool>> exp) => Select.WhereIf(condition, exp);
 | 
						||
 | 
						||
        /// <summary>
 | 
						||
        /// 仓储对象
 | 
						||
        /// </summary>
 | 
						||
        protected IBaseRepository<TEntity> Repository { get; set; }
 | 
						||
 | 
						||
        /// <summary>
 | 
						||
        /// 附加实体,在更新数据时,只更新变化的部分
 | 
						||
        /// </summary>
 | 
						||
        public TEntity Attach()
 | 
						||
        {
 | 
						||
            if (this.Repository == null)
 | 
						||
                this.Repository = Orm.GetRepository<TEntity>();
 | 
						||
 | 
						||
            var item = this as TEntity;
 | 
						||
            this.Repository.Attach(item);
 | 
						||
            return item;
 | 
						||
        }
 | 
						||
    }
 | 
						||
}
 |