mirror of
				https://github.com/nsnail/FreeSql.git
				synced 2025-11-04 01:05:27 +08:00 
			
		
		
		
	initial commit
This commit is contained in:
		@@ -0,0 +1,43 @@
 | 
			
		||||
<Project Sdk="Microsoft.NET.Sdk">
 | 
			
		||||
 | 
			
		||||
	<PropertyGroup>
 | 
			
		||||
		<TargetFrameworks>netstandard2.0;net45;net40</TargetFrameworks>
 | 
			
		||||
		<GeneratePackageOnBuild>true</GeneratePackageOnBuild>
 | 
			
		||||
		<Authors>FreeSql;ncc;YeXiangQin</Authors>
 | 
			
		||||
		<Description>FreeSql 扩展包,实现 低代码、零实体、ZeroEntity,并且支持导航属性,级联保存 等功能.</Description>
 | 
			
		||||
		<PackageProjectUrl>https://github.com/2881099/FreeSql</PackageProjectUrl>
 | 
			
		||||
		<RepositoryUrl>https://github.com/2881099/FreeSql</RepositoryUrl>
 | 
			
		||||
		<RepositoryType>git</RepositoryType>
 | 
			
		||||
		<PackageLicenseExpression>MIT</PackageLicenseExpression>
 | 
			
		||||
		<PackageTags>FreeSql;ORM;低代码;</PackageTags>
 | 
			
		||||
		<PackageId>$(AssemblyName)</PackageId>
 | 
			
		||||
		<PackageIcon>logo.png</PackageIcon>
 | 
			
		||||
		<Title>$(AssemblyName)</Title>
 | 
			
		||||
		<IsPackable>true</IsPackable>
 | 
			
		||||
		<GenerateAssemblyInfo>true</GenerateAssemblyInfo>
 | 
			
		||||
		<SignAssembly>true</SignAssembly>
 | 
			
		||||
		<AssemblyOriginatorKeyFile>key.snk</AssemblyOriginatorKeyFile>
 | 
			
		||||
		<DelaySign>false</DelaySign>
 | 
			
		||||
		<Version>3.2.833</Version>
 | 
			
		||||
		<PackageReadmeFile>readme.md</PackageReadmeFile>
 | 
			
		||||
	</PropertyGroup>
 | 
			
		||||
 | 
			
		||||
	<ItemGroup>
 | 
			
		||||
		<None Include="../../readme.md" Pack="true" PackagePath="\" />
 | 
			
		||||
		<None Include="../../logo.png" Pack="true" PackagePath="\" />
 | 
			
		||||
	</ItemGroup>
 | 
			
		||||
 | 
			
		||||
	<PropertyGroup Condition="'$(Configuration)|$(TargetFramework)|$(Platform)'=='Debug|netstandard2.0|AnyCPU'">
 | 
			
		||||
		<DocumentationFile>FreeSql.Extensions.ZeroEntity.xml</DocumentationFile>
 | 
			
		||||
		<WarningLevel>3</WarningLevel>
 | 
			
		||||
	</PropertyGroup>
 | 
			
		||||
 | 
			
		||||
	<PropertyGroup Condition="'$(TargetFramework)' == 'net40'">
 | 
			
		||||
		<DefineConstants>net40</DefineConstants>
 | 
			
		||||
	</PropertyGroup>
 | 
			
		||||
 | 
			
		||||
	<ItemGroup>
 | 
			
		||||
		<ProjectReference Include="..\..\FreeSql\FreeSql.csproj" />
 | 
			
		||||
	</ItemGroup>
 | 
			
		||||
	
 | 
			
		||||
</Project>
 | 
			
		||||
@@ -0,0 +1,88 @@
 | 
			
		||||
<?xml version="1.0"?>
 | 
			
		||||
<doc>
 | 
			
		||||
    <assembly>
 | 
			
		||||
        <name>FreeSql.Extensions.ZeroEntity</name>
 | 
			
		||||
    </assembly>
 | 
			
		||||
    <members>
 | 
			
		||||
        <member name="M:FreeSql.Extensions.ZeroEntity.ZeroDbContext.#ctor(IFreeSql,FreeSql.Extensions.ZeroEntity.TableDescriptor[],System.Boolean)">
 | 
			
		||||
            <summary>
 | 
			
		||||
            创建新的ZeroDbCotext实例
 | 
			
		||||
            </summary>
 | 
			
		||||
            <param name="orm">IfreeSql 对象</param>
 | 
			
		||||
            <param name="schemas">动态表结构描述</param>
 | 
			
		||||
            <param name="syncStructure">是否强制同步表结构</param>
 | 
			
		||||
            <exception cref="T:FreeSql.Extensions.ZeroEntity.Models.SchemaValidationResult"> Schema 未验证通过时抛出验证异常</exception>
 | 
			
		||||
        </member>
 | 
			
		||||
        <member name="M:FreeSql.Extensions.ZeroEntity.ZeroDbContext.#ctor(IFreeSql)">
 | 
			
		||||
            <summary>
 | 
			
		||||
            初始化一个 ZeroDbContext 对象,暂不指定任何Schema
 | 
			
		||||
            </summary>
 | 
			
		||||
            <param name="orm"></param>
 | 
			
		||||
        </member>
 | 
			
		||||
        <member name="M:FreeSql.Extensions.ZeroEntity.ZeroDbContext.SyncTableStructure(System.String)">
 | 
			
		||||
            <summary>
 | 
			
		||||
            同步指定表结构
 | 
			
		||||
            </summary>
 | 
			
		||||
            <param name="name"></param>
 | 
			
		||||
        </member>
 | 
			
		||||
        <member name="P:FreeSql.Extensions.ZeroEntity.ZeroDbContext.Select">
 | 
			
		||||
            <summary>
 | 
			
		||||
            【有状态管理】自动 Include 查询
 | 
			
		||||
            </summary>
 | 
			
		||||
        </member>
 | 
			
		||||
        <member name="M:FreeSql.Extensions.ZeroEntity.ZeroDbContext.SelectNoTracking(System.String)">
 | 
			
		||||
            <summary>
 | 
			
		||||
            【无状态管理】指定表查询
 | 
			
		||||
            </summary>
 | 
			
		||||
        </member>
 | 
			
		||||
        <member name="P:FreeSql.Extensions.ZeroEntity.ZeroDbContext.ChangeReport.ChangeInfo.BeforeObject">
 | 
			
		||||
            <summary>
 | 
			
		||||
            Type = Update 的时候,获取更新之前的对象
 | 
			
		||||
            </summary>
 | 
			
		||||
        </member>
 | 
			
		||||
        <member name="P:FreeSql.Extensions.ZeroEntity.ZeroDbContext.ChangeReport.Report">
 | 
			
		||||
            <summary>
 | 
			
		||||
            实体变化记录
 | 
			
		||||
            </summary>
 | 
			
		||||
        </member>
 | 
			
		||||
        <member name="P:FreeSql.Extensions.ZeroEntity.ZeroDbContext.ChangeReport.OnChange">
 | 
			
		||||
            <summary>
 | 
			
		||||
            实体变化事件
 | 
			
		||||
            </summary>
 | 
			
		||||
        </member>
 | 
			
		||||
        <member name="M:FreeSql.Extensions.ZeroEntity.ZeroDbContext.SelectImpl.LeftJoin(System.String,System.String[])">
 | 
			
		||||
            <summary>
 | 
			
		||||
            举例1:LeftJoin("table1", "id", "user.id") -> LEFT JOIN [table1] b ON b.[id] = a.[id]<para></para>
 | 
			
		||||
            举例2:LeftJoin("table1", "id", "user.id", "xid", "user.xid") -> LEFT JOIN [table1] b ON b.[id] = [a].id] AND b.[xid] = a.[xid]
 | 
			
		||||
            </summary>
 | 
			
		||||
        </member>
 | 
			
		||||
        <member name="M:FreeSql.Extensions.ZeroEntity.ZeroDbContext.SelectImpl.InnerJoin(System.String,System.String[])">
 | 
			
		||||
            <summary>
 | 
			
		||||
            举例1:InnerJoin("table1", "id", "user.id") -> INNER JOIN [table1] b ON b.[id] = a.[id]<para></para>
 | 
			
		||||
            举例2:InnerJoin("table1", "id", "user.id", "xid", "user.xid") -> INNER JOIN [table1] b ON b.[id] = [a].id] AND b.[xid] = a.[xid]
 | 
			
		||||
            </summary>
 | 
			
		||||
        </member>
 | 
			
		||||
        <member name="M:FreeSql.Extensions.ZeroEntity.ZeroDbContext.SelectImpl.RightJoin(System.String,System.String[])">
 | 
			
		||||
            <summary>
 | 
			
		||||
            举例1:RightJoin("table1", "id", "user.id") -> RIGTH JOIN [table1] b ON b.[id] = a.[id]<para></para>
 | 
			
		||||
            举例2:RightJoin("table1", "id", "user.id", "xid", "user.xid") -> RIGTH JOIN [table1] b ON b.[id] = [a].id] AND b.[xid] = a.[xid]
 | 
			
		||||
            </summary>
 | 
			
		||||
        </member>
 | 
			
		||||
        <member name="M:FreeSql.Extensions.ZeroEntity.ZeroDbContext.SelectImpl.Where(System.Collections.Generic.IEnumerable{System.Collections.Generic.Dictionary{System.String,System.Object}})">
 | 
			
		||||
            <summary>
 | 
			
		||||
            WHERE [Id] IN (...)
 | 
			
		||||
            </summary>
 | 
			
		||||
        </member>
 | 
			
		||||
        <member name="M:FreeSql.Extensions.ZeroEntity.ZeroDbContext.SelectImpl.Where(System.Object)">
 | 
			
		||||
            <summary>
 | 
			
		||||
            Where(new { Year = 2017, CategoryId = 198, IsPublished = true })<para></para>
 | 
			
		||||
            WHERE [Year] = 2017 AND [CategoryId] = 198 AND [IsPublished] = 1
 | 
			
		||||
            </summary>
 | 
			
		||||
        </member>
 | 
			
		||||
        <member name="M:FreeSql.Extensions.ZeroEntity.ZeroDbContext.SelectImpl.Where(System.String,System.Object)">
 | 
			
		||||
            <summary>
 | 
			
		||||
            WHERE [field] = ..
 | 
			
		||||
            </summary>
 | 
			
		||||
        </member>
 | 
			
		||||
    </members>
 | 
			
		||||
