- 增加 低代码扩展包 FreeSql.Extensions.ZoreEntity;

This commit is contained in:
2881099 2023-12-13 20:59:25 +08:00
parent 2368377b0d
commit b814921a6a
8 changed files with 2727 additions and 100 deletions

View File

@ -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 扩展包,实现 低代码、零实体类型,并且支持导航属性,级联保存 等功能.</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.806-preview20231214</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.ZoreEntity.xml</DocumentationFile>
<WarningLevel>3</WarningLevel>
</PropertyGroup>
<PropertyGroup Condition="'$(TargetFramework)' == 'net40'">
<DefineConstants>net40</DefineConstants>
</PropertyGroup>
<ItemGroup>
<ProjectReference Include="..\..\FreeSql\FreeSql.csproj" />
</ItemGroup>
</Project>

View File

@ -0,0 +1,67 @@
<?xml version="1.0"?>
<doc>
<assembly>
<name>FreeSql.Extensions.ZoreEntity</name>
</assembly>
<members>
<member name="P:FreeSql.Extensions.ZoreEntity.ZoreDbContext.Select">
<summary>
【有状态管理】自动 Include 查询
</summary>
</member>
<member name="M:FreeSql.Extensions.ZoreEntity.ZoreDbContext.SelectNoTracking(System.String)">
<summary>
【无状态管理】指定表查询
</summary>
</member>
<member name="P:FreeSql.Extensions.ZoreEntity.ZoreDbContext.ChangeReport.ChangeInfo.BeforeObject">
<summary>
Type = Update 的时候,获取更新之前的对象
</summary>
</member>
<member name="P:FreeSql.Extensions.ZoreEntity.ZoreDbContext.ChangeReport.Report">
<summary>
实体变化记录
</summary>
</member>
<member name="P:FreeSql.Extensions.ZoreEntity.ZoreDbContext.ChangeReport.OnChange">
<summary>
实体变化事件
</summary>
</member>
<member name="M:FreeSql.Extensions.ZoreEntity.ZoreDbContext.SelectImpl.LeftJoin(System.String,System.String[])">
<summary>
举例1LeftJoin("table1", "id", "user.id") -> LEFT JOIN [table1] b ON b.[id] = a.[id]<para></para>
举例2LeftJoin("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.ZoreEntity.ZoreDbContext.SelectImpl.InnerJoin(System.String,System.String[])">
<summary>
举例1InnerJoin("table1", "id", "user.id") -> INNER JOIN [table1] b ON b.[id] = a.[id]<para></para>
举例2InnerJoin("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.ZoreEntity.ZoreDbContext.SelectImpl.RightJoin(System.String,System.String[])">
<summary>
举例1RightJoin("table1", "id", "user.id") -> RIGTH JOIN [table1] b ON b.[id] = a.[id]<para></para>
举例2RightJoin("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.ZoreEntity.ZoreDbContext.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.ZoreEntity.ZoreDbContext.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.ZoreEntity.ZoreDbContext.SelectImpl.Where(System.String,System.Object)">
<summary>
WHERE [field] = ..
</summary>
</member>
</members>
</doc>

View File

@ -0,0 +1,915 @@
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.ZoreEntity
{
partial class ZoreDbContext
{
public class SelectImpl
{
ZoreDbContext _dbcontext;
IFreeSql _orm => _dbcontext._orm;
List<ZoreTableInfo> _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(ZoreDbContext 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(ZoreTableInfo 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>
/// 举例1LeftJoin("table1", "id", "user.id") -> LEFT JOIN [table1] b ON b.[id] = a.[id]<para></para>
/// 举例2LeftJoin("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>
/// 举例1InnerJoin("table1", "id", "user.id") -> INNER JOIN [table1] b ON b.[id] = a.[id]<para></para>
/// 举例2InnerJoin("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>
/// 举例1RightJoin("table1", "id", "user.id") -> RIGTH JOIN [table1] b ON b.[id] = a.[id]<para></para>
/// 举例2RightJoin("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 ZoreTableInfo 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(ZoreTableInfo 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(ZoreTableInfo 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}";
_select.GroupBy(_groupBy);
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);
}
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),
};
}
}
}
}

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,97 @@
using FreeSql.DataAnnotations;
using FreeSql.Internal.Model;
using System;
using System.Collections.Generic;
namespace FreeSql.Extensions.ZoreEntity
{
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 ZoreTableRef
{
internal string NavigateKey { get; set; }
public TableRefType RefType { get; set; }
internal ZoreTableInfo Table { get; set; }
internal ZoreTableInfo RefTable { get; set; }
internal ZoreTableInfo 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 ZoreTableInfo : TableInfo
{
public Dictionary<string, ZoreTableRef> Navigates { get; set; } = new Dictionary<string, ZoreTableRef>(StringComparer.OrdinalIgnoreCase);
}
}

Binary file not shown.

View File

@ -89,6 +89,8 @@ Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "FreeSql.Provider.QuestDb",
EndProject EndProject
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "FreeSql.Provider.Xugu", "Providers\FreeSql.Provider.Xugu\FreeSql.Provider.Xugu.csproj", "{353F3732-0704-40F2-972B-036E9CC01881}" Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "FreeSql.Provider.Xugu", "Providers\FreeSql.Provider.Xugu\FreeSql.Provider.Xugu.csproj", "{353F3732-0704-40F2-972B-036E9CC01881}"
EndProject EndProject
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "FreeSql.Extensions.ZoreEntity", "Extensions\FreeSql.Extensions.ZoreEntity\FreeSql.Extensions.ZoreEntity.csproj", "{4367B7AC-604F-4503-A1D4-643ADBFCF703}"
EndProject
Global Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|Any CPU = Debug|Any CPU Debug|Any CPU = Debug|Any CPU
@ -495,6 +497,18 @@ Global
{353F3732-0704-40F2-972B-036E9CC01881}.Release|x64.Build.0 = Release|Any CPU {353F3732-0704-40F2-972B-036E9CC01881}.Release|x64.Build.0 = Release|Any CPU
{353F3732-0704-40F2-972B-036E9CC01881}.Release|x86.ActiveCfg = Release|Any CPU {353F3732-0704-40F2-972B-036E9CC01881}.Release|x86.ActiveCfg = Release|Any CPU
{353F3732-0704-40F2-972B-036E9CC01881}.Release|x86.Build.0 = Release|Any CPU {353F3732-0704-40F2-972B-036E9CC01881}.Release|x86.Build.0 = Release|Any CPU
{4367B7AC-604F-4503-A1D4-643ADBFCF703}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{4367B7AC-604F-4503-A1D4-643ADBFCF703}.Debug|Any CPU.Build.0 = Debug|Any CPU
{4367B7AC-604F-4503-A1D4-643ADBFCF703}.Debug|x64.ActiveCfg = Debug|Any CPU
{4367B7AC-604F-4503-A1D4-643ADBFCF703}.Debug|x64.Build.0 = Debug|Any CPU
{4367B7AC-604F-4503-A1D4-643ADBFCF703}.Debug|x86.ActiveCfg = Debug|Any CPU
{4367B7AC-604F-4503-A1D4-643ADBFCF703}.Debug|x86.Build.0 = Debug|Any CPU
{4367B7AC-604F-4503-A1D4-643ADBFCF703}.Release|Any CPU.ActiveCfg = Release|Any CPU
{4367B7AC-604F-4503-A1D4-643ADBFCF703}.Release|Any CPU.Build.0 = Release|Any CPU
{4367B7AC-604F-4503-A1D4-643ADBFCF703}.Release|x64.ActiveCfg = Release|Any CPU
{4367B7AC-604F-4503-A1D4-643ADBFCF703}.Release|x64.Build.0 = Release|Any CPU
{4367B7AC-604F-4503-A1D4-643ADBFCF703}.Release|x86.ActiveCfg = Release|Any CPU
{4367B7AC-604F-4503-A1D4-643ADBFCF703}.Release|x86.Build.0 = Release|Any CPU
EndGlobalSection EndGlobalSection
GlobalSection(SolutionProperties) = preSolution GlobalSection(SolutionProperties) = preSolution
HideSolutionNode = FALSE HideSolutionNode = FALSE
@ -529,10 +543,11 @@ Global
{9D7EA01A-110A-4A0C-A46B-9A0FBC88DD3D} = {94C8A78D-AA15-47B2-A348-530CD86BFC1B} {9D7EA01A-110A-4A0C-A46B-9A0FBC88DD3D} = {94C8A78D-AA15-47B2-A348-530CD86BFC1B}
{FEE501EB-60D1-4370-BC65-F939BCA7F32E} = {2A381C57-2697-427B-9F10-55DA11FD02E4} {FEE501EB-60D1-4370-BC65-F939BCA7F32E} = {2A381C57-2697-427B-9F10-55DA11FD02E4}
{353F3732-0704-40F2-972B-036E9CC01881} = {2A381C57-2697-427B-9F10-55DA11FD02E4} {353F3732-0704-40F2-972B-036E9CC01881} = {2A381C57-2697-427B-9F10-55DA11FD02E4}
{4367B7AC-604F-4503-A1D4-643ADBFCF703} = {4A92E8A6-9A6D-41A1-9CDA-DE10899648AA}
EndGlobalSection EndGlobalSection
GlobalSection(ExtensibilityGlobals) = postSolution GlobalSection(ExtensibilityGlobals) = postSolution
SolutionGuid = {089687FA-5D21-40AC-BA8A-AA0D1E1H7F98}
RESX_PrefixTranslations = True
RESX_NeutralResourcesLanguage = en-US RESX_NeutralResourcesLanguage = en-US
RESX_PrefixTranslations = True
SolutionGuid = {089687FA-5D21-40AC-BA8A-AA0D1E1H7F98}
EndGlobalSection EndGlobalSection
EndGlobal EndGlobal

View File

@ -1104,82 +1104,6 @@
</summary> </summary>
<returns></returns> <returns></returns>
</member> </member>
<member name="T:FreeSql.Extensions.DynamicEntity.DynamicCompileBuilder">
<summary>
动态创建实体类型
</summary>
</member>
<member name="M:FreeSql.Extensions.DynamicEntity.DynamicCompileBuilder.#ctor(IFreeSql,System.String,System.Attribute[])">
<summary>
配置Class
</summary>
<param name="className">类名</param>
<param name="attributes">类标记的特性[Table(Name = "xxx")] [Index(xxxx)]</param>
<returns></returns>
</member>
<member name="M:FreeSql.Extensions.DynamicEntity.DynamicCompileBuilder.Property(System.String,System.Type,System.Attribute[])">
<summary>
配置属性
</summary>
<param name="propertyName">属性名称</param>
<param name="propertyType">属性类型</param>
<param name="attributes">属性标记的特性-支持多个</param>
<returns></returns>
</member>
<member name="M:FreeSql.Extensions.DynamicEntity.DynamicCompileBuilder.Property(System.String,System.Type,System.Boolean,System.Attribute[])">
<summary>
配置属性
</summary>
<param name="propertyName">属性名称</param>
<param name="propertyType">属性类型</param>
<param name="isOverride">该属性是否重写父类属性</param>
<param name="attributes">属性标记的特性-支持多个</param>
<returns></returns>
</member>
<member name="M:FreeSql.Extensions.DynamicEntity.DynamicCompileBuilder.Property(System.String,System.Type,System.Boolean,System.Object,System.Attribute[])">
<summary>
配置属性
</summary>
<param name="propertyName">属性名称</param>
<param name="propertyType">属性类型</param>
<param name="isOverride">该属性是否重写父类属性</param>
<param name="defaultValue">属性默认值</param>
<param name="attributes">属性标记的特性-支持多个</param>
<returns></returns>
</member>
<member name="M:FreeSql.Extensions.DynamicEntity.DynamicCompileBuilder.Extend(System.Type)">
<summary>
配置父类
</summary>
<param name="superClass">父类类型</param>
<returns></returns>
</member>
<member name="M:FreeSql.Extensions.DynamicEntity.DynamicCompileBuilder.OverrideProperty(System.Reflection.Emit.TypeBuilder@,System.Reflection.Emit.MethodBuilder,FreeSql.Extensions.DynamicEntity.DynamicCompileBuilder.PropertyMethodEnum,System.String)">
<summary>
Override属性
</summary>
<param name="typeBuilder"></param>
</member>
<member name="M:FreeSql.Extensions.DynamicEntity.DynamicCompileBuilder.Build">
<summary>
Emit动态创建出Class - Type
</summary>
<returns></returns>
</member>
<member name="M:FreeSql.Extensions.DynamicEntity.DynamicCompileBuilder.FirstCharToLower(System.String)">
<summary>
首字母小写
</summary>
<param name="input"></param>
<returns></returns>
</member>
<member name="M:FreeSql.Extensions.DynamicEntity.DynamicCompileBuilder.FirstCharToUpper(System.String)">
<summary>
首字母大写
</summary>
<param name="input"></param>
<returns></returns>
</member>
<member name="M:FreeSql.Extensions.EntityUtil.EntityUtilExtensions.GetEntityKeyString(IFreeSql,System.Type,System.Object,System.Boolean,System.String)"> <member name="M:FreeSql.Extensions.EntityUtil.EntityUtilExtensions.GetEntityKeyString(IFreeSql,System.Type,System.Object,System.Boolean,System.String)">
<summary> <summary>
获取实体的主键值,以 "*|_,[,_|*" 分割,当任意一个主键属性无值时,返回 null 获取实体的主键值,以 "*|_,[,_|*" 分割,当任意一个主键属性无值时,返回 null
@ -5884,28 +5808,6 @@
请使用 fsql.InsertDict(dict) 方法插入字典数据 请使用 fsql.InsertDict(dict) 方法插入字典数据
</summary> </summary>
</member> </member>
<member name="M:FreeSqlGlobalDynamicEntityExtensions.DynamicEntity(FreeSql.ICodeFirst,System.String,System.Attribute[])">
<summary>
动态构建Class Type
</summary>
<returns></returns>
</member>
<member name="M:FreeSqlGlobalDynamicEntityExtensions.CreateInstance(FreeSql.Internal.Model.TableInfo,System.Collections.Generic.Dictionary{System.String,System.Object})">
<summary>
根据字典,创建 table 对应的实体对象
</summary>
<param name="table"></param>
<param name="dict"></param>
<returns></returns>
</member>
<member name="M:FreeSqlGlobalDynamicEntityExtensions.CreateDictionary(FreeSql.Internal.Model.TableInfo,System.Object)">
<summary>
根据实体对象,创建 table 对应的字典
</summary>
<param name="table"></param>
<param name="instance"></param>
<returns></returns>
</member>
<member name="M:FreeSqlGlobalExpressionCallExtensions.Between(System.DateTime,System.DateTime,System.DateTime)"> <member name="M:FreeSqlGlobalExpressionCallExtensions.Between(System.DateTime,System.DateTime,System.DateTime)">
<summary> <summary>
C# that >= between &amp;&amp; that &lt;= and<para></para> C# that >= between &amp;&amp; that &lt;= and<para></para>
@ -6422,3 +6324,115 @@
</member> </member>
</members> </members>
</doc> </doc>
`0})">
<summary>
插入数据,传入实体集合
</summary>
<typeparam name="T1"></typeparam>
<param name="source"></param>
<returns></returns>
</member>
<member name="M:IFreeSql.InsertOrUpdate``1">
<summary>
插入或更新数据,此功能依赖数据库特性(低版本可能不支持),参考如下:<para></para>
MySql 5.6+: on duplicate key update<para></para>
PostgreSQL 9.4+: on conflict do update<para></para>
SqlServer 2008+: merge into<para></para>
Oracle 11+: merge into<para></para>
Sqlite: replace into<para></para>
达梦: merge into<para></para>
人大金仓on conflict do update<para></para>
神通merge into<para></para>
MsAccess不支持<para></para>
注意区别FreeSql.Repository 仓储也有 InsertOrUpdate 方法(不依赖数据库特性)
</summary>
<typeparam name="T1"></typeparam>
<returns></returns>
</member>
<member name="M:IFreeSql.Update``1">
<summary>
修改数据
</summary>
<typeparam name="T1"></typeparam>
<returns></returns>
</member>
<member name="M:IFreeSql.Update``1(System.Object)">
<summary>
修改数据,传入动态条件,如:主键值 | new[]{主键值1,主键值2} | TEntity1 | new[]{TEntity1,TEntity2} | new{id=1}
</summary>
<typeparam name="T1"></typeparam>
<param name="dywhere">主键值、主键值集合、实体、实体集合、匿名对象、匿名对象集合</param>
<returns></returns>
</member>
<member name="M:IFreeSql.Select``1">
<summary>
查询数据
</summary>
<typeparam name="T1"></typeparam>
<returns></returns>
</member>
<member name="M:IFreeSql.Select``1(System.Object)">
<summary>
查询数据,传入动态条件,如:主键值 | new[]{主键值1,主键值2} | TEntity1 | new[]{TEntity1,TEntity2} | new{id=1}
</summary>
<typeparam name="T1"></typeparam>
<param name="dywhere">主键值、主键值集合、实体、实体集合、匿名对象、匿名对象集合</param>
<returns></returns>
</member>
<member name="M:IFreeSql.Delete``1">
<summary>
删除数据
</summary>
<typeparam name="T1"></typeparam>
<returns></returns>
</member>
<member name="M:IFreeSql.Delete``1(System.Object)">
<summary>
删除数据,传入动态条件,如:主键值 | new[]{主键值1,主键值2} | TEntity1 | new[]{TEntity1,TEntity2} | new{id=1}
</summary>
<typeparam name="T1"></typeparam>
<param name="dywhere">主键值、主键值集合、实体、实体集合、匿名对象、匿名对象集合</param>
<returns></returns>
</member>
<member name="M:IFreeSql.Transaction(System.Action)">
<summary>
开启事务(不支持异步)<para></para>
v1.5.0 关闭了线程事务超时自动提交的机制
</summary>
<param name="handler">事务体 () => {}</param>
</member>
<member name="M:IFreeSql.Transaction(System.Data.IsolationLevel,System.Action)">
<summary>
开启事务(不支持异步)<para></para>
v1.5.0 关闭了线程事务超时自动提交的机制
</summary>
<param name="isolationLevel"></param>
<param name="handler">事务体 () => {}</param>
</member>
<member name="P:IFreeSql.Ado">
<summary>
数据库访问对象
</summary>
</member>
<member name="P:IFreeSql.Aop">
<summary>
所有拦截方法都在这里
</summary>
</member>
<member name="P:IFreeSql.CodeFirst">
<summary>
CodeFirst 模式开发相关方法
</summary>
</member>
<member name="P:IFreeSql.DbFirst">
<summary>
DbFirst 模式开发相关方法
</summary>
</member>
<member name="P:IFreeSql.GlobalFilter">
<summary>
全局过滤设置,可默认附加为 Select/Update/Delete 条件
</summary>
</member>
</members>
</doc>