</doc>
 | 
			
		||||
@@ -0,0 +1,9 @@
 | 
			
		||||
using System;
 | 
			
		||||
 | 
			
		||||
namespace FreeSql.Extensions.ZeroEntity.Models
 | 
			
		||||
{
 | 
			
		||||
    public class SchemaValidationException : Exception
 | 
			
		||||
    {
 | 
			
		||||
        public SchemaValidationException(string message) : base(message) { }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -0,0 +1,21 @@
 | 
			
		||||
using System;
 | 
			
		||||
using System.Collections.Generic;
 | 
			
		||||
using System.Text;
 | 
			
		||||
 | 
			
		||||
namespace FreeSql.Extensions.ZeroEntity.Models
 | 
			
		||||
{
 | 
			
		||||
    public class SchemaValidationResult
 | 
			
		||||
    {
 | 
			
		||||
        public readonly static SchemaValidationResult _schemaValidationResult = new SchemaValidationResult("");
 | 
			
		||||
 | 
			
		||||
        public static SchemaValidationResult SuccessedResult => _schemaValidationResult;
 | 
			
		||||
 | 
			
		||||
        public SchemaValidationResult(string errorMessage)
 | 
			
		||||
        {
 | 
			
		||||
            ErrorMessage = errorMessage;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        public string ErrorMessage { get; set; }
 | 
			
		||||
        public bool Succeeded { get; set; } = false;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -0,0 +1,916 @@
 | 
			
		||||
using FreeSql.DataAnnotations;
 | 
			
		||||
using FreeSql.Internal;
 | 
			
		||||
using FreeSql.Internal.CommonProvider;
 | 
			
		||||
using FreeSql.Internal.Model;
 | 
			
		||||
using System;
 | 
			
		||||
using System.Collections.Generic;
 | 
			
		||||
using System.Data;
 | 
			
		||||
using System.Data.Common;
 | 
			
		||||
using System.Linq;
 | 
			
		||||
using System.Text;
 | 
			
		||||
using T = System.Collections.Generic.Dictionary<string, object>;
 | 
			
		||||
 | 
			
		||||
namespace FreeSql.Extensions.ZeroEntity
 | 
			
		||||
{
 | 
			
		||||
	partial class ZeroDbContext
 | 
			
		||||
	{
 | 
			
		||||
		public class SelectImpl
 | 
			
		||||
		{
 | 
			
		||||
			ZeroDbContext _dbcontext;
 | 
			
		||||
			IFreeSql _orm => _dbcontext._orm;
 | 
			
		||||
			List<ZeroTableInfo> _tables => _dbcontext._tables;
 | 
			
		||||
			int _mainTableIndex = -1;
 | 
			
		||||
			List<TableAliasInfo> _tableAlias;
 | 
			
		||||
			ISelect<TestDynamicFilterInfo> _select;
 | 
			
		||||
			Select0Provider _selectProvider;
 | 
			
		||||
			string _field;
 | 
			
		||||
			Dictionary<string, string> _fieldAlias;
 | 
			
		||||
			List<Action<DbDataReaderContext>> _fieldReader;
 | 
			
		||||
			string _groupBy;
 | 
			
		||||
			List<DbParameter> _params = new List<DbParameter>();
 | 
			
		||||
			CommonUtils _common => _selectProvider._commonUtils;
 | 
			
		||||
			bool _useStates = true;
 | 
			
		||||
			bool _includeAll = false;
 | 
			
		||||
 | 
			
		||||
			SelectImpl() { }
 | 
			
		||||
			internal SelectImpl(ZeroDbContext dbcontext, string tableName)
 | 
			
		||||
			{
 | 
			
		||||
				_dbcontext = dbcontext;
 | 
			
		||||
				var tableIndex = _tables.FindIndex(a => a.CsName.ToLower() == tableName?.ToLower());
 | 
			
		||||
				if (tableIndex == -1) throw new Exception($"未定义表名 {tableName}");
 | 
			
		||||
 | 
			
		||||
				_mainTableIndex = tableIndex;
 | 
			
		||||
				_tableAlias = new List<TableAliasInfo>();
 | 
			
		||||
				_select = _orm.Select<TestDynamicFilterInfo>()
 | 
			
		||||
					.AsTable((t, old) => _tables[tableIndex].DbName);
 | 
			
		||||
				_selectProvider = _select as Select0Provider;
 | 
			
		||||
				_fieldAlias = new Dictionary<string, string>();
 | 
			
		||||
				_fieldReader = new List<Action<DbDataReaderContext>>();
 | 
			
		||||
				FlagFetchResult(_tables[_mainTableIndex], "a", "");
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			public SelectImpl NoTracking()
 | 
			
		||||
			{
 | 
			
		||||
				_useStates = false;
 | 
			
		||||
				return this;
 | 
			
		||||
			}
 | 
			
		||||
			public SelectImpl IncludeAll()
 | 
			
		||||
			{
 | 
			
		||||
				var ignores = new Dictionary<string, bool>();
 | 
			
		||||
				_includeAll = true;
 | 
			
		||||
				LocalAutoInclude(_tables[_mainTableIndex], "a");
 | 
			
		||||
				return this;
 | 
			
		||||
 | 
			
		||||
				void LocalAutoInclude(ZeroTableInfo table, string alias, string navPath = "")
 | 
			
		||||
				{
 | 
			
		||||
					if (ignores.ContainsKey(table.CsName)) return;
 | 
			
		||||
					ignores.Add(table.CsName, true);
 | 
			
		||||
					TableAliasInfo tableAlias = null;
 | 
			
		||||
					if (table != _tables[_mainTableIndex])
 | 
			
		||||
						tableAlias = FlagFetchResult(table, alias.ToString(), navPath);
 | 
			
		||||
 | 
			
		||||
					var leftJoins = table.Navigates.Where(a => a.Value.RefType == TableRefType.ManyToOne || a.Value.RefType == TableRefType.OneToOne).ToArray();
 | 
			
		||||
					foreach (var join in leftJoins)
 | 
			
		||||
					{
 | 
			
		||||
						var joinTable = join.Value.RefTable;
 | 
			
		||||
						if (ignores.ContainsKey(joinTable.CsName)) return;
 | 
			
		||||
 | 
			
		||||
						var joinAlias = GetMaxAlias();
 | 
			
		||||
						var joinOn = string.Join(" AND ", join.Value.RefColumns.Select((bname, idx) =>
 | 
			
		||||
							$"{joinAlias}.{_common.QuoteSqlName(join.Value.RefTable.ColumnsByCs[bname].Attribute.Name)} = {alias}.{_common.QuoteSqlName(join.Value.Table.ColumnsByCs[join.Value.Columns[idx]].Attribute.Name)}"));
 | 
			
		||||
						_select.LeftJoin($"{_common.QuoteSqlName(join.Value.RefTable.DbName)} {joinAlias} ON {joinOn}");
 | 
			
		||||
 | 
			
		||||
						LocalAutoInclude(joinTable, joinAlias, $"{navPath}.{join.Key}");
 | 
			
		||||
					}
 | 
			
		||||
					if (tableAlias == null) tableAlias = _tableAlias.Where(a => a.Alias == alias).First();
 | 
			
		||||
					var includeManys = table.Navigates.Where(a => a.Value.RefType == TableRefType.OneToMany || a.Value.RefType == TableRefType.ManyToMany).ToArray();
 | 
			
		||||
					foreach (var includeMany in includeManys)
 | 
			
		||||
						tableAlias.IncludeMany.Add(NativeTuple.Create(includeMany.Key, (Action<SelectImpl>)null));
 | 
			
		||||
				}
 | 
			
		||||
			}
 | 
			
		||||
			public SelectImpl Include(string navigate, Action<SelectImpl> then = null)
 | 
			
		||||
			{
 | 
			
		||||
				var alias = _tableAlias[0];
 | 
			
		||||
				var navPath = navigate.Split('.');
 | 
			
		||||
				var navPath2 = "";
 | 
			
		||||
				for (var navIdx = 0; navIdx < navPath.Length; navIdx++)
 | 
			
		||||
				{
 | 
			
		||||
					if (alias.Table.Navigates.TryGetValue(navPath[navIdx], out var nav) == false) throw new Exception($"{alias.Table.CsName} 未定义导航属性 {navPath[navIdx]}");
 | 
			
		||||
					if (nav.RefType == TableRefType.OneToMany || nav.RefType == TableRefType.ManyToMany)
 | 
			
		||||
					{
 | 
			
		||||
						if (navIdx < navPath.Length - 1) throw new Exception($"导航属性 OneToMany/ManyToMany {navPath[navIdx]} 不能处于中间位置");
 | 
			
		||||
						if (alias.IncludeMany.Where(a => a.Item1 == nav.NavigateKey).Any() == false)
 | 
			
		||||
							alias.IncludeMany.Add(NativeTuple.Create(nav.NavigateKey, then));
 | 
			
		||||
						break;
 | 
			
		||||
					}
 | 
			
		||||
					navPath2 = navIdx == 0 ? nav.NavigateKey : $"{navPath2}.{nav.NavigateKey}";
 | 
			
		||||
					var navAlias = _tableAlias.Where(a => string.Join(".", a.NavPath) == navPath2).FirstOrDefault();
 | 
			
		||||
					if (navAlias == null)
 | 
			
		||||
					{
 | 
			
		||||
						var joinAlias = GetMaxAlias();
 | 
			
		||||
						var joinOn = string.Join(" AND ", nav.RefColumns.Select((bname, idx) =>
 | 
			
		||||
							$"{joinAlias}.{_common.QuoteSqlName(nav.RefTable.ColumnsByCs[bname].Attribute.Name)} = {alias.Alias}.{_common.QuoteSqlName(nav.Table.ColumnsByCs[nav.Columns[idx]].Attribute.Name)}"));
 | 
			
		||||
						_select.LeftJoin($"{_common.QuoteSqlName(nav.RefTable.DbName)} {joinAlias} ON {joinOn}");
 | 
			
		||||
						navAlias = FlagFetchResult(nav.RefTable, joinAlias, navPath2);
 | 
			
		||||
					}
 | 
			
		||||
					alias = navAlias;
 | 
			
		||||
				}
 | 
			
		||||
				return this;
 | 
			
		||||
			}
 | 
			
		||||
			//public SelectImpl IncludeSubQuery(string resultKey, string tableName, Action<SelectImpl> then)
 | 
			
		||||
			//{
 | 
			
		||||
			//	var query = _dbcontext.SelectNoTracking(tableName);
 | 
			
		||||
			//	query._tableAlias.AddRange(_tableAlias);
 | 
			
		||||
			//	return this;
 | 
			
		||||
			//}
 | 
			
		||||
 | 
			
		||||
			/// <summary>
 | 
			
		||||
			/// 举例1:LeftJoin("table1", "id", "user.id") -> LEFT JOIN [table1] b ON b.[id] = a.[id]<para></para>
 | 
			
		||||
			/// 举例2:LeftJoin("table1", "id", "user.id", "xid", "user.xid") -> LEFT JOIN [table1] b ON b.[id] = [a].id] AND b.[xid] = a.[xid]
 | 
			
		||||
			/// </summary>
 | 
			
		||||
			public SelectImpl LeftJoin(string tableName, params string[] onFields) => Join("LEFT JOIN", tableName, onFields);
 | 
			
		||||
			/// <summary>
 | 
			
		||||
			/// 举例1:InnerJoin("table1", "id", "user.id") -> INNER JOIN [table1] b ON b.[id] = a.[id]<para></para>
 | 
			
		||||
			/// 举例2:InnerJoin("table1", "id", "user.id", "xid", "user.xid") -> INNER JOIN [table1] b ON b.[id] = [a].id] AND b.[xid] = a.[xid]
 | 
			
		||||
			/// </summary>
 | 
			
		||||
			public SelectImpl InnerJoin(string tableName, params string[] onFields) => Join("INNER JOIN", tableName, onFields);
 | 
			
		||||
			/// <summary>
 | 
			
		||||
			/// 举例1:RightJoin("table1", "id", "user.id") -> RIGTH JOIN [table1] b ON b.[id] = a.[id]<para></para>
 | 
			
		||||
			/// 举例2:RightJoin("table1", "id", "user.id", "xid", "user.xid") -> RIGTH JOIN [table1] b ON b.[id] = [a].id] AND b.[xid] = a.[xid]
 | 
			
		||||
			/// </summary>
 | 
			
		||||
			public SelectImpl RightJoin(string tableName, params string[] onFields) => Join("RIGTH JOIN", tableName, onFields);
 | 
			
		||||
			SelectImpl Join(string joinType, string tableName, params string[] onFields)
 | 
			
		||||
			{
 | 
			
		||||
				if (onFields == null || onFields.Length == 0) throw new Exception($"{joinType} 参数 {nameof(onFields)} 不能为空");
 | 
			
		||||
				if (onFields.Length % 2 != 0) throw new Exception($"{joinType} 参数 {nameof(onFields)} 数组长度必须为偶数,正确:.LeftJoin(\"table1\", \"id\", \"user.id\")");
 | 
			
		||||
 | 
			
		||||
				var table = _tables.Where(a => a.CsName.ToLower() == tableName?.ToLower()).FirstOrDefault();
 | 
			
		||||
				if (table == null) throw new Exception($"未定义表名 {tableName}");
 | 
			
		||||
				var alias = GetMaxAlias();
 | 
			
		||||
				var navKey = tableName;
 | 
			
		||||
				for (var a = 2; true; a++)
 | 
			
		||||
				{
 | 
			
		||||
					if (_tables[_mainTableIndex].ColumnsByCs.ContainsKey(navKey))
 | 
			
		||||
					{
 | 
			
		||||
						navKey = $"{tableName}{a}";
 | 
			
		||||
						continue;
 | 
			
		||||
					}
 | 
			
		||||
					if (_tableAlias.Any(b => b.NavPath.Length == 1 && b.NavPath.Last() == navKey))
 | 
			
		||||
					{
 | 
			
		||||
						navKey = $"{tableName}{a}";
 | 
			
		||||
						continue;
 | 
			
		||||
					}
 | 
			
		||||
					break;
 | 
			
		||||
				}
 | 
			
		||||
				FlagFetchResult(table, alias, navKey);
 | 
			
		||||
				var joinOn = new string[onFields.Length / 2];
 | 
			
		||||
				for (var a = 0; a < onFields.Length; a += 2)
 | 
			
		||||
				{
 | 
			
		||||
					var field1 = ParseField(table, alias, onFields[a]);
 | 
			
		||||
					if (field1 == null) throw new Exception($"未匹配字段名 {onFields[a]}");
 | 
			
		||||
					var field2 = ParseField(table, alias, onFields[a + 1]);
 | 
			
		||||
					if (field2 == null) throw new Exception($"未匹配字段名 {onFields[a + 1]}");
 | 
			
		||||
					joinOn[a / 2] = $"{field1.Item1} = {field2.Item1}";
 | 
			
		||||
				}
 | 
			
		||||
				_select.RawJoin($"{joinType} {_common.QuoteSqlName(table.DbName)} {alias} ON {string.Join(" AND ", joinOn)}");
 | 
			
		||||
				return this;
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			class TableAliasInfo
 | 
			
		||||
			{
 | 
			
		||||
				public string Alias { get; set; }
 | 
			
		||||
				public ZeroTableInfo Table { get; set; }
 | 
			
		||||
				public string[] NavPath { get; set; }
 | 
			
		||||
				public List<NativeTuple<string, Action<SelectImpl>>> IncludeMany { get; set; } = new List<NativeTuple<string, Action<SelectImpl>>>();
 | 
			
		||||
			}
 | 
			
		||||
			class DbDataReaderContext
 | 
			
		||||
			{
 | 
			
		||||
				public DbDataReader Reader { get; set; }
 | 
			
		||||
				public int Index { get; set; }
 | 
			
		||||
				public T Result { get; set; }
 | 
			
		||||
			}
 | 
			
		||||
			string GetMaxAlias()
 | 
			
		||||
			{
 | 
			
		||||
				var max = (int)_tableAlias.Where(a => a.Alias.Length == 1).Max(a => a.Alias[0]);
 | 
			
		||||
				if (max < 'a') max = 'a';
 | 
			
		||||
				for (var a = 1; true; a++)
 | 
			
		||||
				{
 | 
			
		||||
					var alias = ((char)(max + a)).ToString();
 | 
			
		||||
					if (_tableAlias.Where(b => b.Alias == alias).Any()) continue;
 | 
			
		||||
					return alias;
 | 
			
		||||
				}
 | 
			
		||||
			}
 | 
			
		||||
			TableAliasInfo FlagFetchResult(ZeroTableInfo table, string alias, string navPath)
 | 
			
		||||
			{
 | 
			
		||||
				var tableAlias = _tableAlias.Where(a => a.Alias == alias).FirstOrDefault();
 | 
			
		||||
				if (tableAlias == null)
 | 
			
		||||
				{
 | 
			
		||||
					var navPathArray = navPath.Split('.').Where(a => string.IsNullOrWhiteSpace(a) == false).ToArray();
 | 
			
		||||
					_tableAlias.Add(tableAlias = new TableAliasInfo
 | 
			
		||||
					{
 | 
			
		||||
						Alias = alias,
 | 
			
		||||
						Table = table,
 | 
			
		||||
						NavPath = navPathArray
 | 
			
		||||
					});
 | 
			
		||||
				}
 | 
			
		||||
				var sbfield = new StringBuilder();
 | 
			
		||||
				if (string.IsNullOrEmpty(_field) == false) sbfield.Append(", ").Append(_field);
 | 
			
		||||
				foreach (var col in table.Columns.Values)
 | 
			
		||||
				{
 | 
			
		||||
					var colName = col.Attribute.Name;
 | 
			
		||||
					for (var a = 2; true; a++)
 | 
			
		||||
					{
 | 
			
		||||
						if (_fieldAlias.ContainsKey(colName)) colName = $"{col.Attribute.Name}{a}";
 | 
			
		||||
						else break;
 | 
			
		||||
					}
 | 
			
		||||
					_fieldAlias.Add(colName, col.Attribute.Name);
 | 
			
		||||
					sbfield.Append(", ").Append(alias).Append(".").Append(_common.QuoteSqlName(col.Attribute.Name));
 | 
			
		||||
					if (colName != col.Attribute.Name) sbfield.Append(_common.FieldAsAlias(colName));
 | 
			
		||||
				}
 | 
			
		||||
				_field = sbfield.Remove(0, 2).ToString();
 | 
			
		||||
				sbfield.Clear();
 | 
			
		||||
 | 
			
		||||
				_fieldReader.Add(dr =>
 | 
			
		||||
				{
 | 
			
		||||
					var pkIsNull = false;
 | 
			
		||||
					foreach (var col in table.Columns.Values)
 | 
			
		||||
					{
 | 
			
		||||
						if (pkIsNull == false && col.Attribute.IsPrimary)
 | 
			
		||||
						{
 | 
			
		||||
							pkIsNull = dr.Reader.IsDBNull(dr.Index);
 | 
			
		||||
							if (pkIsNull) dr.Result.Clear();
 | 
			
		||||
						}
 | 
			
		||||
						if (pkIsNull == false) dr.Result[col.CsName] = Utils.GetDataReaderValue(col.CsType, dr.Reader.GetValue(dr.Index));
 | 
			
		||||
						dr.Index++;
 | 
			
		||||
					}
 | 
			
		||||
				});
 | 
			
		||||
				return tableAlias;
 | 
			
		||||
			}
 | 
			
		||||
			T FetchResult(DbDataReader reader)
 | 
			
		||||
			{
 | 
			
		||||
				var fieldIndex = 0;
 | 
			
		||||
				var result = new T();
 | 
			
		||||
				for (var aliasIndex = 0; aliasIndex < _tableAlias.Count; aliasIndex++)
 | 
			
		||||
				{
 | 
			
		||||
					var navValue = result;
 | 
			
		||||
					var drctx = new DbDataReaderContext { Index = fieldIndex, Reader = reader };
 | 
			
		||||
					if (aliasIndex == 0)
 | 
			
		||||
					{
 | 
			
		||||
						drctx.Result = result;
 | 
			
		||||
					}
 | 
			
		||||
					else
 | 
			
		||||
					{
 | 
			
		||||
						var isNull = false;
 | 
			
		||||
						for (var navidx = 0; navidx < _tableAlias[aliasIndex].NavPath.Length - 1; navidx++)
 | 
			
		||||
						{
 | 
			
		||||
							var navKey = _tableAlias[aliasIndex].NavPath[navidx];
 | 
			
		||||
							if (navValue.ContainsKey(navKey) == false)
 | 
			
		||||
							{
 | 
			
		||||
								isNull = true;
 | 
			
		||||
								break;
 | 
			
		||||
							}
 | 
			
		||||
							navValue = navValue[navKey] as T;
 | 
			
		||||
							if (navValue == null)
 | 
			
		||||
							{
 | 
			
		||||
								isNull = true;
 | 
			
		||||
								break;
 | 
			
		||||
							}
 | 
			
		||||
						}
 | 
			
		||||
						if (isNull)
 | 
			
		||||
						{
 | 
			
		||||
							fieldIndex += _tableAlias[aliasIndex].Table.Columns.Count;
 | 
			
		||||
							continue;
 | 
			
		||||
						}
 | 
			
		||||
						drctx.Result = new T();
 | 
			
		||||
						navValue[_tableAlias[aliasIndex].NavPath.LastOrDefault()] = drctx.Result;
 | 
			
		||||
					}
 | 
			
		||||
					_fieldReader[aliasIndex](drctx);
 | 
			
		||||
					fieldIndex = drctx.Index;
 | 
			
		||||
					if (aliasIndex > 0 && drctx.Result.Any() == false) navValue.Remove(_tableAlias[aliasIndex].NavPath.LastOrDefault());
 | 
			
		||||
				}
 | 
			
		||||
				return result;
 | 
			
		||||
			}
 | 
			
		||||
			
 | 
			
		||||
			void IncludeMany(TableAliasInfo alias, NativeTuple<string, Action<SelectImpl>> navMany, List<T> list, List<string> flagIndexs = null)
 | 
			
		||||
			{
 | 
			
		||||
				if (list?.Any() != true) return;
 | 
			
		||||
				if (flagIndexs == null) flagIndexs = new List<string>();
 | 
			
		||||
				flagIndexs.Add(alias.Table.CsName);
 | 
			
		||||
 | 
			
		||||
				var nav = alias.Table.Navigates[navMany.Item1];
 | 
			
		||||
				if (_includeAll && flagIndexs.Contains(nav.RefTable.CsName)) return;
 | 
			
		||||
 | 
			
		||||
				if (nav.RefType == TableRefType.OneToMany)
 | 
			
		||||
				{
 | 
			
		||||
					var subTable = nav.RefTable;
 | 
			
		||||
					var subSelect = new SelectImpl(_dbcontext, subTable.CsName);
 | 
			
		||||
					if (_includeAll) subSelect.IncludeAll();
 | 
			
		||||
 | 
			
		||||
					Func<Dictionary<string, bool>> getWhereDic = () =>
 | 
			
		||||
					{
 | 
			
		||||
						var sbDic = new Dictionary<string, bool>();
 | 
			
		||||
						for (var y = 0; y < list.Count; y++)
 | 
			
		||||
						{
 | 
			
		||||
							var sbWhereOne = new StringBuilder();
 | 
			
		||||
							sbWhereOne.Append("(");
 | 
			
		||||
							for (var z = 0; z < nav.Columns.Count; z++)
 | 
			
		||||
							{
 | 
			
		||||
								if (z > 0) sbWhereOne.Append(" AND ");
 | 
			
		||||
								var refcol = nav.RefTable.ColumnsByCs[nav.RefColumns[z]];
 | 
			
		||||
								var val = Utils.GetDataReaderValue(refcol.Attribute.MapType, list[y][nav.Columns[z]]);
 | 
			
		||||
								sbWhereOne.Append(_common.FormatSql($"a.{_common.QuoteSqlName(refcol.Attribute.Name)}={{0}}", val));
 | 
			
		||||
							}
 | 
			
		||||
							sbWhereOne.Append(")");
 | 
			
		||||
							var whereOne = sbWhereOne.ToString();
 | 
			
		||||
							sbWhereOne.Clear();
 | 
			
		||||
							if (sbDic.ContainsKey(whereOne) == false) sbDic.Add(whereOne, true);
 | 
			
		||||
						}
 | 
			
		||||
						return sbDic;
 | 
			
		||||
					};
 | 
			
		||||
					if (nav.Columns.Count == 1)
 | 
			
		||||
					{
 | 
			
		||||
						var refcol = nav.RefTable.ColumnsByCs[nav.RefColumns[0]];
 | 
			
		||||
						var args1 = $"a.{_common.QuoteSqlName(refcol.Attribute.Name)}";
 | 
			
		||||
						var left = _common.FormatSql("{0}", new object[] { list.Select(a => a[nav.Columns[0]]).Distinct().ToArray() });
 | 
			
		||||
						subSelect._select.Where($"({args1} in {left.Replace(",   \r\n    \r\n", $") \r\n OR {args1} in (")})");
 | 
			
		||||
					}
 | 
			
		||||
					else
 | 
			
		||||
					{
 | 
			
		||||
						var sbDic = getWhereDic();
 | 
			
		||||
						var sbWhere = new StringBuilder();
 | 
			
		||||
						foreach (var sbd in sbDic)
 | 
			
		||||
							sbWhere.Append(" OR ").Append(sbd.Key);
 | 
			
		||||
						subSelect._select.Where(sbWhere.Remove(0, 4).ToString());
 | 
			
		||||
						sbWhere.Clear();
 | 
			
		||||
						sbDic.Clear();
 | 
			
		||||
					}
 | 
			
		||||
					navMany.Item2?.Invoke(subSelect);
 | 
			
		||||
					var subList = subSelect.ToListPrivate(null, null, flagIndexs);
 | 
			
		||||
					foreach (var item in list)
 | 
			
		||||
					{
 | 
			
		||||
						item[nav.NavigateKey] = subList.Where(a =>
 | 
			
		||||
						{
 | 
			
		||||
							for (var z = 0; z < nav.Columns.Count; z++)
 | 
			
		||||
								if (CompareEntityPropertyValue(nav.Table.ColumnsByCs[nav.Columns[z]].Attribute.MapType, item[nav.Columns[z]], a[nav.RefColumns[z]]) == false)
 | 
			
		||||
									return false;
 | 
			
		||||
							return true;
 | 
			
		||||
						}).ToList();
 | 
			
		||||
					}
 | 
			
		||||
					subList.Clear();
 | 
			
		||||
				}
 | 
			
		||||
				else if (nav.RefType == TableRefType.ManyToMany)
 | 
			
		||||
				{
 | 
			
		||||
					var subTable = nav.RefTable;
 | 
			
		||||
					var subSelect = new SelectImpl(_dbcontext, subTable.CsName);
 | 
			
		||||
					if (_includeAll) subSelect.IncludeAll();
 | 
			
		||||
 | 
			
		||||
					var middleJoinOn = string.Join(" AND ", nav.RefColumns.Select((bname, idx) =>
 | 
			
		||||
							$"midtb.{_common.QuoteSqlName(nav.RefMiddleTable.ColumnsByCs[nav.MiddleColumns[nav.Columns.Count + idx]].Attribute.Name)} = a.{_common.QuoteSqlName(nav.RefTable.ColumnsByCs[bname].Attribute.Name)}"));
 | 
			
		||||
					subSelect._select.InnerJoin($"{_common.QuoteSqlName(nav.RefMiddleTable.DbName)} midtb ON {middleJoinOn}");
 | 
			
		||||
 | 
			
		||||
					Func<Dictionary<string, bool>> getWhereDic = () =>
 | 
			
		||||
					{
 | 
			
		||||
						var sbDic = new Dictionary<string, bool>();
 | 
			
		||||
						for (var y = 0; y < list.Count; y++)
 | 
			
		||||
						{
 | 
			
		||||
							var sbWhereOne = new StringBuilder();
 | 
			
		||||
							sbWhereOne.Append("(");
 | 
			
		||||
							for (var z = 0; z < nav.Columns.Count; z++)
 | 
			
		||||
							{
 | 
			
		||||
								if (z > 0) sbWhereOne.Append(" AND ");
 | 
			
		||||
								var midcol = nav.RefMiddleTable.ColumnsByCs[nav.MiddleColumns[z]];
 | 
			
		||||
								var val = Utils.GetDataReaderValue(midcol.Attribute.MapType, list[y][nav.Columns[z]]);
 | 
			
		||||
								sbWhereOne.Append(_common.FormatSql($"midtb.{_common.QuoteSqlName(midcol.Attribute.Name)}={{0}}", val));
 | 
			
		||||
							}
 | 
			
		||||
							sbWhereOne.Append(")");
 | 
			
		||||
							var whereOne = sbWhereOne.ToString();
 | 
			
		||||
							sbWhereOne.Clear();
 | 
			
		||||
							if (sbDic.ContainsKey(whereOne) == false) sbDic.Add(whereOne, true);
 | 
			
		||||
						}
 | 
			
		||||
						return sbDic;
 | 
			
		||||
					};
 | 
			
		||||
					if (nav.Columns.Count == 1)
 | 
			
		||||
					{
 | 
			
		||||
						var midcol = nav.RefMiddleTable.ColumnsByCs[nav.MiddleColumns[0]];
 | 
			
		||||
						var args1 = $"midtb.{_common.QuoteSqlName(midcol.Attribute.Name)}";
 | 
			
		||||
						var left = _common.FormatSql("{0}", new object[] { list.Select(a => a[nav.Columns[0]]).Distinct().ToArray() });
 | 
			
		||||
						subSelect._select.Where($"({args1} in {left.Replace(",   \r\n    \r\n", $") \r\n OR {args1} in (")})");
 | 
			
		||||
					}
 | 
			
		||||
					else
 | 
			
		||||
					{
 | 
			
		||||
						var sbDic = getWhereDic();
 | 
			
		||||
						var sbWhere = new StringBuilder();
 | 
			
		||||
						foreach (var sbd in sbDic)
 | 
			
		||||
							sbWhere.Append(" OR ").Append(sbd.Key);
 | 
			
		||||
						subSelect._select.Where(sbWhere.Remove(0, 4).ToString());
 | 
			
		||||
						sbWhere.Clear();
 | 
			
		||||
						sbDic.Clear();
 | 
			
		||||
					}
 | 
			
		||||
					navMany.Item2?.Invoke(subSelect);
 | 
			
		||||
					var subList = subSelect.ToListPrivate(
 | 
			
		||||
						string.Join(", ", nav.MiddleColumns.Select((a, idx) => $"midtb.{_common.QuoteSqlName(nav.RefMiddleTable.ColumnsByCs[a].Attribute.Name)}{_common.FieldAsAlias($"midtb_field__{idx}")}")),
 | 
			
		||||
						(dict, dr) =>
 | 
			
		||||
						{
 | 
			
		||||
							var fieldCount = dr.FieldCount - nav.MiddleColumns.Count;
 | 
			
		||||
							for (var z = 0; z < nav.MiddleColumns.Count; z++)
 | 
			
		||||
								dict[$"midtb_field__{z}"] = Utils.GetDataReaderValue(nav.RefMiddleTable.ColumnsByCs[nav.MiddleColumns[z]].CsType, dr.GetValue(fieldCount + z));
 | 
			
		||||
						}, flagIndexs);
 | 
			
		||||
					foreach (var item in list)
 | 
			
		||||
					{
 | 
			
		||||
						item[nav.NavigateKey] = subList.Where(a =>
 | 
			
		||||
						{
 | 
			
		||||
							for (var z = 0; z < nav.Columns.Count; z++)
 | 
			
		||||
								if (CompareEntityPropertyValue(nav.Table.ColumnsByCs[nav.Columns[z]].Attribute.MapType, item[nav.Columns[z]], a[$"midtb_field__{z}"]) == false)
 | 
			
		||||
									return false;
 | 
			
		||||
							return true;
 | 
			
		||||
						}).ToList();
 | 
			
		||||
					}
 | 
			
		||||
					foreach (var subItem in subList)
 | 
			
		||||
						for (var z = 0; z < nav.MiddleColumns.Count; z++)
 | 
			
		||||
							subItem.Remove($"midtb_field__{z}");
 | 
			
		||||
					subList.Clear();
 | 
			
		||||
				}
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			public string ToSql(string field = null)
 | 
			
		||||
			{
 | 
			
		||||
				if (string.IsNullOrWhiteSpace(field)) return _select.ToSql(_field);
 | 
			
		||||
				return _select.ToSql(field);
 | 
			
		||||
			}
 | 
			
		||||
			List<T> ToListPrivate(string otherField, Action<T, DbDataReader> otherReader, List<string> flagIndexs = null)
 | 
			
		||||
			{
 | 
			
		||||
				var sql = string.IsNullOrWhiteSpace(otherField) ? this.ToSql() : this.ToSql($"{_field},{otherField}");
 | 
			
		||||
				var ret = new List<T>();
 | 
			
		||||
				var dbParms = _params.ToArray();
 | 
			
		||||
				var before = new Aop.CurdBeforeEventArgs(_tables[_mainTableIndex].Type, _tables[_mainTableIndex], Aop.CurdType.Select, sql, dbParms);
 | 
			
		||||
				_orm.Aop.CurdBeforeHandler?.Invoke(this, before);
 | 
			
		||||
				Exception exception = null;
 | 
			
		||||
				try
 | 
			
		||||
				{
 | 
			
		||||
					_orm.Ado.ExecuteReader(_dbcontext._transaction?.Connection, _dbcontext._transaction, fetch =>
 | 
			
		||||
					{
 | 
			
		||||
						var item = FetchResult(fetch.Object);
 | 
			
		||||
						otherReader?.Invoke(item, fetch.Object);
 | 
			
		||||
						ret.Add(item);
 | 
			
		||||
					}, CommandType.Text, sql, _dbcontext._commandTimeout, dbParms);
 | 
			
		||||
				}
 | 
			
		||||
				catch (Exception ex)
 | 
			
		||||
				{
 | 
			
		||||
					exception = ex;
 | 
			
		||||
					throw;
 | 
			
		||||
				}
 | 
			
		||||
				finally
 | 
			
		||||
				{
 | 
			
		||||
					var after = new Aop.CurdAfterEventArgs(before, exception, ret);
 | 
			
		||||
					_orm.Aop.CurdAfterHandler?.Invoke(this, after);
 | 
			
		||||
				}
 | 
			
		||||
				foreach (var join in _tableAlias)
 | 
			
		||||
				{
 | 
			
		||||
					if (join.IncludeMany.Any() == false) continue;
 | 
			
		||||
					var list = new List<T>();
 | 
			
		||||
					if (join.Alias == "a") list = ret;
 | 
			
		||||
					else
 | 
			
		||||
					{
 | 
			
		||||
						foreach (var obj in ret)
 | 
			
		||||
						{
 | 
			
		||||
							T item = obj;
 | 
			
		||||
							foreach (var navKey in join.NavPath)
 | 
			
		||||
							{
 | 
			
		||||
								if (string.IsNullOrWhiteSpace(navKey)) continue;
 | 
			
		||||
								item.TryGetValue(navKey, out var obj2);
 | 
			
		||||
								item = obj2 as T;
 | 
			
		||||
								if (item == null) break;
 | 
			
		||||
							}
 | 
			
		||||
							if (item != null) list.Add(item);
 | 
			
		||||
						}
 | 
			
		||||
					}
 | 
			
		||||
					foreach(var navMany in join.IncludeMany)
 | 
			
		||||
						IncludeMany(join, navMany, list, flagIndexs);
 | 
			
		||||
				}
 | 
			
		||||
				if (_useStates && flagIndexs == null)
 | 
			
		||||
					foreach (var item in ret)
 | 
			
		||||
						_dbcontext.AttachCascade(_tables[_mainTableIndex], item, true);
 | 
			
		||||
				return ret;
 | 
			
		||||
			}
 | 
			
		||||
			public List<T> ToList() => ToListPrivate(null, null);
 | 
			
		||||
			public T ToOne() => ToListPrivate(null, null).FirstOrDefault();
 | 
			
		||||
			public T First() => ToOne();
 | 
			
		||||
			public bool Any() => _select.Any();
 | 
			
		||||
			public long Count() => _select.Count();
 | 
			
		||||
			public SelectImpl Count(out long count)
 | 
			
		||||
			{
 | 
			
		||||
				_select.Count(out count);
 | 
			
		||||
				return this;
 | 
			
		||||
			}
 | 
			
		||||
			public SelectImpl WithTransaction(DbTransaction transaction)
 | 
			
		||||
			{
 | 
			
		||||
				_select.WithTransaction(transaction);
 | 
			
		||||
				return this;
 | 
			
		||||
			}
 | 
			
		||||
			public SelectImpl WithConnection(DbConnection connection)
 | 
			
		||||
			{
 | 
			
		||||
				_select.WithConnection(connection);
 | 
			
		||||
				return this;
 | 
			
		||||
			}
 | 
			
		||||
			public SelectImpl CommandTimeout(int timeout)
 | 
			
		||||
			{
 | 
			
		||||
				_select.CommandTimeout(timeout);
 | 
			
		||||
				return this;
 | 
			
		||||
			}
 | 
			
		||||
			public SelectImpl Distinct()
 | 
			
		||||
			{
 | 
			
		||||
				_select.Distinct();
 | 
			
		||||
				return this;
 | 
			
		||||
			}
 | 
			
		||||
			public SelectImpl Master()
 | 
			
		||||
			{
 | 
			
		||||
				_select.Master();
 | 
			
		||||
				return this;
 | 
			
		||||
			}
 | 
			
		||||
			public SelectImpl ForUpdate(bool nowait = false)
 | 
			
		||||
			{
 | 
			
		||||
				_select.ForUpdate(nowait);
 | 
			
		||||
				return this;
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			NativeTuple<string, ColumnInfo> ParseField(ZeroTableInfo firstTable, string firstTableAlias, string property)
 | 
			
		||||
			{
 | 
			
		||||
				if (string.IsNullOrEmpty(property)) return null;
 | 
			
		||||
				var field = property.Split('.').Select(a => a.Trim()).ToArray();
 | 
			
		||||
 | 
			
		||||
				if (field.Length == 1)
 | 
			
		||||
				{
 | 
			
		||||
					if (firstTable != null && firstTable.ColumnsByCs.TryGetValue(field[0], out var col2) == true)
 | 
			
		||||
						return NativeTuple.Create($"{firstTableAlias}.{_common.QuoteSqlName(col2.Attribute.Name)}", col2);
 | 
			
		||||
 | 
			
		||||
					foreach (var ta2 in _tableAlias)
 | 
			
		||||
					{
 | 
			
		||||
						if (ta2.Table.ColumnsByCs.TryGetValue(field[0], out col2))
 | 
			
		||||
							return NativeTuple.Create($"{ta2.Alias}.{_common.QuoteSqlName(col2.Attribute.Name)}", col2);
 | 
			
		||||
					}
 | 
			
		||||
				}
 | 
			
		||||
				else if (field.Length == 2)
 | 
			
		||||
				{
 | 
			
		||||
					if (firstTable != null && firstTable.CsName.ToLower() == field[0].ToLower() && firstTable.ColumnsByCs.TryGetValue(field[1], out var col2) == true)
 | 
			
		||||
						return NativeTuple.Create($"{firstTableAlias}.{_common.QuoteSqlName(col2.Attribute.Name)}", col2);
 | 
			
		||||
 | 
			
		||||
					var ta2s = _tableAlias.Where(a => a.Table.CsName.ToLower() == field[0].ToLower()).ToArray();
 | 
			
		||||
					if (ta2s.Length == 1 && ta2s[0].Table.ColumnsByCs.TryGetValue(field[1], out col2) == true)
 | 
			
		||||
						return NativeTuple.Create($"{ta2s[0].Alias}.{_common.QuoteSqlName(col2.Attribute.Name)}", col2);
 | 
			
		||||
					if (ta2s.Length > 1)
 | 
			
		||||
					{
 | 
			
		||||
						ta2s = _tableAlias.Where(a => a.Table.CsName == field[0]).ToArray();
 | 
			
		||||
						if (ta2s.Length == 1 && ta2s[0].Table.ColumnsByCs.TryGetValue(field[1], out col2) == true)
 | 
			
		||||
							return NativeTuple.Create($"{ta2s[0].Alias}.{_common.QuoteSqlName(col2.Attribute.Name)}", col2);
 | 
			
		||||
					}
 | 
			
		||||
					if (_tableAlias.Where(a => a.Alias == field[0]).FirstOrDefault()?.Table.ColumnsByCs.TryGetValue(field[1], out col2) == true)
 | 
			
		||||
						return NativeTuple.Create($"{field[0]}.{_common.QuoteSqlName(col2.Attribute.Name)}", col2);
 | 
			
		||||
				}
 | 
			
		||||
 | 
			
		||||
				var navPath = string.Join(".", field.Skip(1).Take(field.Length - 1));
 | 
			
		||||
				var ta = _tableAlias.Where(a => string.Join(".", a.NavPath) == navPath).FirstOrDefault();
 | 
			
		||||
				if (ta?.Table.ColumnsByCs.TryGetValue(field.Last(), out var col) == true)
 | 
			
		||||
					return NativeTuple.Create($"{ta.Alias}.{_common.QuoteSqlName(col.Attribute.Name)}", col);
 | 
			
		||||
				throw new Exception(CoreStrings.Cannot_Match_Property(property));
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			/// <summary>
 | 
			
		||||
			/// WHERE [Id] IN (...)
 | 
			
		||||
			/// </summary>
 | 
			
		||||
			public SelectImpl Where(IEnumerable<T> items)
 | 
			
		||||
			{
 | 
			
		||||
				var alias = _tableAlias.Where(a => a.Table == _tables[_mainTableIndex]).FirstOrDefault()?.Alias;
 | 
			
		||||
				if (!string.IsNullOrWhiteSpace(alias)) alias = $"{alias}.";
 | 
			
		||||
				var where = _common.WhereItems(_tables[_mainTableIndex].Primarys, alias, items);
 | 
			
		||||
				_select.Where(where);
 | 
			
		||||
				return this;
 | 
			
		||||
			}
 | 
			
		||||
			/// <summary>
 | 
			
		||||
			/// Where(new { Year = 2017, CategoryId = 198, IsPublished = true })<para></para>
 | 
			
		||||
			/// WHERE [Year] = 2017 AND [CategoryId] = 198 AND [IsPublished] = 1
 | 
			
		||||
			/// </summary>
 | 
			
		||||
			public SelectImpl Where(object multipleFields)
 | 
			
		||||
			{
 | 
			
		||||
				if (multipleFields == null) return this;
 | 
			
		||||
				foreach (var prop in multipleFields.GetType().GetProperties())
 | 
			
		||||
					WhereDynamicFilter(new DynamicFilterInfo { Field = prop.Name, Operator = DynamicFilterOperator.Eq, Value = prop.GetValue(multipleFields, null) });
 | 
			
		||||
				return this;
 | 
			
		||||
			}
 | 
			
		||||
			/// <summary>
 | 
			
		||||
			/// WHERE [field] = ..
 | 
			
		||||
			/// </summary>
 | 
			
		||||
			public SelectImpl Where(string field, object value) => WhereDynamicFilter(new DynamicFilterInfo { Field = field, Operator = DynamicFilterOperator.Eq, Value = value });
 | 
			
		||||
			public SelectImpl Where(string field, string @operator, object value)
 | 
			
		||||
			{
 | 
			
		||||
				switch (@operator?.ToLower().Trim())
 | 
			
		||||
				{
 | 
			
		||||
					case "=":
 | 
			
		||||
					case "==":
 | 
			
		||||
					case "eq":
 | 
			
		||||
						return WhereDynamicFilter(new DynamicFilterInfo { Field = field, Operator = DynamicFilterOperator.Eq, Value = value });
 | 
			
		||||
					case "!=":
 | 
			
		||||
					case "<>":
 | 
			
		||||
						return WhereDynamicFilter(new DynamicFilterInfo { Field = field, Operator = DynamicFilterOperator.NotEqual, Value = value });
 | 
			
		||||
					case ">":
 | 
			
		||||
						return WhereDynamicFilter(new DynamicFilterInfo { Field = field, Operator = DynamicFilterOperator.GreaterThan, Value = value });
 | 
			
		||||
					case ">=":
 | 
			
		||||
						return WhereDynamicFilter(new DynamicFilterInfo { Field = field, Operator = DynamicFilterOperator.GreaterThanOrEqual, Value = value });
 | 
			
		||||
					case "<":
 | 
			
		||||
						return WhereDynamicFilter(new DynamicFilterInfo { Field = field, Operator = DynamicFilterOperator.LessThan, Value = value });
 | 
			
		||||
					case "<=":
 | 
			
		||||
						return WhereDynamicFilter(new DynamicFilterInfo { Field = field, Operator = DynamicFilterOperator.LessThanOrEqual, Value = value });
 | 
			
		||||
					case "like":
 | 
			
		||||
					case "contains":
 | 
			
		||||
						return WhereDynamicFilter(new DynamicFilterInfo { Field = field, Operator = DynamicFilterOperator.Contains, Value = value });
 | 
			
		||||
					case "!like":
 | 
			
		||||
					case "notlike":
 | 
			
		||||
					case "not like":
 | 
			
		||||
						return WhereDynamicFilter(new DynamicFilterInfo { Field = field, Operator = DynamicFilterOperator.NotContains, Value = value });
 | 
			
		||||
					case "in":
 | 
			
		||||
						return WhereDynamicFilter(new DynamicFilterInfo { Field = field, Operator = DynamicFilterOperator.Any, Value = value });
 | 
			
		||||
					case "!in":
 | 
			
		||||
					case "notin":
 | 
			
		||||
					case "not in":
 | 
			
		||||
						return WhereDynamicFilter(new DynamicFilterInfo { Field = field, Operator = DynamicFilterOperator.Any, Value = value });
 | 
			
		||||
				}
 | 
			
		||||
				throw new Exception($"未实现 {@operator}");
 | 
			
		||||
			}
 | 
			
		||||
			public SelectImpl WhereColumns(string field1, string @operator, string field2)
 | 
			
		||||
			{
 | 
			
		||||
				var field1Result = ParseField(null, null, field1);
 | 
			
		||||
				if (field1Result == null) throw new Exception($"未匹配字段名 {field1}");
 | 
			
		||||
				var field2Result = ParseField(null, null, field2);
 | 
			
		||||
				if (field2Result == null) throw new Exception($"未匹配字段名 {field2}");
 | 
			
		||||
				switch (@operator?.ToLower().Trim())
 | 
			
		||||
				{
 | 
			
		||||
					case "=":
 | 
			
		||||
					case "==":
 | 
			
		||||
					case "eq":
 | 
			
		||||
						_select.Where($"{field1Result.Item1} = {field2Result.Item1}");
 | 
			
		||||
						return this;
 | 
			
		||||
					case "!=":
 | 
			
		||||
					case "<>":
 | 
			
		||||
						_select.Where($"{field1Result.Item1} <> {field2Result.Item1}");
 | 
			
		||||
						return this;
 | 
			
		||||
					case ">":
 | 
			
		||||
						_select.Where($"{field1Result.Item1} > {field2Result.Item1}");
 | 
			
		||||
						return this;
 | 
			
		||||
					case ">=":
 | 
			
		||||
						_select.Where($"{field1Result.Item1} >= {field2Result.Item1}");
 | 
			
		||||
						return this;
 | 
			
		||||
					case "<":
 | 
			
		||||
						_select.Where($"{field1Result.Item1} < {field2Result.Item1}");
 | 
			
		||||
						return this;
 | 
			
		||||
					case "<=":
 | 
			
		||||
						_select.Where($"{field1Result.Item1} <= {field2Result.Item1}");
 | 
			
		||||
						return this;
 | 
			
		||||
				}
 | 
			
		||||
				throw new Exception($"未实现 {@operator}");
 | 
			
		||||
			}
 | 
			
		||||
			public SelectImpl WhereDynamicFilter(DynamicFilterInfo filter)
 | 
			
		||||
			{
 | 
			
		||||
				var sql = ParseDynamicFilter(filter);
 | 
			
		||||
				_selectProvider._where.Append(sql);
 | 
			
		||||
				return this;
 | 
			
		||||
			}
 | 
			
		||||
			string ParseDynamicFilter(DynamicFilterInfo filter)
 | 
			
		||||
			{
 | 
			
		||||
				var replacedFilter = new DynamicFilterInfo();
 | 
			
		||||
				var replacedMap = new List<NativeTuple<string, string>>();
 | 
			
		||||
				LocalCloneFilter(filter, replacedFilter);
 | 
			
		||||
				var oldWhere = _selectProvider._where.ToString();
 | 
			
		||||
				var newWhere = "";
 | 
			
		||||
				try
 | 
			
		||||
				{
 | 
			
		||||
					_selectProvider._where.Clear();
 | 
			
		||||
					_select.WhereDynamicFilter(replacedFilter);
 | 
			
		||||
					newWhere = _selectProvider._where.ToString();
 | 
			
		||||
				}
 | 
			
		||||
				finally
 | 
			
		||||
				{
 | 
			
		||||
					_selectProvider._where.Clear().Append(oldWhere);
 | 
			
		||||
				}
 | 
			
		||||
				foreach (var rm in replacedMap)
 | 
			
		||||
				{
 | 
			
		||||
					var find = $"a.{_common.QuoteSqlName(rm.Item1)}";
 | 
			
		||||
					var idx = newWhere.IndexOf(find);
 | 
			
		||||
					if (idx != -1) newWhere = $"{newWhere.Substring(0, idx)}{rm.Item2}{newWhere.Substring(idx + find.Length)}";
 | 
			
		||||
				}
 | 
			
		||||
				return newWhere;
 | 
			
		||||
 | 
			
		||||
				void LocalCloneFilter(DynamicFilterInfo source, DynamicFilterInfo target)
 | 
			
		||||
				{
 | 
			
		||||
					target.Field = source.Field;
 | 
			
		||||
					target.Operator = source.Operator;
 | 
			
		||||
					target.Value = source.Value;
 | 
			
		||||
					target.Logic = source.Logic;
 | 
			
		||||
					if (string.IsNullOrWhiteSpace(source.Field) == false)
 | 
			
		||||
					{
 | 
			
		||||
						var parseResult = ParseField(null, null, source.Field);
 | 
			
		||||
						if (parseResult != null)
 | 
			
		||||
						{
 | 
			
		||||
							if (TestDynamicFilterInfo._dictTypeToPropertyname.TryGetValue(parseResult.Item2.Attribute.MapType, out var pname))
 | 
			
		||||
								target.Field = pname;
 | 
			
		||||
							else
 | 
			
		||||
								target.Field = TestDynamicFilterInfo._dictTypeToPropertyname[typeof(string)];
 | 
			
		||||
							replacedMap.Add(NativeTuple.Create(target.Field, parseResult.Item1));
 | 
			
		||||
						}
 | 
			
		||||
					}
 | 
			
		||||
					if (source.Filters?.Any() == true)
 | 
			
		||||
					{
 | 
			
		||||
						target.Filters = new List<DynamicFilterInfo>();
 | 
			
		||||
						foreach (var sourceChild in source.Filters)
 | 
			
		||||
						{
 | 
			
		||||
							var targetChild = new DynamicFilterInfo();
 | 
			
		||||
							target.Filters.Add(targetChild);
 | 
			
		||||
							LocalCloneFilter(sourceChild, targetChild);
 | 
			
		||||
						}
 | 
			
		||||
					}
 | 
			
		||||
				}
 | 
			
		||||
			}
 | 
			
		||||
			public class SubQuery
 | 
			
		||||
			{
 | 
			
		||||
				internal SelectImpl _parentQuery;
 | 
			
		||||
				internal SubQuery() { }
 | 
			
		||||
				public SelectImpl From(string tableName)
 | 
			
		||||
				{
 | 
			
		||||
					var query = _parentQuery._dbcontext.SelectNoTracking(tableName);
 | 
			
		||||
					query._selectProvider._tables[0].Alias =
 | 
			
		||||
						query._tableAlias[0].Alias = $"sub_{_parentQuery._tableAlias[0].Alias}";
 | 
			
		||||
					query._tableAlias.AddRange(_parentQuery._tableAlias);
 | 
			
		||||
					return query;
 | 
			
		||||
				}
 | 
			
		||||
			}
 | 
			
		||||
			public SelectImpl WhereExists(Func<SubQuery, SelectImpl> q)
 | 
			
		||||
			{
 | 
			
		||||
				var query = q?.Invoke(new SubQuery { _parentQuery = this });
 | 
			
		||||
				switch (_orm.Ado.DataType)
 | 
			
		||||
				{
 | 
			
		||||
					case DataType.Oracle:
 | 
			
		||||
					case DataType.OdbcOracle:
 | 
			
		||||
					case DataType.CustomOracle:
 | 
			
		||||
					case DataType.Dameng:
 | 
			
		||||
					case DataType.OdbcDameng:
 | 
			
		||||
					case DataType.GBase:
 | 
			
		||||
						query.Limit(-1);
 | 
			
		||||
						break;
 | 
			
		||||
					default:
 | 
			
		||||
						query.Limit(1); //#462 ORACLE rownum <= 2 会影响索引变慢
 | 
			
		||||
						break;
 | 
			
		||||
				}
 | 
			
		||||
				_selectProvider._where.Append($" AND EXISTS({query.ToSql("1").Replace(" \r\n", " \r\n    ")})");
 | 
			
		||||
				return this;
 | 
			
		||||
			}
 | 
			
		||||
			public SelectImpl GroupByRaw(string sql)
 | 
			
		||||
			{
 | 
			
		||||
				if (string.IsNullOrWhiteSpace(sql)) return this;
 | 
			
		||||
				_useStates = false;
 | 
			
		||||
				_groupBy = $"{_groupBy}, {sql}";
 | 
			
		||||
				_useStates = false;
 | 
			
		||||
				_select.GroupBy(_groupBy.Substring(2));
 | 
			
		||||
				return this;
 | 
			
		||||
			}
 | 
			
		||||
			public SelectImpl GroupBy(string[] fields)
 | 
			
		||||
			{
 | 
			
		||||
				var count = 0;
 | 
			
		||||
				for (var a = 0; a < fields.Length; a++)
 | 
			
		||||
				{
 | 
			
		||||
					if (string.IsNullOrWhiteSpace(fields[a])) continue;
 | 
			
		||||
					var field1 = ParseField(null, null, fields[a]);
 | 
			
		||||
					if (field1 == null) throw new Exception($"未匹配字段名 {fields[a]}");
 | 
			
		||||
					_groupBy = $"{_groupBy}, {field1.Item1}";
 | 
			
		||||
					count++;
 | 
			
		||||
				}
 | 
			
		||||
				if (count > 0)
 | 
			
		||||
				{
 | 
			
		||||
					_useStates = false;
 | 
			
		||||
					_select.GroupBy(_groupBy.Substring(2));
 | 
			
		||||
				}
 | 
			
		||||
				return this;
 | 
			
		||||
			}
 | 
			
		||||
			public SelectImpl HavingRaw(string sql)
 | 
			
		||||
			{
 | 
			
		||||
				_select.Having(sql);
 | 
			
		||||
				return this;
 | 
			
		||||
			}
 | 
			
		||||
			public SelectImpl OrderByRaw(string sql)
 | 
			
		||||
			{
 | 
			
		||||
				_select.OrderBy(sql);
 | 
			
		||||
				return this;
 | 
			
		||||
			}
 | 
			
		||||
			SelectImpl OrderBy(bool isdesc, string[] fields)
 | 
			
		||||
			{
 | 
			
		||||
				for (var a = 0; a < fields.Length; a ++)
 | 
			
		||||
				{
 | 
			
		||||
					if (string.IsNullOrWhiteSpace(fields[a])) continue;
 | 
			
		||||
					var field1 = ParseField(null, null, fields[a]);
 | 
			
		||||
					if (field1 == null) throw new Exception($"未匹配字段名 {fields[a]}");
 | 
			
		||||
					if (isdesc) _select.OrderBy($"{field1.Item1} DESC");
 | 
			
		||||
					else _select.OrderBy(field1.Item1);
 | 
			
		||||
				}
 | 
			
		||||
				return this;
 | 
			
		||||
			}
 | 
			
		||||
			public SelectImpl OrderBy(params string[] fields) => OrderBy(false, fields);
 | 
			
		||||
			public SelectImpl OrderByDescending(params string[] fields) => OrderBy(true, fields);
 | 
			
		||||
			public SelectImpl Offset(int offset)
 | 
			
		||||
			{
 | 
			
		||||
				_select.Offset(offset);
 | 
			
		||||
				return this;
 | 
			
		||||
			}
 | 
			
		||||
			public SelectImpl Limit(int limit)
 | 
			
		||||
			{
 | 
			
		||||
				_select.Limit(limit);
 | 
			
		||||
				return this;
 | 
			
		||||
			}
 | 
			
		||||
			public SelectImpl Skip(int offset) => Offset(offset);
 | 
			
		||||
			public SelectImpl Take(int limit) => Limit(limit);
 | 
			
		||||
			public SelectImpl Page(int pageNumber, int pageSize)
 | 
			
		||||
			{
 | 
			
		||||
				_select.Page(pageNumber, pageSize);
 | 
			
		||||
				return this;
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			TResult InternalQuerySingle<TResult>(string field) => _orm.Ado.CommandFluent(this.ToSql(field))
 | 
			
		||||
				.WithConnection(_selectProvider._connection)
 | 
			
		||||
				.WithTransaction(_selectProvider._transaction).QuerySingle<TResult>();
 | 
			
		||||
			public decimal Sum(string field)
 | 
			
		||||
			{
 | 
			
		||||
				var field1 = ParseField(null, null, field);
 | 
			
		||||
				if (field1 == null) throw new Exception($"未匹配字段名 {field}");
 | 
			
		||||
				return InternalQuerySingle<decimal>($"sum({field1.Item1})");
 | 
			
		||||
			}
 | 
			
		||||
			public TMember Min<TMember>(string field)
 | 
			
		||||
			{
 | 
			
		||||
				var field1 = ParseField(null, null, field);
 | 
			
		||||
				if (field1 == null) throw new Exception($"未匹配字段名 {field}");
 | 
			
		||||
				return InternalQuerySingle<TMember>($"min({field1.Item1})");
 | 
			
		||||
			}
 | 
			
		||||
			public TMember Max<TMember>(string field)
 | 
			
		||||
			{
 | 
			
		||||
				var field1 = ParseField(null, null, field);
 | 
			
		||||
				if (field1 == null) throw new Exception($"未匹配字段名 {field}");
 | 
			
		||||
				return InternalQuerySingle<TMember>($"max({field1.Item1})");
 | 
			
		||||
			}
 | 
			
		||||
			public double Avg(string field)
 | 
			
		||||
			{
 | 
			
		||||
				var field1 = ParseField(null, null, field);
 | 
			
		||||
				if (field1 == null) throw new Exception($"未匹配字段名 {field}");
 | 
			
		||||
				return InternalQuerySingle<double>($"avg({field1.Item1})");
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
			[Table(DisableSyncStructure = true)]
 | 
			
		||||
			class TestDynamicFilterInfo
 | 
			
		||||
			{
 | 
			
		||||
				public Guid DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf00 { get; set; }
 | 
			
		||||
				public bool DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf01 { get; set; }
 | 
			
		||||
				public string DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf02 { get; set; }
 | 
			
		||||
				public char DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf03 { get; set; }
 | 
			
		||||
				public DateTime DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf04 { get; set; }
 | 
			
		||||
				public DateTimeOffset DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf05 { get; set; }
 | 
			
		||||
				public TimeSpan DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf06 { get; set; }
 | 
			
		||||
 | 
			
		||||
				public int DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf07 { get; set; }
 | 
			
		||||
				public long DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf08 { get; set; }
 | 
			
		||||
				public short DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf09 { get; set; }
 | 
			
		||||
				public sbyte DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf10 { get; set; }
 | 
			
		||||
 | 
			
		||||
				public uint DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf11 { get; set; }
 | 
			
		||||
				public ulong DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf12 { get; set; }
 | 
			
		||||
				public ushort DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf13 { get; set; }
 | 
			
		||||
				public byte DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf14 { get; set; }
 | 
			
		||||
				public byte[] DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf15 { get; set; }
 | 
			
		||||
 | 
			
		||||
				public double DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf16 { get; set; }
 | 
			
		||||
				public float DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf17 { get; set; }
 | 
			
		||||
				public decimal DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf18 { get; set; }
 | 
			
		||||
 | 
			
		||||
				internal static Dictionary<Type, string> _dictTypeToPropertyname = new Dictionary<Type, string>
 | 
			
		||||
				{
 | 
			
		||||
					[typeof(Guid)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf00),
 | 
			
		||||
					[typeof(bool)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf01),
 | 
			
		||||
					[typeof(string)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf02),
 | 
			
		||||
					[typeof(char)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf03),
 | 
			
		||||
					[typeof(DateTime)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf04),
 | 
			
		||||
					[typeof(DateTimeOffset)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf05),
 | 
			
		||||
					[typeof(TimeSpan)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf06),
 | 
			
		||||
 | 
			
		||||
					[typeof(int)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf07),
 | 
			
		||||
					[typeof(long)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf08),
 | 
			
		||||
					[typeof(short)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf09),
 | 
			
		||||
					[typeof(sbyte)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf10),
 | 
			
		||||
 | 
			
		||||
					[typeof(uint)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf11),
 | 
			
		||||
					[typeof(ulong)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf12),
 | 
			
		||||
					[typeof(ushort)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf13),
 | 
			
		||||
					[typeof(byte)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf14),
 | 
			
		||||
					[typeof(byte[])] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf15),
 | 
			
		||||
 | 
			
		||||
					[typeof(double)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf16),
 | 
			
		||||
					[typeof(float)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf17),
 | 
			
		||||
					[typeof(decimal)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf18),
 | 
			
		||||
				};
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										1537
									
								
								Extensions/FreeSql.Extensions.ZeroEntity/ZeroDbContext.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1537
									
								
								Extensions/FreeSql.Extensions.ZeroEntity/ZeroDbContext.cs
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										97
									
								
								Extensions/FreeSql.Extensions.ZeroEntity/ZeroDescriptor.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										97
									
								
								Extensions/FreeSql.Extensions.ZeroEntity/ZeroDescriptor.cs
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,97 @@
 | 
			
		||||
using FreeSql.DataAnnotations;
 | 
			
		||||
using FreeSql.Internal.Model;
 | 
			
		||||
using System;
 | 
			
		||||
using System.Collections.Generic;
 | 
			
		||||
 | 
			
		||||
namespace FreeSql.Extensions.ZeroEntity
 | 
			
		||||
{
 | 
			
		||||
	public class TableDescriptor
 | 
			
		||||
	{
 | 
			
		||||
		public string Name { get; set; }
 | 
			
		||||
		public string DbName { get; set; }
 | 
			
		||||
		public string AsTable { get; set; }
 | 
			
		||||
		public bool DisableSyncStructure { get; set; }
 | 
			
		||||
		public string Comment { get; set; }
 | 
			
		||||
		public List<ColumnDescriptor> Columns { get; } = new List<ColumnDescriptor>();
 | 
			
		||||
		public List<NavigateDescriptor> Navigates { get; } = new List<NavigateDescriptor>();
 | 
			
		||||
		public List<IndexDescriptor> Indexes { get; } = new List<IndexDescriptor>();
 | 
			
		||||
 | 
			
		||||
		public class ColumnDescriptor
 | 
			
		||||
		{
 | 
			
		||||
			public string Name { get; set; }
 | 
			
		||||
			public string DbType { get; set; }
 | 
			
		||||
			bool? _IsPrimary, _IsIdentity, _IsNullable, _IsVersion;
 | 
			
		||||
			public bool IsPrimary { get => _IsPrimary ?? false; set => _IsPrimary = value; }
 | 
			
		||||
			public bool IsIdentity { get => _IsIdentity ?? false; set => _IsIdentity = value; }
 | 
			
		||||
			public bool IsNullable { get => _IsNullable ?? false; set => _IsNullable = value; }
 | 
			
		||||
			public bool IsVersion { get => _IsVersion ?? false; set => _IsVersion = value; }
 | 
			
		||||
			public Type MapType { get; set; }
 | 
			
		||||
			public DateTimeKind ServerTime { get; set; }
 | 
			
		||||
			public string InsertValueSql { get; set; }
 | 
			
		||||
			int? _StringLength;
 | 
			
		||||
			public int StringLength { get => _StringLength ?? 0; set => _StringLength = value; }
 | 
			
		||||
			int? _Precision;
 | 
			
		||||
			public int Precision { get => _Precision ?? 0; set => _Precision = value; }
 | 
			
		||||
			int? _Scale;
 | 
			
		||||
			public int Scale { get => _Scale ?? 0; set => _Scale = value; }
 | 
			
		||||
			public string Comment { get; set; }
 | 
			
		||||
 | 
			
		||||
			public ColumnAttribute ToAttribute()
 | 
			
		||||
			{
 | 
			
		||||
				var attr = new ColumnAttribute
 | 
			
		||||
				{
 | 
			
		||||
					Name = Name,
 | 
			
		||||
					DbType = DbType,
 | 
			
		||||
					MapType = MapType,
 | 
			
		||||
					ServerTime = ServerTime,
 | 
			
		||||
					InsertValueSql = InsertValueSql,
 | 
			
		||||
				};
 | 
			
		||||
				if (_IsPrimary != null) attr.IsPrimary = IsPrimary;
 | 
			
		||||
				if (_IsIdentity != null) attr.IsIdentity = IsIdentity;
 | 
			
		||||
				if (_IsNullable != null) attr.IsNullable = IsNullable;
 | 
			
		||||
				if (_IsVersion != null) attr.IsVersion = IsVersion;
 | 
			
		||||
				if (_StringLength != null) attr.StringLength = StringLength;
 | 
			
		||||
				if (_Precision != null) attr.Precision = Precision;
 | 
			
		||||
				if (_Scale != null) attr.Scale = Scale;
 | 
			
		||||
				return attr;
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
		public class IndexDescriptor
 | 
			
		||||
		{
 | 
			
		||||
			public string Name { get; set; }
 | 
			
		||||
			public string Fields { get; set; }
 | 
			
		||||
			public bool IsUnique { get; set; }
 | 
			
		||||
			public IndexMethod IndexMethod { get; set; }
 | 
			
		||||
		}
 | 
			
		||||
		public class NavigateDescriptor
 | 
			
		||||
		{
 | 
			
		||||
			public string Name { get; set; }
 | 
			
		||||
			public NavigateType Type { get; set; }
 | 
			
		||||
			public string RelTable { get; set; }
 | 
			
		||||
			public string Bind { get; set; }
 | 
			
		||||
			public string ManyToMany { get; set; }
 | 
			
		||||
		}
 | 
			
		||||
		public enum NavigateType
 | 
			
		||||
		{
 | 
			
		||||
			OneToOne, ManyToOne, OneToMany, ManyToMany
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	class ZeroTableRef
 | 
			
		||||
	{
 | 
			
		||||
		internal string NavigateKey { get; set; }
 | 
			
		||||
		public TableRefType RefType { get; set; }
 | 
			
		||||
		internal ZeroTableInfo Table { get; set; }
 | 
			
		||||
		internal ZeroTableInfo RefTable { get; set; }
 | 
			
		||||
		internal ZeroTableInfo RefMiddleTable { get; set; }
 | 
			
		||||
 | 
			
		||||
		public List<string> Columns { get; set; } = new List<string>();
 | 
			
		||||
		public List<string> MiddleColumns { get; set; } = new List<string>();
 | 
			
		||||
		public List<string> RefColumns { get; set; } = new List<string>();
 | 
			
		||||
	}
 | 
			
		||||
	class ZeroTableInfo : TableInfo
 | 
			
		||||
	{
 | 
			
		||||
		public Dictionary<string, ZeroTableRef> Navigates { get; set; } = new Dictionary<string, ZeroTableRef>(StringComparer.OrdinalIgnoreCase);
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										
											BIN
										
									
								
								Extensions/FreeSql.Extensions.ZeroEntity/key.snk
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								Extensions/FreeSql.Extensions.ZeroEntity/key.snk
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
		Reference in New Issue
	
	Block a user