mirror of
https://github.com/nsnail/FreeSql.git
synced 2025-04-23 03:02:51 +08:00
2741 lines
184 KiB
C#
2741 lines
184 KiB
C#
using FreeSql.DataAnnotations;
|
||
using FreeSql.Internal.CommonProvider;
|
||
using FreeSql.Internal.Model;
|
||
using System;
|
||
using System.Collections;
|
||
using System.Collections.Concurrent;
|
||
using System.Collections.Generic;
|
||
using System.Collections.ObjectModel;
|
||
using System.Data.Common;
|
||
using System.Globalization;
|
||
using System.Linq;
|
||
using System.Linq.Expressions;
|
||
using System.Reflection;
|
||
using System.Text;
|
||
using System.Text.RegularExpressions;
|
||
using System.Threading;
|
||
|
||
namespace FreeSql.Internal
|
||
{
|
||
public abstract class BaseDiyMemberExpression
|
||
{
|
||
/// <summary>
|
||
/// 临时 LambdaExpression.Parameter
|
||
/// </summary>
|
||
public ParameterExpression _lambdaParameter;
|
||
public ReadAnonymousTypeInfo _map;
|
||
public string _field;
|
||
public ReadAnonymousTypeInfo ParseExpMapResult { get; protected set; }
|
||
public abstract string ParseExp(Expression[] members);
|
||
}
|
||
|
||
public abstract class CommonExpression
|
||
{
|
||
|
||
public CommonUtils _common;
|
||
public AdoProvider _ado => _adoPriv ?? (_adoPriv = _common._orm.Ado as AdoProvider);
|
||
AdoProvider _adoPriv;
|
||
public CommonExpression(CommonUtils common)
|
||
{
|
||
_common = common;
|
||
}
|
||
|
||
internal const int ReadAnonymousFieldAsCsName = -53129;
|
||
internal const int ReadAnonymousFieldAsCsNameGroupBy = -10000;
|
||
internal string GetFieldAsCsName(string csname)
|
||
{
|
||
csname = _common.QuoteSqlName(csname);
|
||
if (_common.CodeFirst.IsSyncStructureToLower) csname = csname.ToLower();
|
||
if (_common.CodeFirst.IsSyncStructureToUpper) csname = csname.ToUpper();
|
||
return csname;
|
||
}
|
||
internal bool EndsWithDbNestedField(string dbField, string dbNestedField)
|
||
{
|
||
if (string.IsNullOrWhiteSpace(dbNestedField)) return true;
|
||
switch (_ado.DataType)
|
||
{
|
||
case DataType.SqlServer:
|
||
case DataType.OdbcSqlServer:
|
||
return dbField.EndsWith(dbNestedField, StringComparison.CurrentCultureIgnoreCase);
|
||
}
|
||
return dbField.EndsWith(dbNestedField);
|
||
}
|
||
public bool ReadAnonymousField(List<SelectTableInfo> _tables, Func<Type, string, string> _tableRule, StringBuilder field, ReadAnonymousTypeInfo parent, ref int index, Expression exp, Select0Provider select,
|
||
BaseDiyMemberExpression diymemexp, List<GlobalFilter.Item> whereGlobalFilter, List<string> findIncludeMany, List<Expression> findSubSelectMany, bool isAllDtoMap)
|
||
{
|
||
void LocalSetFieldAlias(ref int localIndex, bool isdiymemexp)
|
||
{
|
||
if (localIndex >= 0)
|
||
{
|
||
parent.DbNestedField = $"as{++localIndex}";
|
||
field.Append(_common.FieldAsAlias(parent.DbNestedField));
|
||
}
|
||
else if (isdiymemexp && diymemexp?.ParseExpMapResult != null)
|
||
parent.DbNestedField = diymemexp.ParseExpMapResult.DbNestedField;
|
||
else if (string.IsNullOrEmpty(parent.CsName) == false)
|
||
{
|
||
parent.DbNestedField = GetFieldAsCsName(parent.CsName);
|
||
if (localIndex == ReadAnonymousFieldAsCsName && EndsWithDbNestedField(parent.DbField, parent.DbNestedField) == false) //DbField 和 CsName 相同的时候,不处理
|
||
field.Append(_common.FieldAsAlias(parent.DbNestedField));
|
||
}
|
||
}
|
||
|
||
Func<ExpTSC> getTSC = () => new ExpTSC { _tables = _tables, _tableRule = _tableRule, diymemexp = diymemexp, tbtype = SelectTableInfoType.From, isQuoteName = true, isDisableDiyParse = false, style = ExpressionStyle.Where, whereGlobalFilter = whereGlobalFilter, dbParams = select?._params }; //#462 添加 DbParams 解决
|
||
switch (exp.NodeType)
|
||
{
|
||
case ExpressionType.Quote: return ReadAnonymousField(_tables, _tableRule, field, parent, ref index, (exp as UnaryExpression)?.Operand, select, diymemexp, whereGlobalFilter, findIncludeMany, findSubSelectMany, isAllDtoMap);
|
||
case ExpressionType.Lambda: return ReadAnonymousField(_tables, _tableRule, field, parent, ref index, (exp as LambdaExpression)?.Body, select, diymemexp, whereGlobalFilter, findIncludeMany, findSubSelectMany, isAllDtoMap);
|
||
case ExpressionType.Negate:
|
||
case ExpressionType.NegateChecked:
|
||
parent.DbField = $"-({ExpressionLambdaToSql((exp as UnaryExpression)?.Operand, getTSC())})";
|
||
field.Append(", ").Append(parent.DbField);
|
||
LocalSetFieldAlias(ref index, false);
|
||
if (parent.CsType == null && exp.Type.IsValueType) parent.CsType = exp.Type;
|
||
return false;
|
||
case ExpressionType.Convert: return ReadAnonymousField(_tables, _tableRule, field, parent, ref index, (exp as UnaryExpression)?.Operand, select, diymemexp, whereGlobalFilter, findIncludeMany, findSubSelectMany, isAllDtoMap);
|
||
case ExpressionType.Constant:
|
||
var constExp = exp as ConstantExpression;
|
||
//处理自定义SQL语句,如: ToList(new {
|
||
// ccc = "now()",
|
||
// partby = "sum(num) over(PARTITION BY server_id,os,rid,chn order by id desc)"
|
||
//}),有缺点即 ccc partby 接受类型都是 string,可配合 Convert.ToXxx 类型转换,请看下面的兼容
|
||
if (constExp.Type.FullName == "System.String")
|
||
{
|
||
var constExpValue = constExp.Value?.ToString() ?? "NULL";
|
||
if (constExpValue == string.Empty) constExpValue = _common.FormatSql("{0}", "");
|
||
parent.DbField = constExpValue;
|
||
}
|
||
else
|
||
parent.DbField = _common.FormatSql("{0}", constExp?.Value);
|
||
field.Append(", ").Append(parent.DbField);
|
||
LocalSetFieldAlias(ref index, false);
|
||
if (parent.CsType == null && exp.Type.IsValueType) parent.CsType = exp.Type;
|
||
return false;
|
||
case ExpressionType.Conditional:
|
||
var condExp = exp as ConditionalExpression;
|
||
if (condExp.Test.IsParameter() == false) return ReadAnonymousField(_tables, _tableRule, field, parent, ref index,
|
||
(bool)Expression.Lambda(condExp.Test).Compile().DynamicInvoke() ? condExp.IfTrue : condExp.IfFalse, select, diymemexp, whereGlobalFilter, findIncludeMany, findSubSelectMany, isAllDtoMap);
|
||
break;
|
||
case ExpressionType.Call:
|
||
var callExp = exp as MethodCallExpression;
|
||
if (callExp.Method.Name == "ToList" && callExp.Object?.Type.FullName.StartsWith("FreeSql.ISelect`") == true)
|
||
{
|
||
parent.SubSelectMany = exp;
|
||
parent.CsType = exp.Type.GetGenericArguments().FirstOrDefault();
|
||
findSubSelectMany?.Add(exp);
|
||
return false;
|
||
}
|
||
//处理自定义SQL语句,如: ToList(new {
|
||
// ccc = Convert.ToDateTime("now()"),
|
||
// partby = Convert.ToDecimal("sum(num) over(PARTITION BY server_id,os,rid,chn order by id desc)")
|
||
//})
|
||
if (callExp.Method?.DeclaringType.FullName == "System.Convert" &&
|
||
callExp.Method.Name.StartsWith("To") &&
|
||
callExp.Arguments[0].NodeType == ExpressionType.Constant &&
|
||
callExp.Arguments[0].Type.FullName == "System.String")
|
||
parent.DbField = (callExp.Arguments[0] as ConstantExpression).Value?.ToString() ?? "NULL";
|
||
else
|
||
parent.DbField = ExpressionLambdaToSql(exp, getTSC());
|
||
field.Append(", ").Append(parent.DbField);
|
||
LocalSetFieldAlias(ref index, false);
|
||
if (parent.CsType == null && exp.Type.IsValueType) parent.CsType = exp.Type;
|
||
return false;
|
||
case ExpressionType.Parameter:
|
||
case ExpressionType.MemberAccess:
|
||
if ((_common.GetTableByEntity(exp.Type) != null || exp.Type.IsAnonymousType() && diymemexp != null) &&
|
||
//判断 [JsonMap] 并非导航对象
|
||
(exp.NodeType == ExpressionType.Parameter || exp is MemberExpression expMem && (
|
||
_common.GetTableByEntity(expMem.Expression.Type)?.ColumnsByCs.ContainsKey(expMem.Member.Name) == false ||
|
||
expMem.Expression.NodeType == ExpressionType.Parameter && expMem.Expression.Type.IsAnonymousType()) //<>h__TransparentIdentifier 是 Linq To Sql 的类型判断,此时为匿名类型
|
||
)
|
||
)
|
||
{
|
||
//加载表所有字段
|
||
var map = new List<SelectColumnInfo>();
|
||
ExpressionSelectColumn_MemberAccess(_tables, _tableRule, map, SelectTableInfoType.From, exp, true, diymemexp);
|
||
if (map.Any() == false)
|
||
{
|
||
if (diymemexp != null && diymemexp.ParseExpMapResult != null)
|
||
{
|
||
var withTempQueryParser = diymemexp as Select0Provider.WithTempQueryParser;
|
||
diymemexp.ParseExpMapResult.CopyTo(parent);
|
||
foreach (var child in parent.GetAllChilds())
|
||
{
|
||
if (withTempQueryParser != null)
|
||
child.DbField = $"{withTempQueryParser.ParseExpMatchedTable.Alias}.{child.DbNestedField}";
|
||
field.Append(", ").Append(child.DbField);
|
||
if (index >= 0)
|
||
{
|
||
child.DbNestedField = $"as{++index}";
|
||
field.Append(_common.FieldAsAlias(child.DbNestedField));
|
||
}
|
||
}
|
||
return false;
|
||
}
|
||
throw new Exception($"未能加载它的所有成员,不支持解析表达式树 {exp}");
|
||
}
|
||
var tb = parent.Table = map.First().Table.Table;
|
||
parent.CsType = tb.Type;
|
||
parent.Consturctor = tb.Type.InternalGetTypeConstructor0OrFirst();
|
||
parent.IsEntity = true;
|
||
for (var idx = 0; idx < map.Count; idx++)
|
||
{
|
||
var child = new ReadAnonymousTypeInfo
|
||
{
|
||
Property = tb.Properties.TryGetValue(map[idx].Column.CsName, out var tryprop) ? tryprop : tb.Type.GetProperty(map[idx].Column.CsName, BindingFlags.Public | BindingFlags.Instance),
|
||
CsName = map[idx].Column.CsName,
|
||
DbField = $"{map[idx].Table.Alias}.{_common.QuoteSqlName(map[idx].Column.Attribute.Name)}",
|
||
DbNestedField = _common.QuoteSqlName(map[idx].Column.Attribute.Name),
|
||
CsType = map[idx].Column.CsType,
|
||
MapType = map[idx].Column.Attribute.MapType
|
||
};
|
||
field.Append(", ").Append(_common.RereadColumn(map[idx].Column, child.DbField));
|
||
if (index >= 0)
|
||
{
|
||
child.DbNestedField = $"as{++index}";
|
||
field.Append(_common.FieldAsAlias(child.DbNestedField));
|
||
}
|
||
parent.Childs.Add(child);
|
||
}
|
||
if (_tables?.Count > 1)
|
||
{ //如果下级导航属性被 Include 过,则将他们也查询出来
|
||
foreach (var tr in tb.GetAllTableRef())
|
||
{
|
||
var memtbref = tr.Value;
|
||
if (memtbref.Exception != null) continue;
|
||
if (tb.Properties.TryGetValue(tr.Key, out var memProp) == false) continue;
|
||
switch (memtbref.RefType)
|
||
{
|
||
case TableRefType.ManyToMany:
|
||
case TableRefType.OneToMany:
|
||
case TableRefType.PgArrayToMany:
|
||
continue;
|
||
}
|
||
if (_tables.Any(a => a.Alias == $"{map.First().Table.Alias}__{memProp.Name}") == false) continue;
|
||
|
||
var child = new ReadAnonymousTypeInfo
|
||
{
|
||
Property = memProp,
|
||
CsName = memProp.Name,
|
||
CsType = memProp.PropertyType,
|
||
MapType = memProp.PropertyType
|
||
};
|
||
parent.Childs.Add(child);
|
||
ReadAnonymousField(_tables, _tableRule, field, child, ref index, Expression.MakeMemberAccess(exp, memProp), select, diymemexp, whereGlobalFilter, findIncludeMany, findSubSelectMany, false);
|
||
}
|
||
}
|
||
}
|
||
else
|
||
{
|
||
if (_tables != null && select != null && findIncludeMany != null && select._includeToList.Any() && exp.Type.IsGenericType &&
|
||
typeof(IEnumerable).IsAssignableFrom(exp.Type) &&
|
||
typeof(ICollection<>).MakeGenericType(exp.Type.GetGenericArguments().FirstOrDefault()).IsAssignableFrom(exp.Type))
|
||
{
|
||
var includeKey = "";
|
||
var memExp = exp as MemberExpression;
|
||
while (memExp != null)
|
||
{
|
||
includeKey = $"{memExp.Member.Name}.{includeKey}";
|
||
if (memExp.Expression.NodeType == ExpressionType.Parameter) break;
|
||
memExp = memExp.Expression as MemberExpression;
|
||
}
|
||
if (memExp != null && string.IsNullOrEmpty(includeKey) == false)
|
||
{
|
||
includeKey = includeKey.TrimEnd('.');
|
||
if (select._includeInfo.ContainsKey(includeKey))
|
||
{
|
||
parent.IncludeManyKey = includeKey;
|
||
parent.CsType = exp.Type.GetGenericArguments().FirstOrDefault();
|
||
findIncludeMany?.Add(includeKey);
|
||
return false;
|
||
}
|
||
}
|
||
}
|
||
if (diymemexp != null && exp is MemberExpression expMem2 && expMem2.Member.Name == "Key" && expMem2.Expression.Type.FullName.StartsWith("FreeSql.ISelectGroupingAggregate`"))
|
||
{
|
||
field.Append(diymemexp._field);
|
||
string dbNestedField = null;
|
||
if (diymemexp._map.Childs.Any() == false) //处理 GroupBy(a => a.Title) ToSql(g => new { tit = a.Key }, FieldAliasOptions.AsProperty) 问题
|
||
{
|
||
if (index >= 0)
|
||
{
|
||
dbNestedField = $"as{++index}";
|
||
field.Append(_common.FieldAsAlias(dbNestedField));
|
||
}
|
||
else if (string.IsNullOrEmpty(parent.CsName) == false)
|
||
{
|
||
dbNestedField = GetFieldAsCsName(parent.CsName);
|
||
if (index == ReadAnonymousFieldAsCsName && EndsWithDbNestedField(diymemexp._field, dbNestedField) == false) //DbField 和 CsName 相同的时候,不处理
|
||
field.Append(_common.FieldAsAlias(dbNestedField));
|
||
}
|
||
}
|
||
var parentProp = parent.Property;
|
||
diymemexp._map.CopyTo(parent); //可能会清空 parent.DbNestedField、CsName 值
|
||
parent.Property = parentProp; //若不加此行,会引用 GroupBy(..).ToList(a => new Dto { key = a.Key }) null 错误,CopyTo 之后 Property 变为 null
|
||
if (string.IsNullOrWhiteSpace(dbNestedField) == false)
|
||
parent.DbNestedField = dbNestedField;
|
||
return false;
|
||
}
|
||
if (parent.CsType == null) parent.CsType = exp.Type;
|
||
try
|
||
{
|
||
var pdbfield = parent.DbField = ExpressionLambdaToSql(exp, getTSC());
|
||
if (parent.MapType == null || _tables?.Any(a => a.Table?.IsRereadSql == true) == true)
|
||
{
|
||
var findcol = SearchColumnByField(_tables, null, parent.DbField);
|
||
if (parent.MapType == null) parent.MapType = findcol?.Attribute.MapType ?? exp.Type;
|
||
if (findcol != null) pdbfield = _common.RereadColumn(findcol, pdbfield);
|
||
}
|
||
field.Append(", ").Append(pdbfield);
|
||
LocalSetFieldAlias(ref index, _tables != null ||
|
||
SelectGroupingProvider._ParseExpOnlyDbField.Value != pdbfield);
|
||
}
|
||
finally
|
||
{
|
||
SelectGroupingProvider._ParseExpOnlyDbField.Value = null;
|
||
}
|
||
return false;
|
||
}
|
||
return false;
|
||
case ExpressionType.MemberInit:
|
||
var initExp = exp as MemberInitExpression;
|
||
parent.CsType = initExp.Type;
|
||
parent.Consturctor = initExp.NewExpression.Constructor;
|
||
if (initExp.NewExpression?.Arguments.Count > 0)
|
||
{
|
||
//处理构造参数
|
||
for (var a = 0; a < initExp.NewExpression.Arguments.Count; a++)
|
||
{
|
||
var child = new ReadAnonymousTypeInfo
|
||
{
|
||
Property = null,
|
||
CsName = initExp.NewExpression.Members != null ? initExp.NewExpression.Members[a].Name : (initExp.NewExpression.Arguments[a] as MemberExpression)?.Member.Name,
|
||
CsType = initExp.NewExpression.Arguments[a].Type,
|
||
MapType = initExp.NewExpression.Arguments[a].Type
|
||
};
|
||
parent.Childs.Add(child);
|
||
ReadAnonymousField(_tables, _tableRule, field, child, ref index, initExp.NewExpression.Arguments[a], select, diymemexp, whereGlobalFilter, findIncludeMany, findSubSelectMany, false);
|
||
}
|
||
}
|
||
else if (isAllDtoMap && _tables != null && _tables.Any() && initExp.NewExpression.Type != _tables.FirstOrDefault().Table.Type)
|
||
{
|
||
var dicBindings = initExp.Bindings?.Select(a => a.Member.Name).Distinct().ToDictionary(a => a, a => false);
|
||
//dto 映射
|
||
var dtoProps = initExp.NewExpression.Type.GetPropertiesDictIgnoreCase().Values;
|
||
foreach (var dtoProp in dtoProps)
|
||
{
|
||
foreach (var dtTb in _tables)
|
||
{
|
||
if (dtTb.Table.ColumnsByCs.TryGetValue(dtoProp.Name, out var trydtocol) == false)
|
||
{
|
||
if (diymemexp != null && dtTb.Parameter != null && dtTb.Parameter.Type.GetPropertiesDictIgnoreCase().TryGetValue(dtoProp.Name, out var dtTbProp))
|
||
{
|
||
var dbfield = diymemexp.ParseExp(new Expression[] { Expression.MakeMemberAccess(dtTb.Parameter, dtTbProp) });
|
||
if (diymemexp.ParseExpMapResult != null)
|
||
{
|
||
var diychild = new ReadAnonymousTypeInfo
|
||
{
|
||
Property = dtoProp,
|
||
CsName = dtoProp.Name,
|
||
CsType = dtTbProp.PropertyType,
|
||
MapType = dtTbProp.PropertyType
|
||
};
|
||
parent.Childs.Add(diychild);
|
||
diychild.DbField = $"{dtTb.Alias}.{diymemexp.ParseExpMapResult.DbNestedField}";
|
||
diychild.DbNestedField = diymemexp.ParseExpMapResult.DbNestedField;
|
||
field.Append(", ").Append(diychild.DbField);
|
||
if (index >= 0)
|
||
{
|
||
diychild.DbNestedField = $"as{++index}";
|
||
field.Append(_common.FieldAsAlias(diychild.DbNestedField));
|
||
}
|
||
break;
|
||
}
|
||
}
|
||
continue;
|
||
}
|
||
if (trydtocol.Attribute.IsIgnore == true) continue;
|
||
if (dicBindings?.ContainsKey(dtoProp.Name) == true) continue;
|
||
|
||
var child = new ReadAnonymousTypeInfo
|
||
{
|
||
Property = dtoProp,
|
||
CsName = dtoProp.Name,
|
||
CsType = trydtocol.CsType, // dtoProp.PropertyType,
|
||
MapType = trydtocol.Attribute.MapType
|
||
};
|
||
parent.Childs.Add(child);
|
||
if (dtTb.Parameter != null)
|
||
ReadAnonymousField(_tables, _tableRule, field, child, ref index, Expression.Property(
|
||
dtTb.Parameter.Type == dtTb.Table.Type ? (Expression)dtTb.Parameter : Expression.TypeAs(dtTb.Parameter, dtTb.Table.Type),
|
||
dtTb.Table.Properties[trydtocol.CsName]), select, diymemexp, whereGlobalFilter, findIncludeMany, findSubSelectMany, isAllDtoMap);
|
||
else
|
||
{
|
||
child.DbField = $"{dtTb.Alias}.{_common.QuoteSqlName(trydtocol.Attribute.Name)}";
|
||
child.DbNestedField = _common.QuoteSqlName(trydtocol.Attribute.Name);
|
||
field.Append(", ").Append(_common.RereadColumn(trydtocol, child.DbField));
|
||
if (index >= 0)
|
||
{
|
||
child.DbNestedField = $"as{++index}";
|
||
field.Append(_common.FieldAsAlias(child.DbNestedField));
|
||
}
|
||
}
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
if (initExp.Bindings?.Count > 0)
|
||
{
|
||
//指定 dto映射
|
||
for (var a = 0; a < initExp.Bindings.Count; a++)
|
||
{
|
||
var initAssignExp = (initExp.Bindings[a] as MemberAssignment);
|
||
if (initAssignExp == null) continue;
|
||
var child = new ReadAnonymousTypeInfo
|
||
{
|
||
Property = initExp.Type.GetProperty(initExp.Bindings[a].Member.Name, BindingFlags.Public | BindingFlags.Instance), //#427 不能使用 BindingFlags.IgnoreCase
|
||
CsName = initExp.Bindings[a].Member.Name,
|
||
CsType = initAssignExp.Expression.Type,
|
||
MapType = initAssignExp.Expression.Type
|
||
};
|
||
if (child.Property == null) child.ReflectionField = initExp.Type.GetField(initExp.Bindings[a].Member.Name, BindingFlags.Public | BindingFlags.Instance);
|
||
parent.Childs.Add(child);
|
||
ReadAnonymousField(_tables, _tableRule, field, child, ref index, initAssignExp.Expression, select, diymemexp, whereGlobalFilter, findIncludeMany, findSubSelectMany, false);
|
||
}
|
||
}
|
||
if (parent.Childs.Any() == false) throw new Exception(CoreStrings.Mapping_Exception_HasNo_SamePropertyName(initExp.NewExpression.Type.Name));
|
||
return true;
|
||
case ExpressionType.New:
|
||
var newExp = exp as NewExpression;
|
||
parent.CsType = newExp.Type;
|
||
parent.Consturctor = newExp.Constructor;
|
||
if (newExp.Arguments?.Count > 0 &&
|
||
(
|
||
newExp.Type.IsAnonymousType() ||
|
||
newExp.Arguments.Any(a =>
|
||
{
|
||
if (a.NodeType != ExpressionType.Constant) return true;
|
||
var constVal = (a as ConstantExpression)?.Value;
|
||
if (constVal == null) return false; //- 修复 实体类拥有构造参数时,ToList\<DTO\> 映射查询无效的 bug;
|
||
if (object.Equals(constVal, a.Type.CreateInstanceGetDefaultValue())) return false;
|
||
return true;
|
||
})
|
||
))
|
||
{
|
||
//处理构造参数
|
||
for (var a = 0; a < newExp.Arguments.Count; a++)
|
||
{
|
||
var csname = newExp.Members != null ? newExp.Members[a].Name : (newExp.Arguments[a] as MemberExpression)?.Member.Name;
|
||
var child = new ReadAnonymousTypeInfo
|
||
{
|
||
Property = null,
|
||
CsName = csname,
|
||
CsType = newExp.Arguments[a].Type,
|
||
MapType = newExp.Arguments[a].Type
|
||
};
|
||
parent.Childs.Add(child);
|
||
ReadAnonymousField(_tables, _tableRule, field, child, ref index, newExp.Arguments[a], select, diymemexp, whereGlobalFilter, findIncludeMany, findSubSelectMany, false);
|
||
if (child.CsName == null)
|
||
child.CsName = csname;
|
||
}
|
||
}
|
||
else
|
||
{
|
||
parent.IsDefaultCtor = true;
|
||
//dto 映射
|
||
var dtoProps2 = newExp.Type.GetPropertiesDictIgnoreCase().Values;
|
||
foreach (var dtoProp in dtoProps2)
|
||
{
|
||
foreach (var dtTb in _tables)
|
||
{
|
||
if (dtTb.Table.ColumnsByCs.TryGetValue(dtoProp.Name, out var trydtocol) == false)
|
||
{
|
||
if (diymemexp != null && dtTb.Parameter != null && dtTb.Parameter.Type.GetPropertiesDictIgnoreCase().TryGetValue(dtoProp.Name, out var dtTbProp))
|
||
{
|
||
var dbfield = diymemexp.ParseExp(new Expression[] { Expression.MakeMemberAccess(dtTb.Parameter, dtTbProp) });
|
||
if (diymemexp.ParseExpMapResult != null)
|
||
{
|
||
var diychild = new ReadAnonymousTypeInfo
|
||
{
|
||
Property = dtoProp,
|
||
CsName = dtoProp.Name,
|
||
CsType = dtTbProp.PropertyType,
|
||
MapType = dtTbProp.PropertyType
|
||
};
|
||
parent.Childs.Add(diychild);
|
||
diychild.DbField = $"{dtTb.Alias}.{diymemexp.ParseExpMapResult.DbNestedField}";
|
||
diychild.DbNestedField = diymemexp.ParseExpMapResult.DbNestedField;
|
||
field.Append(", ").Append(diychild.DbField);
|
||
if (index >= 0)
|
||
{
|
||
diychild.DbNestedField = $"as{++index}";
|
||
field.Append(_common.FieldAsAlias(diychild.DbNestedField));
|
||
}
|
||
break;
|
||
}
|
||
}
|
||
continue;
|
||
}
|
||
if (trydtocol.Attribute.IsIgnore == true) continue;
|
||
|
||
var child = new ReadAnonymousTypeInfo
|
||
{
|
||
Property = dtoProp,
|
||
CsName = dtoProp.Name,
|
||
CsType = trydtocol.CsType, //dtoProp.PropertyType,
|
||
MapType = trydtocol.Attribute.MapType
|
||
};
|
||
parent.Childs.Add(child);
|
||
if (dtTb.Parameter != null)
|
||
ReadAnonymousField(_tables, _tableRule, field, child, ref index, Expression.Property(
|
||
dtTb.Parameter.Type == dtTb.Table.Type ? (Expression)dtTb.Parameter : Expression.TypeAs(dtTb.Parameter, dtTb.Table.Type),
|
||
dtTb.Table.Properties[trydtocol.CsName]), select, diymemexp, whereGlobalFilter, findIncludeMany, findSubSelectMany, isAllDtoMap);
|
||
else
|
||
{
|
||
child.DbField = _common.RereadColumn(trydtocol, $"{dtTb.Alias}.{_common.QuoteSqlName(trydtocol.Attribute.Name)}");
|
||
child.DbNestedField = _common.QuoteSqlName(trydtocol.Attribute.Name);
|
||
field.Append(", ").Append(child.DbField);
|
||
if (index >= 0)
|
||
{
|
||
child.DbNestedField = $"as{++index}";
|
||
field.Append(_common.FieldAsAlias(child.DbNestedField));
|
||
}
|
||
}
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
if (parent.Childs.Any() == false) throw new Exception(CoreStrings.Mapping_Exception_HasNo_SamePropertyName(newExp.Type.Name));
|
||
return true;
|
||
}
|
||
parent.DbField = ExpressionLambdaToSql(exp, getTSC()); //解决 new { a = id + 1 } 翻译后 ((id+1)) 问题
|
||
field.Append(", ").Append(parent.DbField);
|
||
LocalSetFieldAlias(ref index, false);
|
||
if (parent.CsType == null && exp.Type.IsValueType) parent.CsType = exp.Type;
|
||
return false;
|
||
}
|
||
public object ReadAnonymous(ReadAnonymousTypeInfo parent, DbDataReader dr, ref int index, bool notRead, ReadAnonymousDbValueRef dbValue, int rowIndex,
|
||
List<NativeTuple<string, IList, int>> fillIncludeMany, List<NativeTuple<Expression, IList, int>> fillSubSelectMany)
|
||
{
|
||
if (parent.Childs.Any() == false && string.IsNullOrEmpty(parent.IncludeManyKey) && parent.SubSelectMany == null)
|
||
{
|
||
if (notRead)
|
||
{
|
||
++index;
|
||
if (parent.Property != null)
|
||
return Utils.GetDataReaderValue(parent.Property.PropertyType, null);
|
||
return Utils.GetDataReaderValue(parent.CsType, null);
|
||
}
|
||
object objval = Utils.InternalDataReaderGetValue(_common, dr, ++index); // dr.GetValue(++index);
|
||
if (dbValue != null) dbValue.DbValue = objval == DBNull.Value ? null : objval;
|
||
if (parent.CsType != parent.MapType)
|
||
objval = Utils.GetDataReaderValue(parent.MapType, objval);
|
||
objval = Utils.GetDataReaderValue(parent.CsType, objval);
|
||
if (parent.Property != null && parent.CsType != parent.Property.PropertyType)
|
||
objval = Utils.GetDataReaderValue(parent.Property.PropertyType, objval);
|
||
if (objval == DBNull.Value) objval = null;
|
||
return objval;
|
||
}
|
||
var ctorParmsLength = 0;
|
||
object ret;
|
||
if (string.IsNullOrEmpty(parent.IncludeManyKey) == false)
|
||
{
|
||
if (parent.MapType == typeof(ObservableCollection<>).MakeGenericType(parent.CsType))
|
||
ret = parent.MapType.CreateInstanceGetDefaultValue();
|
||
else
|
||
ret = typeof(List<>).MakeGenericType(parent.CsType).CreateInstanceGetDefaultValue();
|
||
fillIncludeMany?.Add(NativeTuple.Create(parent.IncludeManyKey, ret as IList, rowIndex));
|
||
}
|
||
else if (parent.SubSelectMany != null)
|
||
{
|
||
ret = typeof(List<>).MakeGenericType(parent.CsType).CreateInstanceGetDefaultValue();
|
||
fillSubSelectMany?.Add(NativeTuple.Create(parent.SubSelectMany, ret as IList, rowIndex));
|
||
}
|
||
else if (parent.IsDefaultCtor || parent.IsEntity || (ctorParmsLength = parent.Consturctor.GetParameters()?.Length ?? 0) == 0)
|
||
ret = parent.CsType?.CreateInstanceGetDefaultValue() ?? parent.Consturctor.Invoke(null);
|
||
else
|
||
{
|
||
var ctorParms = new object[ctorParmsLength];
|
||
var ctorParmsDefs = parent.Consturctor.GetParameters();
|
||
for (var c = 0; c < ctorParmsLength; c++)
|
||
ctorParms[c] = ReadAnonymous(parent.Childs[c], dr, ref index, notRead, null, rowIndex, fillIncludeMany, fillSubSelectMany);
|
||
ret = parent.Consturctor.Invoke(ctorParms);
|
||
}
|
||
|
||
var isnull = notRead;
|
||
for (var b = ctorParmsLength; b < parent.Childs.Count; b++)
|
||
{
|
||
var dbval = parent.IsEntity ? new ReadAnonymousDbValueRef() : null;
|
||
var objval = ReadAnonymous(parent.Childs[b], dr, ref index, notRead, dbval, rowIndex, fillIncludeMany, fillSubSelectMany);
|
||
if (isnull == false && parent.IsEntity && dbval.DbValue == null && parent.Table != null && parent.Table.ColumnsByCs.TryGetValue(parent.Childs[b].CsName, out var trycol) && trycol.Attribute.IsPrimary)
|
||
isnull = true;
|
||
|
||
if (isnull == false)
|
||
{
|
||
var prop = parent.Childs[b].Property;
|
||
if (prop?.CanWrite == true) prop.SetValue(ret, objval, null);
|
||
else if (prop == null) parent.Childs[b].ReflectionField?.SetValue(ret, objval);
|
||
}
|
||
}
|
||
return isnull ? null : ret;
|
||
}
|
||
public class ReadAnonymousDbValueRef
|
||
{
|
||
public object DbValue { get; set; }
|
||
}
|
||
|
||
public ColumnInfo SearchColumnByField(List<SelectTableInfo> _tables, TableInfo currentTable, string field)
|
||
{
|
||
if (_tables != null)
|
||
{
|
||
var testCol = _common.TrimQuoteSqlName(field).Split(new[] { '.' }, 2);
|
||
if (testCol.Length == 2)
|
||
{
|
||
var testTb = _tables.Where(a => a.Table != null && a.Alias == testCol[0]).ToArray();
|
||
if (testTb.Length == 1 && testTb[0].Table.Columns.TryGetValue(testCol[1], out var trytstcol) == true)
|
||
return trytstcol;
|
||
}
|
||
}
|
||
if (currentTable != null)
|
||
{
|
||
var testCol = _common.TrimQuoteSqlName(field);
|
||
if (currentTable.Columns.TryGetValue(testCol, out var trytstcol))
|
||
return trytstcol;
|
||
}
|
||
return null;
|
||
}
|
||
|
||
public string ExpressionSelectColumn_MemberAccess(List<SelectTableInfo> _tables, Func<Type, string, string> _tableRule, List<SelectColumnInfo> _selectColumnMap, SelectTableInfoType tbtype, Expression exp, bool isQuoteName, BaseDiyMemberExpression diymemexp)
|
||
{
|
||
return ExpressionLambdaToSql(exp, new ExpTSC { _tables = _tables, _tableRule = _tableRule, _selectColumnMap = _selectColumnMap, diymemexp = diymemexp, tbtype = tbtype, isQuoteName = isQuoteName, isDisableDiyParse = false, style = ExpressionStyle.SelectColumns });
|
||
}
|
||
|
||
public string[] ExpressionSelectColumns_MemberAccess_New_NewArrayInit(List<SelectTableInfo> _tables, Func<Type, string, string> _tableRule, Expression exp, bool isQuoteName, BaseDiyMemberExpression diymemexp)
|
||
{
|
||
switch (exp?.NodeType)
|
||
{
|
||
case ExpressionType.Quote: return ExpressionSelectColumns_MemberAccess_New_NewArrayInit(_tables, _tableRule, (exp as UnaryExpression)?.Operand, isQuoteName, diymemexp);
|
||
case ExpressionType.Lambda: return ExpressionSelectColumns_MemberAccess_New_NewArrayInit(_tables, _tableRule, (exp as LambdaExpression)?.Body, isQuoteName, diymemexp);
|
||
case ExpressionType.Convert: return ExpressionSelectColumns_MemberAccess_New_NewArrayInit(_tables, _tableRule, (exp as UnaryExpression)?.Operand, isQuoteName, diymemexp);
|
||
case ExpressionType.Constant: return new[] { ExpressionSelectColumn_MemberAccess(_tables, _tableRule, null, SelectTableInfoType.From, exp, isQuoteName, diymemexp) };
|
||
case ExpressionType.Call:
|
||
case ExpressionType.MemberAccess: return ExpressionSelectColumn_MemberAccess(_tables, _tableRule, null, SelectTableInfoType.From, exp, isQuoteName, diymemexp).Trim('(', ')', '\'').Split(new[] { "','" }, StringSplitOptions.RemoveEmptyEntries);
|
||
case ExpressionType.New:
|
||
var newExp = exp as NewExpression;
|
||
if (newExp == null) break;
|
||
var newExpMembers = new string[newExp.Members.Count];
|
||
for (var a = 0; a < newExpMembers.Length; a++) newExpMembers[a] = ExpressionSelectColumn_MemberAccess(_tables, _tableRule, null, SelectTableInfoType.From, newExp.Arguments[a], isQuoteName, diymemexp);
|
||
return newExpMembers.Distinct().Select(a => a.Trim('\'')).ToArray();
|
||
case ExpressionType.NewArrayInit:
|
||
var newArr = exp as NewArrayExpression;
|
||
if (newArr == null) break;
|
||
var newArrMembers = new List<string>();
|
||
foreach (var newArrExp in newArr.Expressions) newArrMembers.AddRange(ExpressionSelectColumns_MemberAccess_New_NewArrayInit(_tables, _tableRule, newArrExp, isQuoteName, diymemexp));
|
||
return newArrMembers.Distinct().Select(a => a.Trim('\'')).ToArray();
|
||
default: throw new ArgumentException(CoreStrings.Unable_Parse_Expression(exp));
|
||
}
|
||
return new string[0];
|
||
}
|
||
|
||
static readonly Dictionary<ExpressionType, string> dicExpressionOperator = new Dictionary<ExpressionType, string>() {
|
||
{ ExpressionType.OrElse, "OR" },
|
||
{ ExpressionType.Or, "|" },
|
||
{ ExpressionType.AndAlso, "AND" },
|
||
{ ExpressionType.And, "&" },
|
||
{ ExpressionType.GreaterThan, ">" },
|
||
{ ExpressionType.GreaterThanOrEqual, ">=" },
|
||
{ ExpressionType.LessThan, "<" },
|
||
{ ExpressionType.LessThanOrEqual, "<=" },
|
||
{ ExpressionType.NotEqual, "<>" },
|
||
{ ExpressionType.Add, "+" },
|
||
{ ExpressionType.Subtract, "-" },
|
||
{ ExpressionType.Multiply, "*" },
|
||
{ ExpressionType.Divide, "/" },
|
||
{ ExpressionType.Modulo, "%" },
|
||
{ ExpressionType.Equal, "=" },
|
||
};
|
||
|
||
public string ExpressionWhereLambdaNoneForeignObject(List<SelectTableInfo> _tables, Func<Type, string, string> _tableRule, TableInfo table, List<SelectColumnInfo> _selectColumnMap, Expression exp, BaseDiyMemberExpression diymemexp, List<DbParameter> dbParams)
|
||
{
|
||
var sql = ExpressionLambdaToSql(exp, new ExpTSC { _tables = _tables, _tableRule = _tableRule, _selectColumnMap = _selectColumnMap, diymemexp = diymemexp, tbtype = SelectTableInfoType.From, isQuoteName = true, isDisableDiyParse = false, style = ExpressionStyle.Where, currentTable = table, dbParams = dbParams });
|
||
return GetBoolString(exp, sql);
|
||
}
|
||
|
||
public string ExpressionWhereLambda(List<SelectTableInfo> _tables, Func<Type, string, string> _tableRule, Expression exp, BaseDiyMemberExpression diymemexp, List<GlobalFilter.Item> whereGlobalFilter, List<DbParameter> dbParams)
|
||
{
|
||
if (_tables?.Count > 1)
|
||
{
|
||
foreach (var tb in _tables)
|
||
if (tb.Parameter != null && tb.AliasInit.StartsWith("SP10"))
|
||
tb.Alias = tb.Parameter.Name;
|
||
}
|
||
var sql = ExpressionLambdaToSql(exp, new ExpTSC { _tables = _tables, _tableRule = _tableRule, diymemexp = diymemexp, tbtype = SelectTableInfoType.From, isQuoteName = true, isDisableDiyParse = false, style = ExpressionStyle.Where, whereGlobalFilter = whereGlobalFilter, dbParams = dbParams });
|
||
return GetBoolString(exp, sql);
|
||
}
|
||
static ConcurrentDictionary<string, Regex> dicRegexAlias = new ConcurrentDictionary<string, Regex>();
|
||
public void ExpressionJoinLambda(List<SelectTableInfo> _tables, Func<Type, string, string> _tableRule, SelectTableInfoType tbtype, Expression exp, BaseDiyMemberExpression diymemexp, List<GlobalFilter.Item> whereGlobalFilter)
|
||
{
|
||
var tbidx = _tables.Count;
|
||
if (tbidx > 1)
|
||
{
|
||
foreach (var tb in _tables)
|
||
if (tb.Parameter != null && tb.AliasInit.StartsWith("SP10"))
|
||
tb.Alias = tb.Parameter.Name;
|
||
}
|
||
var sql = ExpressionLambdaToSql(exp, new ExpTSC { _tables = _tables, _tableRule = _tableRule, diymemexp = diymemexp, tbtype = tbtype, isQuoteName = true, isDisableDiyParse = false, style = ExpressionStyle.Where, whereGlobalFilter = whereGlobalFilter });
|
||
sql = GetBoolString(exp, sql);
|
||
|
||
if (_tables.Count > tbidx)
|
||
{
|
||
_tables[tbidx].Type = tbtype;
|
||
_tables[tbidx].On = sql;
|
||
for (var a = tbidx + 1; a < _tables.Count; a++)
|
||
_tables[a].Type = SelectTableInfoType.From;
|
||
}
|
||
else
|
||
{
|
||
var find = _tables.Where((a, c) => c > 0 &&
|
||
(a.Type == tbtype || a.Type == SelectTableInfoType.From || a.Type == SelectTableInfoType.LeftJoin) &&
|
||
string.IsNullOrEmpty(a.On) &&
|
||
dicRegexAlias.GetOrAdd(a.Alias, alias => new Regex($@"\b{alias}\.", RegexOptions.Compiled)).IsMatch(sql)).LastOrDefault();
|
||
if (find != null)
|
||
{
|
||
find.Type = tbtype;
|
||
find.On = sql;
|
||
}
|
||
}
|
||
}
|
||
static ConcurrentDictionary<Type, MethodInfo> _dicExpressionLambdaToSqlAsSelectMethodInfo = new ConcurrentDictionary<Type, MethodInfo>();
|
||
static ConcurrentDictionary<Type, MethodInfo> _dicExpressionLambdaToSqlAsSelectWhereMethodInfo = new ConcurrentDictionary<Type, MethodInfo>();
|
||
static ConcurrentDictionary<Type, MethodInfo> _dicExpressionLambdaToSqlAsSelectWhereSqlMethodInfo = new ConcurrentDictionary<Type, MethodInfo>();
|
||
static ConcurrentDictionary<Type, ConcurrentDictionary<string, MethodInfo>> _dicExpressionLambdaToSqlAsSelectAggMethodInfo = new ConcurrentDictionary<Type, ConcurrentDictionary<string, MethodInfo>>();
|
||
internal static ConcurrentDictionary<Type, PropertyInfo> _dicNullableValueProperty = new ConcurrentDictionary<Type, PropertyInfo>();
|
||
static ConcurrentDictionary<Type, Expression> _dicFreeSqlGlobalExtensionsAsSelectExpression = new ConcurrentDictionary<Type, Expression>();
|
||
static MethodInfo MethodDateTimeSubtractDateTime = typeof(DateTime).GetMethod("Subtract", new Type[] { typeof(DateTime) });
|
||
static MethodInfo MethodDateTimeSubtractTimeSpan = typeof(DateTime).GetMethod("Subtract", new Type[] { typeof(TimeSpan) });
|
||
static MethodInfo MethodMathFloor = typeof(Math).GetMethod("Floor", new Type[] { typeof(double) });
|
||
|
||
public string GetBoolString(Expression exp, string sql)
|
||
{
|
||
var isBool = exp.Type.NullableTypeOrThis() == typeof(bool);
|
||
if (exp.NodeType == ExpressionType.MemberAccess && isBool && sql.Contains(" IS ") == false && sql.Contains(" = ") == false)
|
||
return $"{sql} = {formatSql(true, null, null, null)}";
|
||
if (isBool)
|
||
return GetBoolString(sql);
|
||
return sql;
|
||
}
|
||
static string GetBoolString(string sql)
|
||
{
|
||
switch (sql)
|
||
{
|
||
case "1":
|
||
case "'t'": return "1=1";
|
||
case "0":
|
||
case "'f'": return "1=2";
|
||
default: return sql;
|
||
}
|
||
}
|
||
public string ExpressionBinary(string oper, Expression leftExp, Expression rightExp, ExpTSC tsc)
|
||
{
|
||
if (
|
||
leftExp.Type == rightExp.Type &&
|
||
|
||
leftExp.NodeType == ExpressionType.Convert &&
|
||
leftExp is UnaryExpression leftExpUexp &&
|
||
leftExpUexp.Operand?.Type.NullableTypeOrThis().IsEnum == true &&
|
||
|
||
rightExp.NodeType == ExpressionType.Convert &&
|
||
rightExp is UnaryExpression rightExpUexp &&
|
||
rightExpUexp.Operand?.Type.NullableTypeOrThis().IsEnum == true)
|
||
{
|
||
leftExp = leftExpUexp.Operand;
|
||
rightExp = rightExpUexp.Operand;
|
||
}
|
||
switch (oper)
|
||
{
|
||
case "OR":
|
||
case "|":
|
||
case "&":
|
||
case "+":
|
||
case "-":
|
||
if (oper == "+" && (leftExp.Type == typeof(string) || rightExp.Type == typeof(string)))
|
||
return _common.StringConcat(new[] { ExpressionLambdaToSql(leftExp, tsc), ExpressionLambdaToSql(rightExp, tsc) }, new[] { leftExp.Type, rightExp.Type });
|
||
if (oper == "-" && leftExp.Type.NullableTypeOrThis() == typeof(DateTime))
|
||
{
|
||
if (rightExp.Type.NullableTypeOrThis() == typeof(DateTime))
|
||
return ExpressionLambdaToSql(Expression.Call(leftExp, MethodDateTimeSubtractDateTime, rightExp), tsc);
|
||
if (rightExp.Type.NullableTypeOrThis() == typeof(TimeSpan))
|
||
return ExpressionLambdaToSql(Expression.Call(leftExp, MethodDateTimeSubtractTimeSpan, rightExp), tsc);
|
||
}
|
||
if (oper == "OR")
|
||
{
|
||
var leftBool = ExpressionLambdaToSql(leftExp, tsc);
|
||
if (SearchColumnByField(tsc._tables, tsc.currentTable, leftBool) != null) leftBool = $"{leftBool} = {formatSql(true, null, null, null)}";
|
||
else leftBool = GetBoolString(leftBool);
|
||
|
||
var rightBool = ExpressionLambdaToSql(rightExp, tsc);
|
||
if (SearchColumnByField(tsc._tables, tsc.currentTable, rightBool) != null) rightBool = $"{rightBool} = {formatSql(true, null, null, null)}";
|
||
else rightBool = GetBoolString(rightBool);
|
||
return $"({leftBool} {oper} {rightBool})";
|
||
}
|
||
return $"({ExpressionLambdaToSql(leftExp, tsc)} {oper} {ExpressionLambdaToSql(rightExp, tsc)})";
|
||
case "=":
|
||
case "<>":
|
||
if (leftExp.NodeType == ExpressionType.Call &&
|
||
rightExp.NodeType == ExpressionType.Constant)
|
||
{
|
||
var leftExpCall = leftExp as MethodCallExpression;
|
||
//vb 语法,将字符串比较转换为了 CompareString
|
||
if (leftExpCall.Method.Name == "CompareString" &&
|
||
leftExpCall.Method.DeclaringType?.FullName == "Microsoft.VisualBasic.CompilerServices.Operators" &&
|
||
leftExpCall.Arguments.Count == 3 &&
|
||
leftExpCall.Arguments[2].Type == typeof(bool) &&
|
||
rightExp.Type == typeof(int) &&
|
||
(int)(rightExp as ConstantExpression).Value == 0)
|
||
return ExpressionBinary(oper, leftExpCall.Arguments[0], leftExpCall.Arguments[1], tsc);
|
||
}
|
||
var exptb = _common.GetTableByEntity(leftExp.Type);
|
||
if (exptb?.Properties.Any() == true) leftExp = Expression.MakeMemberAccess(leftExp, exptb.Properties[(exptb.Primarys.FirstOrDefault() ?? exptb.Columns.FirstOrDefault().Value)?.CsName]);
|
||
exptb = _common.GetTableByEntity(leftExp.Type);
|
||
if (exptb?.Properties.Any() == true) rightExp = Expression.MakeMemberAccess(rightExp, exptb.Properties[(exptb.Primarys.FirstOrDefault() ?? exptb.Columns.FirstOrDefault().Value).CsName]);
|
||
break;
|
||
}
|
||
|
||
Type oldMapType = null;
|
||
var left = ExpressionLambdaToSql(leftExp, tsc);
|
||
var leftMapColumn = SearchColumnByField(tsc._tables, tsc.currentTable, left);
|
||
var isLeftMapType = leftMapColumn != null && new[] { "AND", "OR", "*", "/", "+", "-" }.Contains(oper) == false && (leftMapColumn.Attribute.MapType != rightExp.Type || leftMapColumn.CsType != rightExp.Type);
|
||
ColumnInfo rightMapColumn = null;
|
||
var isRightMapType = false;
|
||
if (isLeftMapType) oldMapType = tsc.SetMapTypeReturnOld(leftMapColumn.Attribute.MapType);
|
||
|
||
var right = ExpressionLambdaToSql(rightExp, tsc);
|
||
if (right != "NULL" && isLeftMapType &&
|
||
//判断参数化后的bug
|
||
!(right.Contains('@') || right.Contains('?') || right.Contains(':')) &&
|
||
//三元表达式后,取消此条件 #184
|
||
tsc.mapType != null)
|
||
{
|
||
var enumType = leftMapColumn.CsType.NullableTypeOrThis();
|
||
if (enumType.IsEnum)
|
||
{
|
||
rightMapColumn = SearchColumnByField(tsc._tables, tsc.currentTable, right);
|
||
if (rightMapColumn == null)
|
||
right = formatSql(Enum.Parse(enumType, right.StartsWith("N'") ? right.Substring(1).Trim('\'') : right.Trim('\'')), leftMapColumn.Attribute.MapType, leftMapColumn, tsc.dbParams);
|
||
}
|
||
}
|
||
if (leftMapColumn == null)
|
||
{
|
||
rightMapColumn = SearchColumnByField(tsc._tables, tsc.currentTable, right);
|
||
//.Set(a => a.NotTaxTotalCostPrice == report.NotTaxCostPrice * a.CurrentQty) * / + - 解决 report.NotTaxCostPrice 小数点问题
|
||
isRightMapType = rightMapColumn != null && new[] { "AND", "OR", "*", "/", "+", "-" }.Contains(oper) == false && (rightMapColumn.Attribute.MapType != leftExp.Type || rightMapColumn.CsType != leftExp.Type);
|
||
if (isRightMapType)
|
||
{
|
||
oldMapType = tsc.SetMapTypeReturnOld(rightMapColumn.Attribute.MapType);
|
||
left = ExpressionLambdaToSql(leftExp, tsc);
|
||
if (left != "NULL" && isRightMapType &&
|
||
//判断参数化后的bug
|
||
!(left.Contains('@') || left.Contains('?') || left.Contains(':')) &&
|
||
//三元表达式后,取消此条件 #184
|
||
tsc.mapType != null)
|
||
{
|
||
var enumType = rightMapColumn.CsType.NullableTypeOrThis();
|
||
if (enumType.IsEnum)
|
||
left = formatSql(Enum.Parse(enumType, left.StartsWith("N'") ? left.Substring(1).Trim('\'') : left.Trim('\'')), rightMapColumn.Attribute.MapType, rightMapColumn, tsc.dbParams);
|
||
}
|
||
}
|
||
}
|
||
if (leftExp.Type.NullableTypeOrThis() == typeof(bool) && (left.EndsWith(" IS NOT NULL") || left.EndsWith(" IS NULL") || leftExp.NodeType != ExpressionType.MemberAccess && rightExp.NodeType != ExpressionType.MemberAccess))
|
||
{
|
||
var leftExpCall = leftExp as MethodCallExpression;
|
||
if (leftExpCall == null || !(leftExpCall.Method.DeclaringType == typeof(SqlExt) && leftExpCall.Method.Name == nameof(SqlExt.IsNull)))
|
||
{
|
||
if (oper == "=")
|
||
{
|
||
var trueVal = formatSql(true, null, null, null);
|
||
var falseVal = formatSql(false, null, null, null);
|
||
if (left == trueVal) return right;
|
||
else if (left == falseVal) return $"not({right})";
|
||
else if (right == trueVal) return left;
|
||
else if (right == falseVal) return $"not({left})";
|
||
}
|
||
else if (oper == "<>")
|
||
{
|
||
var trueVal = formatSql(true, null, null, null);
|
||
var falseVal = formatSql(false, null, null, null);
|
||
if (left == trueVal) return $"not({right})";
|
||
else if (left == falseVal) return right;
|
||
else if (right == trueVal) return $"not({left})";
|
||
else if (right == falseVal) return left;
|
||
}
|
||
}
|
||
}
|
||
if (left == "NULL")
|
||
{
|
||
var tmp = right;
|
||
right = left;
|
||
left = tmp;
|
||
}
|
||
if (right == "NULL") oper = oper == "=" ? " IS " : " IS NOT ";
|
||
switch (oper)
|
||
{
|
||
case "%": return _common.Mod(left, right, leftExp.Type, rightExp.Type);
|
||
case "/":
|
||
if (leftExp.Type.IsIntegerType() && rightExp.Type.IsIntegerType()) return _common.Div(left, right, leftExp.Type, rightExp.Type);
|
||
break;
|
||
case "AND":
|
||
case "OR":
|
||
if (leftMapColumn != null) left = $"{left} = {formatSql(true, null, null, null)}";
|
||
else left = GetBoolString(left);
|
||
if (rightMapColumn != null) right = $"{right} = {formatSql(true, null, null, null)}";
|
||
else right = GetBoolString(right);
|
||
break;
|
||
}
|
||
tsc.SetMapColumnTmp(null).SetMapTypeReturnOld(oldMapType);
|
||
return $"{left} {oper} {right}";
|
||
}
|
||
static ConcurrentDictionary<Type, bool> _dicTypeExistsExpressionCallAttribute = new ConcurrentDictionary<Type, bool>();
|
||
static ConcurrentDictionary<Type, ConcurrentDictionary<string, bool>> _dicMethodExistsExpressionCallAttribute = new ConcurrentDictionary<Type, ConcurrentDictionary<string, bool>>();
|
||
static ConcurrentDictionary<Type, FieldInfo[]> _dicTypeExpressionCallClassContextFields = new ConcurrentDictionary<Type, FieldInfo[]>();
|
||
static ThreadLocal<List<BaseDiyMemberExpression>> _subSelectParentDiyMemExps = new ThreadLocal<List<BaseDiyMemberExpression>>(); //子查询的所有父自定义查询,比如分组之后的子查询
|
||
static ConcurrentDictionary<Type, MethodInfo> _dicSelectMethodToSql = new ConcurrentDictionary<Type, MethodInfo>();
|
||
public string ExpressionLambdaToSql(Expression exp, ExpTSC tsc)
|
||
{
|
||
if (exp == null) return "";
|
||
if (tsc.dbParams != null && tsc.mapColumnTmp != null && tsc.mapColumnTmp.CsType.NullableTypeOrThis() != exp.Type) tsc.SetMapColumnTmp(null);
|
||
if (tsc.isDisableDiyParse == false)
|
||
{
|
||
var args = new Aop.ParseExpressionEventArgs(exp, ukexp => ExpressionLambdaToSql(ukexp, tsc.CloneDisableDiyParse()), tsc._tables);
|
||
if (_common._orm.Aop.ParseExpressionHandler != null)
|
||
{
|
||
_common._orm.Aop.ParseExpressionHandler(this, args);
|
||
if (string.IsNullOrEmpty(args.Result) == false) return args.Result;
|
||
}
|
||
ParseExpressionNoAsSelect(this, args, tsc._tableRule, tsc.whereGlobalFilter);
|
||
if (string.IsNullOrEmpty(args.Result) == false) return args.Result;
|
||
}
|
||
switch (exp.NodeType)
|
||
{
|
||
case ExpressionType.Not:
|
||
var notExp = (exp as UnaryExpression)?.Operand;
|
||
if (notExp.Type.IsNumberType()) return $"~{ExpressionLambdaToSql(notExp, tsc)}"; //位操作
|
||
if (notExp.NodeType == ExpressionType.MemberAccess)
|
||
{
|
||
var notBody = ExpressionLambdaToSql(notExp, tsc);
|
||
if (notBody.Contains(" IS NULL")) return notBody.Replace(" IS NULL", " IS NOT NULL");
|
||
if (notBody.Contains(" IS NOT NULL")) return notBody.Replace(" IS NOT NULL", " IS NULL");
|
||
if (notBody.Contains("=")) return notBody.Replace("=", "!=");
|
||
if (notBody.Contains("!=")) return notBody.Replace("!=", "=");
|
||
return $"{notBody} = {formatSql(false, null, null, null)}";
|
||
}
|
||
return $"not({ExpressionLambdaToSql(notExp, tsc)})";
|
||
case ExpressionType.Quote: return ExpressionLambdaToSql((exp as UnaryExpression)?.Operand, tsc);
|
||
case ExpressionType.Lambda: return ExpressionLambdaToSql((exp as LambdaExpression)?.Body, tsc);
|
||
//case ExpressionType.Invoke: return formatSql(Expression.Lambda(exp).Compile().DynamicInvoke(), tsc.mapType, tsc.mapColumnTmp, tsc.dbParams);
|
||
case ExpressionType.Invoke:
|
||
var invokeExp = exp as InvocationExpression;
|
||
var invokeReplaceVistor = new FreeSql.Internal.CommonExpression.ReplaceVisitor();
|
||
var invokeReplaceExp = invokeExp.Expression;
|
||
var invokeLambdaExp = invokeReplaceExp as LambdaExpression;
|
||
var len = Math.Min(invokeExp.Arguments.Count, invokeLambdaExp.Parameters.Count);
|
||
for (var a = 0; a < len; a++)
|
||
invokeReplaceExp = invokeReplaceVistor.Modify(invokeReplaceExp, invokeLambdaExp.Parameters[a], invokeExp.Arguments[a]);
|
||
return ExpressionLambdaToSql(invokeReplaceExp, tsc);
|
||
case ExpressionType.TypeAs:
|
||
case ExpressionType.Convert:
|
||
case ExpressionType.ConvertChecked:
|
||
//var othercExp = ExpressionLambdaToSqlOther(exp, tsc);
|
||
//if (string.IsNullOrEmpty(othercExp) == false) return othercExp;
|
||
var expOperand = (exp as UnaryExpression)?.Operand;
|
||
if (expOperand.Type.NullableTypeOrThis().IsEnum && exp.IsParameter() == false)
|
||
return formatSql(Expression.Lambda(exp).Compile().DynamicInvoke(), tsc.mapType, tsc.mapColumnTmp, tsc.dbParams); //bug: Where(a => a.Id = (int)enum)
|
||
return ExpressionLambdaToSql(expOperand, tsc);
|
||
case ExpressionType.Negate:
|
||
case ExpressionType.NegateChecked: return $"-({ExpressionLambdaToSql((exp as UnaryExpression)?.Operand, tsc)})";
|
||
case ExpressionType.Constant: return formatSql((exp as ConstantExpression)?.Value, tsc.mapType, tsc.mapColumnTmp, null);
|
||
case ExpressionType.Conditional:
|
||
var condExp = exp as ConditionalExpression;
|
||
var conditionalTestOldMapType = tsc.SetMapTypeReturnOld(null);
|
||
if (condExp.Test.IsParameter())
|
||
{
|
||
var condExp2 = condExp.Test;
|
||
if (condExp2.NodeType == ExpressionType.MemberAccess) condExp2 = Expression.Equal(condExp2, Expression.Constant(true));
|
||
var conditionalTestSql = ExpressionLambdaToSql(condExp2, tsc);
|
||
tsc.SetMapTypeReturnOld(conditionalTestOldMapType);
|
||
var conditionalSql = _common.IIF(conditionalTestSql, ExpressionLambdaToSql(condExp.IfTrue, tsc), ExpressionLambdaToSql(condExp.IfFalse, tsc));
|
||
tsc.SetMapTypeReturnOld(null);
|
||
return conditionalSql;
|
||
}
|
||
if ((bool)Expression.Lambda(condExp.Test).Compile().DynamicInvoke())
|
||
{
|
||
tsc.SetMapTypeReturnOld(conditionalTestOldMapType);
|
||
var conditionalSql = ExpressionLambdaToSql(condExp.IfTrue, tsc);
|
||
tsc.SetMapTypeReturnOld(null);
|
||
return conditionalSql;
|
||
}
|
||
else
|
||
{
|
||
tsc.SetMapTypeReturnOld(conditionalTestOldMapType);
|
||
var conditionalSql = ExpressionLambdaToSql(condExp.IfFalse, tsc);
|
||
tsc.SetMapTypeReturnOld(null);
|
||
return conditionalSql;
|
||
}
|
||
case ExpressionType.Call:
|
||
tsc.mapType = null;
|
||
var exp3 = exp as MethodCallExpression;
|
||
if (exp3.Object == null && (
|
||
_dicTypeExistsExpressionCallAttribute.GetOrAdd(exp3.Method.DeclaringType, dttp => dttp.GetCustomAttributes(typeof(ExpressionCallAttribute), true).Any()) ||
|
||
exp3.Method.GetCustomAttributes(typeof(ExpressionCallAttribute), true).Any()
|
||
))
|
||
{
|
||
var ecc = new ExpressionCallContext
|
||
{
|
||
_commonExp = this,
|
||
_tsc = tsc,
|
||
DataType = _ado.DataType,
|
||
UserParameters = tsc.dbParams == null ? null : new List<DbParameter>(),
|
||
FormatSql = obj => formatSql(obj, null, null, null)
|
||
};
|
||
var exp3MethodParams = exp3.Method.GetParameters();
|
||
var dbParamsIndex = tsc.dbParams?.Count;
|
||
if (exp3MethodParams.Any())
|
||
{
|
||
ecc.RawExpression.Add(exp3MethodParams[0].Name, exp3.Arguments[0]);
|
||
ecc.ParsedContent.Add(exp3MethodParams[0].Name, exp3MethodParams[0].GetCustomAttributes(typeof(RawValueAttribute), true).Any() ? null : ExpressionLambdaToSql(exp3.Arguments[0], tsc));
|
||
}
|
||
if (tsc.dbParams?.Count > dbParamsIndex) ecc.DbParameter = tsc.dbParams.Last();
|
||
List<DbParameter> oldDbParams = tsc.SetDbParamsReturnOld(null);
|
||
for (var a = 1; a < exp3.Arguments.Count; a++)
|
||
if (exp3.Arguments[a].Type != typeof(ExpressionCallContext))
|
||
{
|
||
ecc.RawExpression.Add(exp3MethodParams[a].Name, exp3.Arguments[a]);
|
||
ecc.ParsedContent.Add(exp3MethodParams[a].Name, exp3MethodParams[a].GetCustomAttributes(typeof(RawValueAttribute), true).Any() ? null : ExpressionLambdaToSql(exp3.Arguments[a], tsc));
|
||
}
|
||
tsc.SetDbParamsReturnOld(oldDbParams);
|
||
|
||
var exp3InvokeParams = new object[exp3.Arguments.Count];
|
||
for (var a = 0; a < exp3.Arguments.Count; a++)
|
||
{
|
||
if (exp3.Arguments[a].Type != typeof(ExpressionCallContext))
|
||
{
|
||
var eccContent = ecc.ParsedContent[exp3MethodParams[a].Name];
|
||
if (eccContent == null)
|
||
{
|
||
var isdyInvoke = true;
|
||
if (exp3.Arguments[a].NodeType == ExpressionType.Call) //判断如果参数也是标记 ExpressionCall
|
||
{
|
||
var exp3ArgsACallExp = exp3.Arguments[a] as MethodCallExpression;
|
||
if (exp3ArgsACallExp.Object == null && (
|
||
_dicTypeExistsExpressionCallAttribute.GetOrAdd(exp3ArgsACallExp.Method.DeclaringType, dttp => dttp.GetCustomAttributes(typeof(ExpressionCallAttribute), true).Any()) ||
|
||
exp3ArgsACallExp.Method.GetCustomAttributes(typeof(ExpressionCallAttribute), true).Any()
|
||
))
|
||
isdyInvoke = false;
|
||
}
|
||
if (isdyInvoke)
|
||
exp3InvokeParams[a] = Expression.Lambda(exp3.Arguments[a]).Compile().DynamicInvoke();
|
||
}
|
||
else if (exp3.Arguments[a].IsParameter())
|
||
exp3InvokeParams[a] = exp3.Arguments[a].Type.CreateInstanceGetDefaultValue();
|
||
|
||
else if (Utils.dicExecuteArrayRowReadClassOrTuple.ContainsKey(exp3.Arguments[a].Type.NullableTypeOrThis()) == false)
|
||
exp3InvokeParams[a] = exp3.Arguments[a].Type.CreateInstanceGetDefaultValue();
|
||
else
|
||
{
|
||
var exp3CsValue = eccContent.StartsWith("N'") ?
|
||
eccContent.Substring(1).Trim('\'').Replace("''", "'") :
|
||
eccContent.Trim('\'').Replace("''", "'");
|
||
switch (_ado.DataType)
|
||
{
|
||
case DataType.MySql:
|
||
case DataType.OdbcMySql:
|
||
exp3CsValue = exp3CsValue.Replace("\\\\", "\\");
|
||
break;
|
||
}
|
||
exp3InvokeParams[a] = Utils.GetDataReaderValue(exp3.Arguments[a].Type, exp3CsValue);
|
||
}
|
||
}
|
||
else
|
||
exp3InvokeParams[a] = ecc;
|
||
}
|
||
var eccFields = _dicTypeExpressionCallClassContextFields.GetOrAdd(exp3.Method.DeclaringType, dttp =>
|
||
dttp.GetFields(BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.GetField | BindingFlags.Static).Where(a => a.FieldType == typeof(ThreadLocal<ExpressionCallContext>)).ToArray());
|
||
if (eccFields.Any() == false)
|
||
throw new Exception(CoreStrings.Custom_Expression_ParsingError(exp3.Method.DeclaringType));
|
||
foreach (var eccField in eccFields)
|
||
typeof(ThreadLocal<ExpressionCallContext>).GetProperty("Value").SetValue(eccField.GetValue(null), ecc, null);
|
||
try
|
||
{
|
||
var invokeReturn = exp3.Method.Invoke(null, exp3InvokeParams);
|
||
if (string.IsNullOrEmpty(ecc.Result) && invokeReturn is string) ecc.Result = string.Concat(invokeReturn);
|
||
if (string.IsNullOrEmpty(ecc.Result) && exp3MethodParams.Any()) ecc.Result = ecc.ParsedContent[exp3MethodParams[0].Name];
|
||
if (ecc.UserParameters?.Any() == true) tsc.dbParams?.AddRange(ecc.UserParameters);
|
||
return ecc.Result;
|
||
}
|
||
finally
|
||
{
|
||
foreach (var eccField in eccFields)
|
||
typeof(ThreadLocal<ExpressionCallContext>).GetProperty("Value").SetValue(eccField.GetValue(null), null, null);
|
||
}
|
||
}
|
||
var callType = exp3.Object?.Type ?? exp3.Method.DeclaringType;
|
||
string other3Exp = null;
|
||
switch (callType.FullName)
|
||
{
|
||
case "System.String": other3Exp = ExpressionLambdaToSqlCallString(exp3, tsc); break;
|
||
case "System.Math": other3Exp = ExpressionLambdaToSqlCallMath(exp3, tsc); break;
|
||
case "System.DateTime": other3Exp = ExpressionLambdaToSqlCallDateTime(exp3, tsc); break;
|
||
case "System.TimeSpan": other3Exp = ExpressionLambdaToSqlCallTimeSpan(exp3, tsc); break;
|
||
case "System.Convert": other3Exp = ExpressionLambdaToSqlCallConvert(exp3, tsc); break;
|
||
}
|
||
if (string.IsNullOrEmpty(other3Exp) == false) return other3Exp;
|
||
if (exp3.Method.Name == "Equals")
|
||
{
|
||
if (exp3.Arguments.Count > 0 && exp3.Object != null) return ExpressionBinary("=", exp3.Object, exp3.Arguments[0], tsc);
|
||
if (exp3.Arguments.Count > 1 && exp3.Method.DeclaringType == typeof(object)) return ExpressionBinary("=", exp3.Arguments[0], exp3.Arguments[1], tsc);
|
||
}
|
||
if (exp3.Method.Name == "Any" && exp3.Method.DeclaringType == typeof(Enumerable))
|
||
{
|
||
//Where(a => idArray.Any(p => (a.Id == p.Key || a.RoleName == p.Key) && a.RoleType == p.Type))
|
||
var exp3MethodGenArgs = exp3.Method.GetGenericArguments();
|
||
var exp3MethodArgs = exp3.Method.GetParameters();
|
||
if (exp3MethodGenArgs.Length == 1 && exp3MethodArgs.Length == 2 && exp3MethodArgs[1].ParameterType == typeof(Func<,>).MakeGenericType(exp3MethodGenArgs[0], typeof(bool)))
|
||
{
|
||
var exp3Value = ExpressionGetValue(exp3.Arguments[0], out var exp3ValueSuccess);
|
||
if (exp3ValueSuccess)
|
||
{
|
||
if (exp3Value == null) return "1=2";
|
||
var exp3ValueIE = exp3Value as IEnumerable;
|
||
var exp3NewExpVisitor = new ReplaceParameterVisitor();
|
||
var exp3sb = new StringBuilder();
|
||
foreach (var exp3ValueItem in exp3ValueIE)
|
||
{
|
||
var exp3NewExp = exp3NewExpVisitor.Modify(exp3.Arguments[1] as LambdaExpression, Expression.Constant(exp3ValueItem, exp3MethodGenArgs[0]));
|
||
exp3sb.Append(" OR ").Append(ExpressionLambdaToSql(exp3NewExp, tsc));
|
||
}
|
||
if (exp3sb.Length == 0) return "1=2";
|
||
return exp3sb.Remove(0, 4).ToString();
|
||
}
|
||
}
|
||
}
|
||
if (callType.FullName.StartsWith("FreeSql.ISelectGroupingAggregate`"))
|
||
{
|
||
switch (exp3.Method.Name)
|
||
{
|
||
case "Count": return exp3.Arguments.Count == 0 ? "count(1)" : $"count({ExpressionLambdaToSql(exp3.Arguments[0], tsc)})";
|
||
case "Sum": return $"sum({ExpressionLambdaToSql(exp3.Arguments[0], tsc)})";
|
||
case "Avg": return $"avg({ExpressionLambdaToSql(exp3.Arguments[0], tsc)})";
|
||
case "Max": return $"max({ExpressionLambdaToSql(exp3.Arguments[0], tsc)})";
|
||
case "Min": return $"min({ExpressionLambdaToSql(exp3.Arguments[0], tsc)})";
|
||
}
|
||
}
|
||
if (callType.FullName.StartsWith("FreeSql.ISelect`"))
|
||
{ //子表查询
|
||
switch (exp3.Method.Name)
|
||
{
|
||
case "Any": //exists
|
||
case "Count":
|
||
case "Sum":
|
||
case "Min":
|
||
case "Max":
|
||
case "Avg":
|
||
case "ToList": //where in
|
||
case "ToOne":
|
||
case "First":
|
||
var anyArgs = exp3.Arguments;
|
||
var exp3Stack = new Stack<Expression>();
|
||
var exp3tmp = exp3.Object;
|
||
if (exp3.Method.Name == "Any" && exp3tmp != null && anyArgs.Any())
|
||
exp3Stack.Push(Expression.Call(exp3tmp, callType.GetMethod("Where", anyArgs.Select(a => a.Type).ToArray()), anyArgs.ToArray()));
|
||
while (exp3tmp != null)
|
||
{
|
||
exp3Stack.Push(exp3tmp);
|
||
switch (exp3tmp.NodeType)
|
||
{
|
||
case ExpressionType.Call:
|
||
var exp3tmpCall = (exp3tmp as MethodCallExpression);
|
||
exp3tmp = exp3tmpCall.Object == null ? exp3tmpCall.Arguments.FirstOrDefault() : exp3tmpCall.Object;
|
||
continue;
|
||
case ExpressionType.MemberAccess: exp3tmp = (exp3tmp as MemberExpression).Expression; continue;
|
||
}
|
||
break;
|
||
}
|
||
object fsql = null;
|
||
Expression fsqlExpLambda = null;
|
||
Select0Provider fsqlSelect0 = null;
|
||
List<SelectTableInfo> fsqltables = null;
|
||
var fsqltable1SetAlias = false;
|
||
var fsqltable1SetAliasGai = 0;
|
||
Type fsqlType = null;
|
||
Stack<Expression> asSelectBefores = new Stack<Expression>();
|
||
var asSelectSql = "";
|
||
Type asSelectEntityType = null;
|
||
MemberExpression asSelectParentExp1 = null;
|
||
Expression asSelectParentExp = null;
|
||
while (exp3Stack.Any())
|
||
{
|
||
exp3tmp = exp3Stack.Pop();
|
||
if (exp3tmp.Type.FullName.StartsWith("FreeSql.ISelect`") && fsql == null)
|
||
{
|
||
if (exp3tmp.NodeType == ExpressionType.Call)
|
||
{
|
||
var exp3tmpCall = (exp3tmp as MethodCallExpression);
|
||
if (exp3tmpCall.Method.Name == "AsSelect" && exp3tmpCall.Object == null)
|
||
{
|
||
var exp3tmpArg1Type = exp3tmpCall.Arguments.FirstOrDefault()?.Type;
|
||
if (exp3tmpArg1Type != null)
|
||
{
|
||
asSelectEntityType = exp3tmpArg1Type.GetElementType() ?? exp3tmpArg1Type.GetGenericArguments().FirstOrDefault();
|
||
if (asSelectEntityType != null)
|
||
{
|
||
fsql = _dicExpressionLambdaToSqlAsSelectMethodInfo.GetOrAdd(asSelectEntityType, asSelectEntityType2 => typeof(IFreeSql).GetMethod("Select", new Type[0]).MakeGenericMethod(asSelectEntityType2))
|
||
.Invoke(_common._orm, null);
|
||
|
||
if (asSelectBefores.Any())
|
||
{
|
||
asSelectParentExp1 = asSelectBefores.Pop() as MemberExpression;
|
||
if (asSelectBefores.Any())
|
||
{
|
||
asSelectParentExp = asSelectBefores.Pop();
|
||
if (asSelectParentExp != null)
|
||
{
|
||
var testExecuteExp = asSelectParentExp;
|
||
if (asSelectParentExp.NodeType == ExpressionType.Parameter) //执行leftjoin关联
|
||
testExecuteExp = Expression.Property(testExecuteExp, _common.GetTableByEntity(asSelectParentExp.Type).ColumnsByCs.First().Key);
|
||
var tsc2 = tsc.Clone_selectColumnMap_diymemexp_tbtype(new List<SelectColumnInfo>(), tsc.diymemexp, SelectTableInfoType.LeftJoin);
|
||
tsc2.isDisableDiyParse = true;
|
||
tsc2.style = ExpressionStyle.AsSelect;
|
||
asSelectSql = ExpressionLambdaToSql(testExecuteExp, tsc2);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
if (new[] { "Where", "WhereIf" }.Contains(exp3tmpCall.Method.Name) && exp3tmpCall.Object != null)
|
||
{
|
||
//这段特别兼容 DbSet.Where 表达式解析 #216
|
||
var exp3tmpTestCall = Expression.Call(exp3tmpCall.Object, exp3tmpCall.Method, exp3tmpCall.Arguments.Select(a =>
|
||
{
|
||
var a2 = a;
|
||
if (a2.NodeType == ExpressionType.Quote) a2 = (a as UnaryExpression)?.Operand;
|
||
if (a2?.NodeType == ExpressionType.Lambda)
|
||
{
|
||
var alambda = a2 as LambdaExpression;
|
||
if (alambda.ReturnType == typeof(bool))
|
||
return Expression.Constant(null, a.Type);// Expression.Lambda(Expression.Constant(true), alambda.Parameters);
|
||
}
|
||
return a;
|
||
//if (a.Type == typeof(Expression<>).MakeGenericType(typeof(Func<,>).MakeGenericType(exp3tmp.Type.GetGenericArguments()[0], typeof(bool))))
|
||
// return Expression.Lambda(Expression.Constant(true),
|
||
}).ToArray());
|
||
fsql = Expression.Lambda(exp3tmpTestCall).Compile().DynamicInvoke();
|
||
var fsqlFindMethod = fsql.GetType().GetMethod(exp3tmpCall.Method.Name, exp3tmpCall.Arguments.Select(a => a.Type).ToArray());
|
||
if (fsqlFindMethod == null)
|
||
throw new Exception(CoreStrings.Unable_Parse_ExpressionMethod(exp3tmpCall.Method.Name));
|
||
var exp3StackOld = exp3Stack;
|
||
exp3Stack = new Stack<Expression>();
|
||
exp3Stack.Push(Expression.Call(Expression.Constant(fsql), fsqlFindMethod, exp3tmpCall.Arguments));
|
||
while (exp3StackOld.Any()) exp3Stack.Push(exp3StackOld.Pop());
|
||
}
|
||
}
|
||
if (fsql == null)
|
||
{
|
||
fsql = Expression.Lambda(exp3tmp).Compile().DynamicInvoke();
|
||
fsqlExpLambda = exp3tmp;
|
||
}
|
||
fsqlType = fsql?.GetType();
|
||
if (fsqlType == null) break;
|
||
fsqlSelect0 = fsql as Select0Provider;
|
||
switch (exp3.Method.Name)
|
||
{
|
||
case "Any": //exists
|
||
switch (_ado.DataType)
|
||
{
|
||
case DataType.Oracle:
|
||
case DataType.OdbcOracle:
|
||
case DataType.Dameng:
|
||
case DataType.OdbcDameng:
|
||
case DataType.GBase:
|
||
break;
|
||
default:
|
||
fsqlSelect0._limit = 1; //#462 ORACLE rownum <= 2 会影响索引变慢
|
||
break;
|
||
}
|
||
break;
|
||
case "ToOne":
|
||
case "First":
|
||
fsqlSelect0._limit = 1; //#462
|
||
break;
|
||
}
|
||
if (tsc.dbParams != null) fsqlSelect0._params = tsc.dbParams;
|
||
fsqltables = fsqlSelect0._tables;
|
||
//fsqltables[0].Alias = $"{tsc._tables[0].Alias}_{fsqltables[0].Alias}";
|
||
if (fsqltables != tsc._tables)
|
||
{
|
||
if (tsc._tables == null && tsc.diymemexp == null) throw new NotSupportedException(CoreStrings.EspeciallySubquery_Cannot_Parsing); //2020-12-11 IUpdate 条件不支持子查询
|
||
if (tsc._tables != null) //groupby is null
|
||
{
|
||
fsqltables.AddRange(tsc._tables.Select(a => new SelectTableInfo
|
||
{
|
||
Alias = a.Alias,
|
||
On = "1=1",
|
||
Table = a.Table,
|
||
Type = SelectTableInfoType.Parent,
|
||
Parameter = a.Parameter
|
||
}));
|
||
}
|
||
}
|
||
if (tsc.whereGlobalFilter != null)
|
||
{
|
||
fsqlSelect0._whereGlobalFilter.Clear();
|
||
if (tsc.whereGlobalFilter.Any()) fsqlSelect0._whereGlobalFilter.AddRange(tsc.whereGlobalFilter);
|
||
}
|
||
}
|
||
else if (fsqlType != null)
|
||
{
|
||
var call3Exp = exp3tmp as MethodCallExpression;
|
||
var method = call3Exp.Method;
|
||
//var method = fsqlType.GetMethod(call3Exp.Method.Name, call3Exp.Arguments.Select(a => a.Type).ToArray());
|
||
//if (call3Exp.Method.ContainsGenericParameters) method.MakeGenericMethod(call3Exp.Method.GetGenericArguments());
|
||
var parms = method.GetParameters();
|
||
var args = new object[call3Exp.Arguments.Count];
|
||
for (var a = 0; a < args.Length; a++)
|
||
{
|
||
var arg3Exp = call3Exp.Arguments[a];
|
||
if (arg3Exp.NodeType == ExpressionType.Constant)
|
||
{
|
||
args[a] = (arg3Exp as ConstantExpression)?.Value;
|
||
}
|
||
else if (arg3Exp == fsqlExpLambda)
|
||
{
|
||
args[a] = fsql;
|
||
}
|
||
else
|
||
{
|
||
var argExp = (arg3Exp as UnaryExpression)?.Operand;
|
||
if (argExp != null)
|
||
{
|
||
if (argExp.NodeType == ExpressionType.Lambda)
|
||
{
|
||
if (fsqltable1SetAlias == false)
|
||
{
|
||
fsqltable1SetAlias = true;
|
||
var argExpLambda = argExp as LambdaExpression;
|
||
var fsqlTypeGenericArgs = fsqlType.GetGenericArguments();
|
||
|
||
if (argExpLambda.Parameters.Count == 1 && argExpLambda.Parameters[0].Type.FullName.StartsWith("FreeSql.Internal.Model.HzyTuple`"))
|
||
{
|
||
for (; fsqltable1SetAliasGai < fsqlTypeGenericArgs.Length; fsqltable1SetAliasGai++)
|
||
fsqltables[fsqltable1SetAliasGai].Alias = "ht" + (fsqltable1SetAliasGai + 1);
|
||
}
|
||
else
|
||
{
|
||
for (; fsqltable1SetAliasGai < fsqlTypeGenericArgs.Length && fsqltable1SetAliasGai < argExpLambda.Parameters.Count; fsqltable1SetAliasGai++)
|
||
{
|
||
var alias = argExpLambda.Parameters[fsqltable1SetAliasGai].Name;
|
||
if (fsqltables.Any(x => x.Type == SelectTableInfoType.Parent && x.Alias == alias)) alias = $"sub_{alias}";
|
||
fsqltables[fsqltable1SetAliasGai].Alias = alias;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
else
|
||
{
|
||
argExp = null;
|
||
}
|
||
}
|
||
args[a] = argExp ?? Expression.Lambda(arg3Exp).Compile().DynamicInvoke();
|
||
//if (args[a] == null) ExpressionLambdaToSql(call3Exp.Arguments[a], fsqltables, null, null, SelectTableInfoType.From, true);
|
||
}
|
||
}
|
||
var isSubSelectPdme = tsc._tables == null && tsc.diymemexp != null || tsc.diymemexp is Select0Provider.WithTempQueryParser;
|
||
try
|
||
{
|
||
if (isSubSelectPdme)
|
||
{
|
||
if (_subSelectParentDiyMemExps.Value == null) _subSelectParentDiyMemExps.Value = new List<BaseDiyMemberExpression>();
|
||
_subSelectParentDiyMemExps.Value.Add(tsc.diymemexp);
|
||
}
|
||
switch (method.Name)
|
||
{
|
||
case nameof(ISelect<object>.From):
|
||
case nameof(ISelect<object>.FromQuery):
|
||
case nameof(ISelect<object>.WithTempQuery):
|
||
fsql = method.Invoke(fsql, args);
|
||
fsqlType = fsql.GetType();
|
||
fsqlSelect0 = fsql as Select0Provider;
|
||
if (tsc.dbParams != null) fsqlSelect0._params = tsc.dbParams;
|
||
fsqltables = fsqlSelect0._tables;
|
||
fsqltable1SetAlias = false;
|
||
if (method.Name == nameof(ISelect<object>.WithTempQuery)) fsqltable1SetAliasGai = 0;
|
||
break;
|
||
default:
|
||
method.Invoke(fsql, args);
|
||
break;
|
||
}
|
||
}
|
||
finally
|
||
{
|
||
if (isSubSelectPdme)
|
||
{
|
||
var psgpdmes = _subSelectParentDiyMemExps.Value;
|
||
if (psgpdmes != null)
|
||
{
|
||
psgpdmes.RemoveAt(psgpdmes.Count - 1);
|
||
if (psgpdmes.Count == 0) _subSelectParentDiyMemExps.Value = null;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
if (fsql == null) asSelectBefores.Push(exp3tmp);
|
||
}
|
||
if (fsql != null)
|
||
{
|
||
if (fsqlSelect0 != null && tsc._tableRule != null && fsqlSelect0._tableRules.Any() == false)
|
||
fsqlSelect0._tableRules.Add(tsc._tableRule);
|
||
|
||
if (asSelectParentExp != null)
|
||
{
|
||
//执行 AsSelect() 的关联,OneToMany,ManyToMany,PgArrayToMany
|
||
if (fsqltables[0].Parameter == null)
|
||
{
|
||
fsqltables[0].Alias = $"tb_{fsqltables.Count}";
|
||
fsqltables[0].Parameter = Expression.Parameter(asSelectEntityType, fsqltables[0].Alias);
|
||
}
|
||
|
||
var parm123Tb = _common.GetTableByEntity(asSelectParentExp.Type);
|
||
var parm123Ref = parm123Tb.GetTableRef(asSelectParentExp1.Member.Name, true);
|
||
if (parm123Ref != null)
|
||
{
|
||
if (parm123Ref.RefType == TableRefType.PgArrayToMany)
|
||
{
|
||
var amtReftbname = ExpressionLambdaToSql(Expression.MakeMemberAccess(asSelectParentExp, parm123Tb.Properties[parm123Tb.ColumnsByPosition[0].CsName]), tsc);
|
||
amtReftbname = amtReftbname.Substring(0, amtReftbname.Length - _common.QuoteSqlName(parm123Tb.ColumnsByPosition[0].Attribute.Name).Length - 1);
|
||
if (parm123Ref.RefColumns[0] == fsqltables[0].Table.Primarys[0])
|
||
{
|
||
var dbinfo = _common._orm.CodeFirst.GetDbInfo(parm123Ref.Columns[0].CsType);
|
||
(fsql as Select0Provider)._where.Append(" AND (").Append($"{amtReftbname}.{_common.QuoteSqlName(parm123Ref.Columns[0].Attribute.Name)} @> ARRAY[{fsqltables[0].Alias}.{_common.QuoteSqlName(parm123Ref.RefColumns[0].Attribute.Name)}]::{dbinfo?.dbtype}").Append(")");
|
||
}
|
||
else if (parm123Ref.Columns[0] == parm123Tb.Primarys[0])
|
||
{
|
||
var dbinfo = _common._orm.CodeFirst.GetDbInfo(parm123Ref.RefColumns[0].CsType);
|
||
(fsql as Select0Provider)._where.Append(" AND (").Append($"{fsqltables[0].Alias}.{_common.QuoteSqlName(parm123Ref.RefColumns[0].Attribute.Name)} @> ARRAY[{amtReftbname}.{_common.QuoteSqlName(parm123Ref.Columns[0].Attribute.Name)}]::{dbinfo?.dbtype}").Append(")");
|
||
}
|
||
else
|
||
{
|
||
;
|
||
}
|
||
}
|
||
else
|
||
{
|
||
var fsqlWhere = _dicExpressionLambdaToSqlAsSelectWhereMethodInfo.GetOrAdd(asSelectEntityType, asSelectEntityType3 =>
|
||
typeof(ISelect<>).MakeGenericType(asSelectEntityType3).GetMethod("Where", new[] {
|
||
typeof(Expression<>).MakeGenericType(typeof(Func<,>).MakeGenericType(asSelectEntityType3, typeof(bool)))
|
||
}));
|
||
var fsqlWhereParam = fsqltables.First().Parameter; //Expression.Parameter(asSelectEntityType);
|
||
Expression fsqlWhereExp = null;
|
||
if (parm123Ref.RefType == TableRefType.ManyToMany)
|
||
{
|
||
//g.mysql.Select<Tag>().Where(a => g.mysql.Select<Song_tag>().Where(b => b.Tag_id == a.Id && b.Song_id == 1).Any());
|
||
var manyTb = _common.GetTableByEntity(parm123Ref.RefMiddleEntityType);
|
||
var manySubSelectWhere = _dicExpressionLambdaToSqlAsSelectWhereMethodInfo.GetOrAdd(parm123Ref.RefMiddleEntityType, refMiddleEntityType3 =>
|
||
typeof(ISelect<>).MakeGenericType(refMiddleEntityType3).GetMethod("Where", new[] {
|
||
typeof(Expression<>).MakeGenericType(typeof(Func<,>).MakeGenericType(refMiddleEntityType3, typeof(bool)))
|
||
}));
|
||
var manySubSelectWhereSql = _dicExpressionLambdaToSqlAsSelectWhereSqlMethodInfo.GetOrAdd(parm123Ref.RefMiddleEntityType, refMiddleEntityType3 =>
|
||
typeof(ISelect0<,>).MakeGenericType(typeof(ISelect<>).MakeGenericType(refMiddleEntityType3), refMiddleEntityType3).GetMethod("Where", new[] { typeof(string), typeof(object) }));
|
||
var manySubSelectAsSelectExp = _dicFreeSqlGlobalExtensionsAsSelectExpression.GetOrAdd(parm123Ref.RefMiddleEntityType, refMiddleEntityType3 =>
|
||
Expression.Call(
|
||
typeof(FreeSqlGlobalExtensions).GetMethods(BindingFlags.Static | BindingFlags.Public).Where(mfil => mfil.Name == "AsSelect" && mfil.GetParameters().Length == 1).FirstOrDefault()?.MakeGenericMethod(refMiddleEntityType3),
|
||
Expression.Constant(Activator.CreateInstance(typeof(List<>).MakeGenericType(refMiddleEntityType3)))
|
||
));
|
||
var manyMainParam = tsc._tables[0].Parameter;
|
||
var manySubSelectWhereParam = Expression.Parameter(parm123Ref.RefMiddleEntityType, $"M{fsqlWhereParam.Name}_M{asSelectParentExp.ToString().Replace(".", "__")}");//, $"{fsqlWhereParam.Name}__");
|
||
Expression manySubSelectWhereExp = null;
|
||
for (var mn = 0; mn < parm123Ref.Columns.Count; mn++)
|
||
{
|
||
var col1 = parm123Ref.MiddleColumns[mn];
|
||
var col2 = parm123Ref.Columns[mn];
|
||
var pexp1 = Expression.Property(manySubSelectWhereParam, col1.CsName);
|
||
var pexp2 = Expression.Property(asSelectParentExp, col2.CsName);
|
||
if (col1.CsType != col2.CsType)
|
||
{
|
||
if (col1.CsType.IsNullableType()) pexp1 = Expression.Property(pexp1, _dicNullableValueProperty.GetOrAdd(col1.CsType, ct1 => ct1.GetProperty("Value")));
|
||
if (col2.CsType.IsNullableType()) pexp2 = Expression.Property(pexp2, _dicNullableValueProperty.GetOrAdd(col2.CsType, ct2 => ct2.GetProperty("Value")));
|
||
}
|
||
var tmpExp = Expression.Equal(pexp1, pexp2);
|
||
if (mn == 0) manySubSelectWhereExp = tmpExp;
|
||
else manySubSelectWhereExp = Expression.AndAlso(manySubSelectWhereExp, tmpExp);
|
||
}
|
||
var manySubSelectExpBoy = Expression.Call(
|
||
manySubSelectAsSelectExp,
|
||
manySubSelectWhere,
|
||
Expression.Lambda(
|
||
manySubSelectWhereExp,
|
||
manySubSelectWhereParam
|
||
)
|
||
);
|
||
Expression fsqlManyWhereExp = null;
|
||
for (var mn = 0; mn < parm123Ref.RefColumns.Count; mn++)
|
||
{
|
||
var col1 = parm123Ref.RefColumns[mn];
|
||
var col2 = parm123Ref.MiddleColumns[mn + parm123Ref.Columns.Count + mn];
|
||
var pexp1 = Expression.Property(fsqlWhereParam, col1.CsName);
|
||
var pexp2 = Expression.Property(manySubSelectWhereParam, col2.CsName);
|
||
if (col1.CsType != col2.CsType)
|
||
{
|
||
if (col1.CsType.IsNullableType()) pexp1 = Expression.Property(pexp1, _dicNullableValueProperty.GetOrAdd(col1.CsType, ct1 => ct1.GetProperty("Value")));
|
||
if (col2.CsType.IsNullableType()) pexp2 = Expression.Property(pexp2, _dicNullableValueProperty.GetOrAdd(col2.CsType, ct2 => ct2.GetProperty("Value")));
|
||
}
|
||
var tmpExp = Expression.Equal(pexp1, pexp2);
|
||
if (mn == 0) fsqlManyWhereExp = tmpExp;
|
||
else fsqlManyWhereExp = Expression.AndAlso(fsqlManyWhereExp, tmpExp);
|
||
}
|
||
MethodInfo manySubSelectAggMethod = null;
|
||
switch (exp3.Method.Name) //https://github.com/dotnetcore/FreeSql/issues/362
|
||
{
|
||
case "Any":
|
||
case "Count":
|
||
fsqltables.Add(new SelectTableInfo { Alias = manySubSelectWhereParam.Name, Parameter = manySubSelectWhereParam, Table = manyTb, Type = SelectTableInfoType.Parent });
|
||
fsqlWhere.Invoke(fsql, new object[] { Expression.Lambda(fsqlManyWhereExp, fsqlWhereParam) });
|
||
var sql2 = fsqlType.GetMethod("ToSql", new Type[] { typeof(string) })?.Invoke(fsql, new object[] { "1" })?.ToString();
|
||
if (string.IsNullOrEmpty(sql2) == false)
|
||
manySubSelectExpBoy = Expression.Call(manySubSelectExpBoy, manySubSelectWhereSql, Expression.Constant($"exists({sql2.Replace(" \r\n", " \r\n ")})"), Expression.Constant(null));
|
||
manySubSelectAggMethod = _dicExpressionLambdaToSqlAsSelectAggMethodInfo.GetOrAdd(parm123Ref.RefMiddleEntityType, _ => new ConcurrentDictionary<string, MethodInfo>()).GetOrAdd(exp3.Method.Name, exp3MethodName =>
|
||
typeof(ISelect0<,>).MakeGenericType(typeof(ISelect<>).MakeGenericType(parm123Ref.RefMiddleEntityType), parm123Ref.RefMiddleEntityType).GetMethod(exp3MethodName, new Type[0]));
|
||
manySubSelectExpBoy = Expression.Call(manySubSelectExpBoy, manySubSelectAggMethod);
|
||
break;
|
||
case "Sum":
|
||
case "Min":
|
||
case "Max":
|
||
case "Avg":
|
||
case "ToList":
|
||
case "ToOne":
|
||
case "First":
|
||
//解析:string.Join(",", w.Roles.AsSelect().ToList(b => b.RoleName)
|
||
var exp3Args0 = (exp3.Arguments[0] as UnaryExpression)?.Operand as LambdaExpression;
|
||
manySubSelectAggMethod = _dicSelectMethodToSql.GetOrAdd(fsqlType, fsqlType2 =>
|
||
fsqlType2.GetMethods().Where(a => a.Name == "ToSql" && a.GetParameters().Length == 2 && a.GetParameters()[1].ParameterType == typeof(FieldAliasOptions) && a.GetGenericArguments().Length == 1).FirstOrDefault());
|
||
if (manySubSelectAggMethod == null || exp3Args0 == null) throw new ArgumentException(CoreStrings.ManyToMany_AsSelect_NotSupport_Sum_Avg_etc);
|
||
manySubSelectAggMethod = manySubSelectAggMethod.MakeGenericMethod(exp3Args0.ReturnType);
|
||
var fsqls0p = fsql as Select0Provider;
|
||
var fsqls0pWhere = fsqls0p._where.ToString();
|
||
fsqls0p._where.Clear();
|
||
var fsqltablesLast = new SelectTableInfo { Alias = manySubSelectWhereParam.Name, Parameter = manySubSelectWhereParam, Table = manyTb, Type = SelectTableInfoType.InnerJoin };
|
||
fsqltables.Add(fsqltablesLast);
|
||
fsqlWhere.Invoke(fsql, new object[] { Expression.Lambda(fsqlManyWhereExp, fsqlWhereParam) });
|
||
fsqltablesLast.NavigateCondition = fsqls0p._where.ToString();
|
||
if (fsqltablesLast.NavigateCondition.StartsWith(" AND (")) fsqltablesLast.NavigateCondition = fsqltablesLast.NavigateCondition.Substring(6, fsqltablesLast.NavigateCondition.Length - 7);
|
||
fsqls0p._where.Clear().Append(fsqls0pWhere);
|
||
var tsc3 = tsc.CloneDisableDiyParse();
|
||
tsc3._tables = tsc._tables.ToList();
|
||
var where2 = ExpressionLambdaToSql(Expression.Lambda(manySubSelectWhereExp, manySubSelectWhereParam), tsc3);
|
||
if (string.IsNullOrEmpty(where2) == false) fsqls0p._where.Append(" AND (").Append(where2).Append(")");
|
||
|
||
switch (exp3.Method.Name)
|
||
{
|
||
case "Sum":
|
||
case "Min":
|
||
case "Max":
|
||
case "Avg":
|
||
var map = new ReadAnonymousTypeInfo();
|
||
var field = new StringBuilder();
|
||
var index = -1;
|
||
|
||
for (var a = 0; a < exp3Args0.Parameters.Count; a++) fsqls0p._tables[a].Parameter = exp3Args0.Parameters[a];
|
||
ReadAnonymousField(fsqls0p._tables, fsqls0p._tableRule, field, map, ref index, exp3Args0, null, null, null, null, null, false);
|
||
var fieldSql = field.Length > 0 ? field.Remove(0, 2).ToString() : null;
|
||
|
||
var sql4 = fsqlType.GetMethod("ToSql", new Type[] { typeof(string) })?.Invoke(fsql, new object[] { $"{exp3.Method.Name.ToLower()}({fieldSql})" })?.ToString();
|
||
asSelectBefores.Clear();
|
||
return _common.IsNull($"({sql4.Replace(" \r\n", " \r\n ")})", formatSql(exp3.Method.ReturnType.CreateInstanceGetDefaultValue(), exp3.Method.ReturnType, null, null));
|
||
}
|
||
|
||
var sql3 = manySubSelectAggMethod.Invoke(fsql, new object[] { exp3Args0, FieldAliasOptions.AsProperty }) as string;
|
||
asSelectBefores.Clear();
|
||
return $"({sql3.Replace(" \r\n", " \r\n ")})";
|
||
}
|
||
asSelectBefores.Clear();
|
||
var tscwhereGlobalFilter = tsc.whereGlobalFilter;
|
||
try
|
||
{
|
||
tsc.whereGlobalFilter = fsqlSelect0._whereGlobalFilter; //ManyToMany 中间表过滤器
|
||
return ExpressionLambdaToSql(manySubSelectExpBoy, tsc);
|
||
}
|
||
finally
|
||
{
|
||
tsc.whereGlobalFilter = tscwhereGlobalFilter;
|
||
}
|
||
}
|
||
for (var mn = 0; mn < parm123Ref.Columns.Count; mn++)
|
||
{
|
||
var col1 = parm123Ref.RefColumns[mn];
|
||
var col2 = parm123Ref.Columns[mn];
|
||
var pexp1 = Expression.Property(fsqlWhereParam, col1.CsName);
|
||
var pexp2 = Expression.Property(asSelectParentExp, col2.CsName);
|
||
if (col1.CsType != col2.CsType)
|
||
{
|
||
if (col1.CsType.IsNullableType()) pexp1 = Expression.Property(pexp1, _dicNullableValueProperty.GetOrAdd(col1.CsType, ct1 => ct1.GetProperty("Value")));
|
||
if (col2.CsType.IsNullableType()) pexp2 = Expression.Property(pexp2, _dicNullableValueProperty.GetOrAdd(col2.CsType, ct2 => ct2.GetProperty("Value")));
|
||
}
|
||
var tmpExp = Expression.Equal(pexp1, pexp2);
|
||
if (mn == 0) fsqlWhereExp = tmpExp;
|
||
else fsqlWhereExp = Expression.AndAlso(fsqlWhereExp, tmpExp);
|
||
}
|
||
fsqlWhere.Invoke(fsql, new object[] { Expression.Lambda(fsqlWhereExp, fsqlWhereParam) });
|
||
}
|
||
}
|
||
}
|
||
asSelectBefores.Clear();
|
||
|
||
switch (exp3.Method.Name)
|
||
{
|
||
case "Any":
|
||
var sql = fsqlType.GetMethod("ToSql", new Type[] { typeof(string) })?.Invoke(fsql, new object[] { "1" })?.ToString();
|
||
if (string.IsNullOrEmpty(sql) == false)
|
||
return $"exists({sql.Replace(" \r\n", " \r\n ")})";
|
||
break;
|
||
case "Count":
|
||
var sqlCount = fsqlType.GetMethod("ToSql", new Type[] { typeof(string) })?.Invoke(fsql, new object[] { "count(1)" })?.ToString();
|
||
if (string.IsNullOrEmpty(sqlCount) == false)
|
||
return $"({sqlCount.Replace(" \r\n", " \r\n ")})";
|
||
break;
|
||
case "Sum":
|
||
case "Min":
|
||
case "Max":
|
||
case "Avg":
|
||
var tscClone1 = tsc.CloneDisableDiyParse();
|
||
tscClone1.subSelect001 = fsql as Select0Provider; //#405 Oracle within group(order by ..)
|
||
tscClone1.isDisableDiyParse = false;
|
||
tscClone1._tables = fsqltables;
|
||
var exp3Args0 = (exp3.Arguments.FirstOrDefault() as UnaryExpression)?.Operand as LambdaExpression;
|
||
if (exp3Args0.Parameters.Count == 1 && exp3Args0.Parameters[0].Type.FullName.StartsWith("FreeSql.Internal.Model.HzyTuple`"))
|
||
exp3Args0 = new ReplaceHzyTupleToMultiParam().Modify(exp3Args0, fsqltables);
|
||
var sqlSumField = $"{exp3.Method.Name.ToLower()}({ExpressionLambdaToSql(exp3Args0, tscClone1)})";
|
||
var sqlSum = tscClone1.subSelect001._limit <= 0 && tscClone1.subSelect001._skip <= 0 ?
|
||
fsqlType.GetMethod("ToSql", new Type[] { typeof(string) })?.Invoke(fsql, new object[] { $"{exp3.Method.Name.ToLower()}({ExpressionLambdaToSql(exp3Args0, tscClone1)})" })?.ToString() :
|
||
tscClone1.subSelect001.GetNestSelectSql(exp3Args0, sqlSumField, tosqlField =>
|
||
fsqlType.GetMethod("ToSql", new Type[] { typeof(string) })?.Invoke(fsql, new object[] { tosqlField })?.ToString());
|
||
if (string.IsNullOrEmpty(sqlSum) == false)
|
||
return tscClone1.subSelect001._limit <= 0 && tscClone1.subSelect001._skip <= 0 ?
|
||
_common.IsNull($"({sqlSum.Replace(" \r\n", " \r\n ")})", formatSql(exp3.Method.ReturnType.CreateInstanceGetDefaultValue(), exp3.Method.ReturnType, null, null)) :
|
||
_common.IsNull($"({sqlSum})", formatSql(exp3.Method.ReturnType.CreateInstanceGetDefaultValue(), exp3.Method.ReturnType, null, null));
|
||
break;
|
||
case "ToList":
|
||
case "ToOne":
|
||
case "First":
|
||
var tscClone2 = tsc.CloneDisableDiyParse();
|
||
var fsqlSelect0p = fsql as Select0Provider;
|
||
tscClone2.subSelect001 = fsqlSelect0p; //#405 Oracle within group(order by ..)
|
||
tscClone2.isDisableDiyParse = false;
|
||
tscClone2._tables = fsqltables;
|
||
var exp3Args02 = (exp3.Arguments.FirstOrDefault() as UnaryExpression)?.Operand as LambdaExpression;
|
||
if (exp3Args02.Parameters.Count == 1 && exp3Args02.Parameters[0].Type.FullName.StartsWith("FreeSql.Internal.Model.HzyTuple`"))
|
||
exp3Args02 = new ReplaceHzyTupleToMultiParam().Modify(exp3Args02, fsqltables);
|
||
var sqlFirstField = ExpressionLambdaToSql(exp3Args02, tscClone2);
|
||
var sqlFirst = fsqlType.GetMethod("ToSql", new Type[] { typeof(string) })?.Invoke(fsql, new object[] { sqlFirstField })?.ToString();
|
||
if (string.IsNullOrEmpty(sqlFirst) == false)
|
||
{
|
||
if (fsqlSelect0p._limit > 0)
|
||
{
|
||
switch (_ado.DataType) //使用 Limit 后的 IN 子查询需要套一层
|
||
{
|
||
case DataType.MySql:
|
||
case DataType.OdbcMySql:
|
||
case DataType.GBase:
|
||
if (exp3.Method.Name == "ToList")
|
||
return $"( SELECT * FROM ({sqlFirst.Replace(" \r\n", " \r\n ")}) ftblmt50 )";
|
||
break;
|
||
}
|
||
}
|
||
return $"({sqlFirst.Replace(" \r\n", " \r\n ")})";
|
||
}
|
||
break;
|
||
}
|
||
}
|
||
asSelectBefores.Clear();
|
||
break;
|
||
}
|
||
}
|
||
other3Exp = ExpressionLambdaToSqlOther(exp3, tsc);
|
||
if (string.IsNullOrEmpty(other3Exp) == false) return other3Exp;
|
||
if (exp3.IsParameter() == false) return formatSql(Expression.Lambda(exp3).Compile().DynamicInvoke(), tsc.mapType, tsc.mapColumnTmp, tsc.dbParams);
|
||
if (exp3.Method.DeclaringType == typeof(Enumerable)) throw new Exception(CoreStrings.Not_Implemented_Expression_UseAsSelect(exp3, exp3.Method.Name, (exp3.Arguments.Count > 1 ? "..." : "")));
|
||
throw new Exception(CoreStrings.Not_Implemented_Expression(exp3));
|
||
case ExpressionType.Parameter:
|
||
case ExpressionType.MemberAccess:
|
||
var exp4 = exp as MemberExpression;
|
||
if (exp4 != null)
|
||
{
|
||
if (exp4.Expression != null && exp4.Expression.Type.IsArray == false && exp4.Expression.Type.IsNullableType())
|
||
return exp4.Member.Name == "HasValue" ? $"{ExpressionLambdaToSql(exp4.Expression, tsc)} IS NOT NULL" : ExpressionLambdaToSql(exp4.Expression, tsc);
|
||
var extRet = "";
|
||
var memberType = exp4.Expression?.Type ?? exp4.Type;
|
||
switch (memberType.FullName)
|
||
{
|
||
case "System.String": extRet = ExpressionLambdaToSqlMemberAccessString(exp4, tsc); break;
|
||
case "System.DateTime": extRet = ExpressionLambdaToSqlMemberAccessDateTime(exp4, tsc); break;
|
||
case "System.TimeSpan": extRet = ExpressionLambdaToSqlMemberAccessTimeSpan(exp4, tsc); break;
|
||
}
|
||
if (string.IsNullOrEmpty(extRet) == false) return extRet;
|
||
var other4Exp = ExpressionLambdaToSqlOther(exp4, tsc);
|
||
if (string.IsNullOrEmpty(other4Exp) == false) return other4Exp;
|
||
}
|
||
var expStack = new Stack<Expression>();
|
||
var expStackConstOrMemberCount = 1;
|
||
expStack.Push(exp);
|
||
MethodCallExpression callExp = null;
|
||
var exp2 = exp4?.Expression;
|
||
while (true)
|
||
{
|
||
switch (exp2?.NodeType)
|
||
{
|
||
case ExpressionType.Constant:
|
||
expStack.Push(exp2);
|
||
expStackConstOrMemberCount++;
|
||
break;
|
||
case ExpressionType.Parameter:
|
||
expStack.Push(exp2);
|
||
break;
|
||
case ExpressionType.MemberAccess:
|
||
expStack.Push(exp2);
|
||
exp2 = (exp2 as MemberExpression).Expression;
|
||
expStackConstOrMemberCount++;
|
||
if (exp2 == null) break;
|
||
continue;
|
||
case ExpressionType.Call:
|
||
callExp = exp2 as MethodCallExpression;
|
||
expStack.Push(exp2);
|
||
exp2 = callExp.Object;
|
||
if (exp2 == null) break;
|
||
continue;
|
||
case ExpressionType.TypeAs:
|
||
case ExpressionType.Convert:
|
||
var oper2 = (exp2 as UnaryExpression).Operand;
|
||
if (oper2.NodeType == ExpressionType.Parameter)
|
||
{
|
||
var oper2Parm = oper2 as ParameterExpression;
|
||
if (exp2.Type.IsAbstract || exp2.Type.IsInterface || exp2.Type.IsAssignableFrom(oper2Parm.Type))
|
||
expStack.Push(oper2Parm);
|
||
else if (oper2Parm.Type != typeof(object) && oper2Parm.Type.IsAssignableFrom(exp2.Type))
|
||
expStack.Push(oper2Parm);
|
||
else
|
||
expStack.Push(Expression.Parameter(exp2.Type, oper2Parm.Name));
|
||
}
|
||
else
|
||
expStack.Push(oper2);
|
||
break;
|
||
}
|
||
break;
|
||
}
|
||
if (expStack.First().NodeType != ExpressionType.Parameter)
|
||
{
|
||
if (expStackConstOrMemberCount == expStack.Count)
|
||
{
|
||
object firstValue = null;
|
||
switch (expStack.First().NodeType)
|
||
{
|
||
case ExpressionType.Constant:
|
||
var expStackFirst = expStack.Pop() as ConstantExpression;
|
||
firstValue = expStackFirst?.Value;
|
||
break;
|
||
case ExpressionType.MemberAccess:
|
||
var expStackFirstMem = expStack.First() as MemberExpression;
|
||
if (expStackFirstMem.Expression?.NodeType == ExpressionType.Constant)
|
||
firstValue = (expStackFirstMem.Expression as ConstantExpression)?.Value;
|
||
else
|
||
return formatSql(Expression.Lambda(exp).Compile().DynamicInvoke(), tsc.mapType, tsc.mapColumnTmp, tsc.dbParams);
|
||
break;
|
||
}
|
||
while (expStack.Any())
|
||
{
|
||
if (firstValue == null) throw new Exception(CoreStrings.Cannot_Be_NULL_Name(exp));
|
||
var expStackItem = expStack.Pop() as MemberExpression;
|
||
if (expStackItem.Member.MemberType == MemberTypes.Property)
|
||
firstValue = ((PropertyInfo)expStackItem.Member).GetValue(firstValue, null);
|
||
else if (expStackItem.Member.MemberType == MemberTypes.Field)
|
||
firstValue = ((FieldInfo)expStackItem.Member).GetValue(firstValue);
|
||
}
|
||
return formatSql(firstValue, tsc.mapType, tsc.mapColumnTmp, tsc.dbParams);
|
||
}
|
||
return formatSql(Expression.Lambda(exp).Compile().DynamicInvoke(), tsc.mapType, tsc.mapColumnTmp, tsc.dbParams);
|
||
}
|
||
if (callExp != null) return ExpressionLambdaToSql(callExp, tsc);
|
||
var diymemexps = new[] { tsc.diymemexp, tsc.subSelect001?._diymemexpWithTempQuery };
|
||
if (_subSelectParentDiyMemExps.Value?.Any() == true)
|
||
diymemexps = diymemexps.Concat(_subSelectParentDiyMemExps.Value).ToArray();
|
||
foreach (var diymemexp in diymemexps)
|
||
{
|
||
if (diymemexp != null)
|
||
{
|
||
var expStackFirst = expStack.First() as ParameterExpression;
|
||
var bidx = expStackFirst.Type.FullName.StartsWith("FreeSql.ISelectGroupingAggregate`") ? 2 : 1; //.Key .Value
|
||
var diyexpMembers = expStack.Where((a, b) => b >= bidx).ToArray();
|
||
if (diyexpMembers.Any() == false && diymemexp != null && diymemexp is Select0Provider.WithTempQueryParser tempQueryParser && tempQueryParser.GetOutsideSelectTable(expStackFirst) != null)
|
||
diyexpMembers = expStack.ToArray();
|
||
var diyexpResult = diymemexp.ParseExp(diyexpMembers);
|
||
if (string.IsNullOrEmpty(diyexpResult) == false) return diyexpResult;
|
||
}
|
||
}
|
||
var psgpdymes = _subSelectParentDiyMemExps.Value; //解决:分组之后的子查询解析
|
||
if (psgpdymes?.Any() == true)
|
||
{
|
||
var expStackFirst = expStack.First();
|
||
if (expStackFirst.NodeType == ExpressionType.Parameter)
|
||
{
|
||
var diyexpResult = psgpdymes.Where(a => a._lambdaParameter == expStackFirst).FirstOrDefault()?.ParseExp(expStack.Where((a, b) => b >= 2).ToArray());
|
||
if (string.IsNullOrEmpty(diyexpResult) == false) return diyexpResult;
|
||
}
|
||
}
|
||
|
||
if (tsc._tables == null)
|
||
{
|
||
var pp = expStack.Pop() as ParameterExpression;
|
||
var memberExp = expStack.Pop() as MemberExpression;
|
||
var tb = _common.GetTableByEntity(pp.Type);
|
||
if (tb.ColumnsByCs.ContainsKey(memberExp.Member.Name) == false)
|
||
{
|
||
if (tb.ColumnsByCsIgnore.ContainsKey(memberExp.Member.Name))
|
||
throw new ArgumentException(CoreStrings.Ignored_Check_Confirm_PublicGetSet(tb.DbName, memberExp.Member.Name));
|
||
throw new ArgumentException(CoreStrings.NotFound_Column(tb.DbName, memberExp.Member.Name));
|
||
}
|
||
var curcol = tb.ColumnsByCs[memberExp.Member.Name];
|
||
if (tsc._selectColumnMap != null)
|
||
tsc._selectColumnMap.Add(new SelectColumnInfo { Table = null, Column = curcol });
|
||
var name = curcol.Attribute.Name;
|
||
if (tsc.isQuoteName) name = _common.QuoteSqlName(name);
|
||
tsc.SetMapColumnTmp(curcol);
|
||
if (string.IsNullOrEmpty(tsc.alias001)) return name;
|
||
return $"{tsc.alias001}.{name}";
|
||
}
|
||
Func<TableInfo, string, bool, ParameterExpression, MemberExpression, SelectTableInfo> getOrAddTable = (tbtmp, alias, isa, parmExp, mp) =>
|
||
{
|
||
var finds = new SelectTableInfo[0];
|
||
if (tsc.style == ExpressionStyle.SelectColumns)
|
||
{
|
||
finds = tsc._tables.Where(a => a.Table.Type == tbtmp.Type && a.Alias == alias).ToArray();
|
||
if (finds.Any() == false && alias.Contains("__") == false)
|
||
finds = tsc._tables.Where(a => a.Table.Type == tbtmp.Type).ToArray();
|
||
if (finds.Any()) finds = new[] { finds.First() };
|
||
}
|
||
if (finds.Length != 1 && isa && parmExp != null)
|
||
finds = tsc._tables.Where(a => a.Parameter == parmExp).ToArray();
|
||
if (finds.Length != 1)
|
||
{
|
||
var navdot = string.IsNullOrEmpty(alias) ? new SelectTableInfo[0] : tsc._tables.Where(a2 => a2.Parameter != null && alias.StartsWith($"{a2.Alias}__")).ToArray();
|
||
if (navdot.Length > 0)
|
||
{
|
||
var isthis = navdot[0] == tsc._tables[0];
|
||
finds = tsc._tables.Where(a2 => (isa && a2.Parameter != null || !isa && a2.Parameter == null) &&
|
||
a2.Table.Type == tbtmp.Type && a2.Alias == alias && a2.Alias.StartsWith($"{navdot[0].Alias}__") &&
|
||
(isthis && a2.Type != SelectTableInfoType.Parent || !isthis && a2.Type == SelectTableInfoType.Parent)).ToArray();
|
||
if (finds.Length == 0)
|
||
finds = tsc._tables.Where(a2 =>
|
||
a2.Table.Type == tbtmp.Type && a2.Alias == alias && a2.Alias.StartsWith($"{navdot[0].Alias}__") &&
|
||
(isthis && a2.Type != SelectTableInfoType.Parent || !isthis && a2.Type == SelectTableInfoType.Parent)).ToArray();
|
||
}
|
||
else
|
||
{
|
||
finds = tsc._tables.Where(a2 => (isa && a2.Parameter != null || isa && a2.Parameter == null) &&
|
||
a2.Table.Type == tbtmp.Type && a2.Alias == alias).ToArray();
|
||
if (finds.Length != 1)
|
||
{
|
||
finds = tsc._tables.Where(a2 => (isa && a2.Parameter != null || isa && a2.Parameter == null) &&
|
||
tbtmp.Type.IsAssignableFrom(a2.Table.Type) && a2.Alias == alias).ToArray();
|
||
if (finds.Length != 1)
|
||
{
|
||
finds = tsc._tables.Where(a2 => (isa && a2.Parameter != null || isa && a2.Parameter == null) &&
|
||
a2.Table.Type == tbtmp.Type).ToArray();
|
||
if (finds.Length != 1)
|
||
{
|
||
finds = tsc._tables.Where(a2 => (isa && a2.Parameter != null || isa && a2.Parameter == null) &&
|
||
a2.Table.Type == tbtmp.Type).ToArray();
|
||
if (finds.Length != 1)
|
||
finds = tsc._tables.Where(a2 => a2.Table.Type == tbtmp.Type).ToArray();
|
||
}
|
||
}
|
||
}
|
||
}
|
||
//finds = tsc._tables.Where((a2, c2) => (isa || a2.Parameter == null) && a2.Table.CsName == tbtmp.CsName && (isthis && a2.Type != SelectTableInfoType.Parent || !isthis)).ToArray(); //外部表,内部表一起查
|
||
//if (finds.Length > 1) {
|
||
// finds = tsc._tables.Where((a2, c2) => (isa || a2.Parameter == null) && a2.Table.CsName == tbtmp.CsName && a2.Type == SelectTableInfoType.Parent && a2.Alias == alias).ToArray(); //查询外部表
|
||
// if (finds.Any() == false) {
|
||
// finds = tsc._tables.Where((a2, c2) => (isa || a2.Parameter == null) && a2.Table.CsName == tbtmp.CsName && a2.Type != SelectTableInfoType.Parent).ToArray(); //查询内部表
|
||
// if (finds.Length > 1)
|
||
// finds = tsc._tables.Where((a2, c2) => (isa || a2.Parameter == null) && a2.Table.CsName == tbtmp.CsName && a2.Type != SelectTableInfoType.Parent && a2.Alias == alias).ToArray();
|
||
// }
|
||
//}
|
||
}
|
||
var find = finds.Length == 1 ? finds.First() : null;
|
||
if (find != null && isa && parmExp != null && find.Parameter != parmExp)
|
||
find.Parameter = parmExp;
|
||
if (find == null)
|
||
{
|
||
tsc._tables.Add(find = new SelectTableInfo { Table = tbtmp, Alias = alias, On = null, Type = mp == null ? tsc.tbtype : SelectTableInfoType.LeftJoin, Parameter = isa ? parmExp : null });
|
||
if (mp?.Expression != null)
|
||
{ //导航条件,OneToOne、ManyToOne
|
||
var firstTb = tsc._tables.First().Table;
|
||
var parentTb = _common.GetTableByEntity(mp.Expression.Type);
|
||
var parentTbRef = parentTb?.GetTableRef(mp.Member.Name, tsc.style == ExpressionStyle.AsSelect);
|
||
if (parentTbRef != null)
|
||
{
|
||
Expression navCondExp = null;
|
||
for (var mn = 0; mn < parentTbRef.Columns.Count; mn++)
|
||
{
|
||
var col1 = parentTbRef.RefColumns[mn];
|
||
var col2 = parentTbRef.Columns[mn];
|
||
var pexp1 = Expression.Property(mp, col1.CsName);
|
||
var pexp2 = Expression.Property(mp.Expression, col2.CsName);
|
||
if (col1.CsType != col2.CsType)
|
||
{
|
||
if (col1.CsType.IsNullableType()) pexp1 = Expression.Property(pexp1, _dicNullableValueProperty.GetOrAdd(col1.CsType, ct1 => ct1.GetProperty("Value")));
|
||
if (col2.CsType.IsNullableType()) pexp2 = Expression.Property(pexp2, _dicNullableValueProperty.GetOrAdd(col2.CsType, ct2 => ct2.GetProperty("Value")));
|
||
}
|
||
var tmpExp = Expression.Equal(pexp1, pexp2);
|
||
if (mn == 0) navCondExp = tmpExp;
|
||
else navCondExp = Expression.AndAlso(navCondExp, tmpExp);
|
||
}
|
||
if (find.Type == SelectTableInfoType.InnerJoin ||
|
||
find.Type == SelectTableInfoType.LeftJoin ||
|
||
find.Type == SelectTableInfoType.RightJoin)
|
||
find.On = ExpressionLambdaToSql(navCondExp, tsc.Clone_selectColumnMap_diymemexp_tbtype(null, null, find.Type));
|
||
else
|
||
find.NavigateCondition = ExpressionLambdaToSql(navCondExp, tsc.Clone_selectColumnMap_diymemexp_tbtype(null, null, find.Type));
|
||
}
|
||
}
|
||
}
|
||
return find;
|
||
};
|
||
|
||
TableInfo tb2 = null;
|
||
ParameterExpression parmExp2 = null;
|
||
string alias2 = "", name2 = "";
|
||
SelectTableInfo find2 = null;
|
||
while (expStack.Count > 0)
|
||
{
|
||
exp2 = expStack.Pop();
|
||
switch (exp2.NodeType)
|
||
{
|
||
case ExpressionType.Constant:
|
||
throw new NotImplementedException($"{CoreStrings.Not_Implemented_MemberAcess_Constant}");
|
||
case ExpressionType.Parameter:
|
||
case ExpressionType.MemberAccess:
|
||
|
||
var exp2Type = exp2.Type;
|
||
if (exp2Type.FullName.StartsWith("FreeSql.ISelectGroupingAggregate`")) exp2Type = exp2Type.GetGenericArguments().LastOrDefault() ?? exp2.Type;
|
||
var mp2 = exp2 as MemberExpression;
|
||
if (mp2?.Member.Name == "Key" && mp2.Expression.Type.FullName.StartsWith("FreeSql.ISelectGroupingAggregate`")) continue;
|
||
|
||
ColumnInfo col2 = null;
|
||
if (tb2?.ColumnsByCs.TryGetValue(mp2.Member.Name, out col2) == true)
|
||
{
|
||
if (tsc._selectColumnMap != null && find2 != null)
|
||
{
|
||
tsc._selectColumnMap.Add(new SelectColumnInfo { Table = find2, Column = col2 });
|
||
return "";
|
||
}
|
||
name2 = col2.Attribute.Name;
|
||
tsc.SetMapColumnTmp(col2);
|
||
break;
|
||
}
|
||
//判断 [JsonMap] 并非导航对象,所以在上面提前判断 ColumnsByCs
|
||
|
||
var tb2tmp = _common.GetTableByEntity(exp2Type);
|
||
var exp2IsParameter = false;
|
||
if (tb2tmp != null)
|
||
{
|
||
if (exp2.NodeType == ExpressionType.Parameter)
|
||
{
|
||
parmExp2 = (exp2 as ParameterExpression);
|
||
alias2 = parmExp2.Name;
|
||
exp2IsParameter = true;
|
||
}
|
||
else if (string.IsNullOrEmpty(alias2) && exp2 is MemberExpression expMem && (
|
||
_common.GetTableByEntity(expMem.Expression.Type)?.ColumnsByCs.ContainsKey(expMem.Member.Name) == false ||
|
||
expMem.Expression.NodeType == ExpressionType.Parameter && expMem.Expression.Type.IsAnonymousType())) //<>h__TransparentIdentifier 是 Linq To Sql 的类型判断,此时为匿名类型
|
||
{
|
||
alias2 = mp2.Member.Name;
|
||
exp2IsParameter = true;
|
||
}
|
||
else
|
||
alias2 = $"{alias2}__{mp2.Member.Name}";
|
||
find2 = getOrAddTable(tb2tmp, alias2, exp2IsParameter, parmExp2, mp2);
|
||
alias2 = find2.Alias;
|
||
tb2 = find2.Table; // tb2tmp; 此处修改解决 Select<BaseEntity>().AsType(Entity1).ToList(a => a)
|
||
}
|
||
if (exp2IsParameter && expStack.Any() == false)
|
||
{ //附加选择的参数所有列
|
||
if (tsc._selectColumnMap != null)
|
||
{
|
||
foreach (var tb2c in tb2.Columns.Values)
|
||
tsc._selectColumnMap.Add(new SelectColumnInfo { Table = find2, Column = tb2c });
|
||
if (tb2.Columns.Any()) return "";
|
||
}
|
||
}
|
||
if (mp2 == null || expStack.Any()) continue;
|
||
if (tb2.ColumnsByCs.ContainsKey(mp2.Member.Name) == false)
|
||
{ //如果选的是对象,附加所有列
|
||
if (tsc._selectColumnMap != null)
|
||
{
|
||
var tb3 = _common.GetTableByEntity(mp2.Type);
|
||
if (tb3 != null)
|
||
{
|
||
var find3 = getOrAddTable(tb2tmp, alias2 /*$"{alias2}__{mp2.Member.Name}"*/, exp2IsParameter, parmExp2, mp2);
|
||
|
||
foreach (var tb3c in tb3.Columns.Values)
|
||
tsc._selectColumnMap.Add(new SelectColumnInfo { Table = find3, Column = tb3c });
|
||
if (tb3.Columns.Any()) return "";
|
||
}
|
||
}
|
||
if (tb2.ColumnsByCsIgnore.ContainsKey(mp2.Member.Name))
|
||
throw new ArgumentException(CoreStrings.Ignored_Check_Confirm_PublicGetSet(tb2.DbName, mp2.Member.Name));
|
||
if (tb2.GetTableRef(mp2.Member.Name, false) != null)
|
||
throw new ArgumentException(CoreStrings.Navigation_Missing_AsSelect(tb2.DbName, mp2.Member.Name));
|
||
throw new ArgumentException(CoreStrings.NotFound_Column(tb2.DbName, mp2.Member.Name));
|
||
}
|
||
col2 = tb2.ColumnsByCs[mp2.Member.Name];
|
||
if (tsc._selectColumnMap != null && find2 != null)
|
||
{
|
||
tsc._selectColumnMap.Add(new SelectColumnInfo { Table = find2, Column = col2 });
|
||
return "";
|
||
}
|
||
name2 = col2.Attribute.Name;
|
||
tsc.SetMapColumnTmp(col2);
|
||
break;
|
||
case ExpressionType.Call: break;
|
||
}
|
||
}
|
||
if (tsc.isQuoteName) name2 = _common.QuoteSqlName(name2);
|
||
return $"{alias2}.{name2}";
|
||
}
|
||
var expBinary = exp as BinaryExpression;
|
||
if (expBinary == null)
|
||
{
|
||
var other99Exp = ExpressionLambdaToSqlOther(exp, tsc);
|
||
if (string.IsNullOrEmpty(other99Exp) == false) return other99Exp;
|
||
return "";
|
||
}
|
||
switch (expBinary.NodeType)
|
||
{
|
||
case ExpressionType.Coalesce:
|
||
return _common.IsNull(ExpressionLambdaToSql(expBinary.Left, tsc), ExpressionLambdaToSql(expBinary.Right, tsc));
|
||
}
|
||
if (dicExpressionOperator.TryGetValue(expBinary.NodeType, out var tryoper) == false)
|
||
{
|
||
if (exp.IsParameter() == false) return formatSql(Expression.Lambda(exp).Compile().DynamicInvoke(), tsc.mapType, tsc.mapColumnTmp, tsc.dbParams);
|
||
return "";
|
||
}
|
||
return ExpressionBinary(tryoper, expBinary.Left, expBinary.Right, tsc);
|
||
}
|
||
|
||
public abstract string ExpressionLambdaToSqlMemberAccessString(MemberExpression exp, ExpTSC tsc);
|
||
public abstract string ExpressionLambdaToSqlMemberAccessDateTime(MemberExpression exp, ExpTSC tsc);
|
||
public abstract string ExpressionLambdaToSqlMemberAccessTimeSpan(MemberExpression exp, ExpTSC tsc);
|
||
public abstract string ExpressionLambdaToSqlCallString(MethodCallExpression exp, ExpTSC tsc);
|
||
public abstract string ExpressionLambdaToSqlCallMath(MethodCallExpression exp, ExpTSC tsc);
|
||
public abstract string ExpressionLambdaToSqlCallDateTime(MethodCallExpression exp, ExpTSC tsc);
|
||
public abstract string ExpressionLambdaToSqlCallTimeSpan(MethodCallExpression exp, ExpTSC tsc);
|
||
public abstract string ExpressionLambdaToSqlCallConvert(MethodCallExpression exp, ExpTSC tsc);
|
||
public abstract string ExpressionLambdaToSqlOther(Expression exp, ExpTSC tsc);
|
||
public string ExpressionConstDateTime(Expression exp) => exp is ConstantExpression operandExpConst ? formatSql(Utils.GetDataReaderValue(typeof(DateTime), operandExpConst.Value), null, null, null) : null;
|
||
|
||
public static object ExpressionGetValue(Expression exp, out bool success)
|
||
{
|
||
success = true;
|
||
var expStack = new Stack<Expression>();
|
||
var expStackConstOrMemberCount = 1;
|
||
var exp2 = exp;
|
||
while (true)
|
||
{
|
||
switch (exp2?.NodeType)
|
||
{
|
||
case ExpressionType.Constant:
|
||
expStack.Push(exp2);
|
||
expStackConstOrMemberCount++;
|
||
break;
|
||
case ExpressionType.Parameter:
|
||
expStack.Push(exp2);
|
||
break;
|
||
case ExpressionType.MemberAccess:
|
||
expStack.Push(exp2);
|
||
exp2 = (exp2 as MemberExpression).Expression;
|
||
expStackConstOrMemberCount++;
|
||
if (exp2 == null) break;
|
||
continue;
|
||
case ExpressionType.Call:
|
||
var callExp = exp2 as MethodCallExpression;
|
||
expStack.Push(exp2);
|
||
exp2 = callExp.Object;
|
||
if (exp2 == null) break;
|
||
continue;
|
||
case ExpressionType.TypeAs:
|
||
case ExpressionType.Convert:
|
||
var oper2 = (exp2 as UnaryExpression).Operand;
|
||
if (oper2.NodeType == ExpressionType.Parameter)
|
||
{
|
||
var oper2Parm = oper2 as ParameterExpression;
|
||
expStack.Push(exp2.Type.IsAbstract || exp2.Type.IsInterface ? oper2Parm : Expression.Parameter(exp2.Type, oper2Parm.Name));
|
||
}
|
||
else
|
||
expStack.Push(oper2);
|
||
break;
|
||
}
|
||
break;
|
||
}
|
||
if (expStack.Any() && expStack.First().NodeType != ExpressionType.Parameter)
|
||
{
|
||
if (expStackConstOrMemberCount == expStack.Count)
|
||
{
|
||
object firstValue = null;
|
||
switch (expStack.First().NodeType)
|
||
{
|
||
case ExpressionType.Constant:
|
||
var expStackFirst = expStack.Pop() as ConstantExpression;
|
||
firstValue = expStackFirst?.Value;
|
||
break;
|
||
case ExpressionType.MemberAccess:
|
||
var expStackFirstMem = expStack.First() as MemberExpression;
|
||
if (expStackFirstMem.Expression?.NodeType == ExpressionType.Constant)
|
||
firstValue = (expStackFirstMem.Expression as ConstantExpression)?.Value;
|
||
else
|
||
return Expression.Lambda(exp).Compile().DynamicInvoke();
|
||
break;
|
||
}
|
||
while (expStack.Any())
|
||
{
|
||
if (firstValue == null) throw new Exception(CoreStrings.Cannot_Be_NULL_Name(exp));
|
||
var expStackItem = expStack.Pop() as MemberExpression;
|
||
if (expStackItem.Member.MemberType == MemberTypes.Property)
|
||
firstValue = ((PropertyInfo)expStackItem.Member).GetValue(firstValue, null);
|
||
else if (expStackItem.Member.MemberType == MemberTypes.Field)
|
||
firstValue = ((FieldInfo)expStackItem.Member).GetValue(firstValue);
|
||
}
|
||
return firstValue;
|
||
}
|
||
return Expression.Lambda(exp).Compile().DynamicInvoke();
|
||
}
|
||
if (exp.IsParameter() == false)
|
||
return Expression.Lambda(exp).Compile().DynamicInvoke();
|
||
success = false;
|
||
return null;
|
||
}
|
||
|
||
public enum ExpressionStyle
|
||
{
|
||
Where, AsSelect, SelectColumns
|
||
}
|
||
public class ExpTSC
|
||
{
|
||
public List<SelectTableInfo> _tables { get; set; }
|
||
public Func<Type, string, string> _tableRule { get; set; }
|
||
public List<SelectColumnInfo> _selectColumnMap { get; set; }
|
||
public BaseDiyMemberExpression diymemexp { get; set; }
|
||
public Select0Provider subSelect001 { get; set; } //#405 Oracle within group(order by ..)
|
||
public SelectTableInfoType tbtype { get; set; }
|
||
public bool isQuoteName { get; set; }
|
||
public bool isDisableDiyParse { get; set; }
|
||
public ExpressionStyle style { get; set; }
|
||
public Type mapType { get; set; }
|
||
public Type mapTypeTmp { get; set; }
|
||
public ColumnInfo mapColumnTmp { get; set; }
|
||
public bool isNotSetMapColumnTmp { get; set; }
|
||
public TableInfo currentTable { get; set; }
|
||
public List<GlobalFilter.Item> whereGlobalFilter { get; set; }
|
||
public List<DbParameter> dbParams { get; set; }
|
||
public string alias001 { get; set; } //单表字段的表别名
|
||
|
||
public ExpTSC SetMapColumnTmp(ColumnInfo col)
|
||
{
|
||
if (isNotSetMapColumnTmp) return this;
|
||
if (col == null)
|
||
{
|
||
this.mapTypeTmp = null;
|
||
this.mapColumnTmp = null;
|
||
}
|
||
else
|
||
{
|
||
this.mapTypeTmp = col.Attribute.MapType == col.CsType ? null : col.Attribute.MapType;
|
||
this.mapColumnTmp = col;
|
||
}
|
||
return this;
|
||
}
|
||
public Type SetMapTypeReturnOld(Type newValue)
|
||
{
|
||
var old = this.mapType;
|
||
this.mapType = newValue;
|
||
return old;
|
||
}
|
||
public List<DbParameter> SetDbParamsReturnOld(List<DbParameter> newValue)
|
||
{
|
||
var old = this.dbParams;
|
||
this.dbParams = newValue;
|
||
return old;
|
||
}
|
||
|
||
public ExpTSC Clone_selectColumnMap_diymemexp_tbtype(List<SelectColumnInfo> v1, BaseDiyMemberExpression v2, SelectTableInfoType v3)
|
||
{
|
||
return new ExpTSC
|
||
{
|
||
_tables = this._tables,
|
||
_tableRule = this._tableRule,
|
||
_selectColumnMap = v1,
|
||
diymemexp = v2,
|
||
tbtype = v3,
|
||
isQuoteName = this.isQuoteName,
|
||
isDisableDiyParse = this.isDisableDiyParse,
|
||
style = this.style,
|
||
//mapType = this.mapType,
|
||
//mapTypeTmp = this.mapTypeTmp,
|
||
//mapColumnTmp = this.mapColumnTmp,
|
||
currentTable = this.currentTable,
|
||
whereGlobalFilter = this.whereGlobalFilter,
|
||
dbParams = this.dbParams,
|
||
alias001 = this.alias001
|
||
};
|
||
}
|
||
public ExpTSC CloneDisableDiyParse()
|
||
{
|
||
return new ExpTSC
|
||
{
|
||
_tables = this._tables,
|
||
_tableRule = this._tableRule,
|
||
_selectColumnMap = this._selectColumnMap,
|
||
diymemexp = this.diymemexp,
|
||
subSelect001 = this.subSelect001,
|
||
tbtype = this.tbtype,
|
||
isQuoteName = this.isQuoteName,
|
||
isDisableDiyParse = true,
|
||
style = this.style,
|
||
mapType = this.mapType,
|
||
mapTypeTmp = this.mapTypeTmp,
|
||
mapColumnTmp = this.mapColumnTmp,
|
||
currentTable = this.currentTable,
|
||
whereGlobalFilter = this.whereGlobalFilter,
|
||
dbParams = this.dbParams,
|
||
alias001 = this.alias001
|
||
};
|
||
}
|
||
}
|
||
|
||
static ConcurrentDictionary<Type, ConcurrentDictionary<string, bool>> _dicGetWhereCascadeSqlError = new ConcurrentDictionary<Type, ConcurrentDictionary<string, bool>>();
|
||
public string GetWhereCascadeSql(SelectTableInfo tb, List<GlobalFilter.Item> filters, bool isMultitb)
|
||
{
|
||
if (filters.Any())
|
||
{
|
||
var newParameter = Expression.Parameter(tb.Table.Type, tb.Alias);
|
||
tb.Parameter = newParameter;
|
||
var sb = new StringBuilder();
|
||
var isEmpty = true;
|
||
|
||
foreach (var fl in filters)
|
||
{
|
||
if (fl.Only && fl.Where.Parameters.FirstOrDefault()?.Type.IsAssignableFrom(tb.Table.Type) == false) continue;
|
||
var dicSqlError = _dicGetWhereCascadeSqlError.GetOrAdd(tb.Table.Type, tp => new ConcurrentDictionary<string, bool>());
|
||
var errorKey = FreeUtil.Sha1($"{(isMultitb ? 1 : 0)},{fl.Where.ToString()}");
|
||
if (dicSqlError.ContainsKey(errorKey)) continue;
|
||
|
||
var visitor = new ReplaceParameterVisitor();
|
||
try
|
||
{
|
||
var expExp = Expression.Lambda(
|
||
typeof(Func<,>).MakeGenericType(tb.Table.Type, typeof(bool)),
|
||
new ReplaceParameterVisitor().Modify(fl.Where, newParameter),
|
||
newParameter
|
||
);
|
||
var whereSql = ExpressionLambdaToSql(expExp.Body, new ExpTSC
|
||
{
|
||
_tables = isMultitb ? new List<SelectTableInfo>(new[] { tb }) : null,
|
||
_selectColumnMap = null,
|
||
diymemexp = null,
|
||
tbtype = SelectTableInfoType.From,
|
||
isQuoteName = true,
|
||
isDisableDiyParse = false,
|
||
style = ExpressionStyle.Where,
|
||
currentTable = tb.Table,
|
||
alias001 = tb.Alias
|
||
});
|
||
whereSql = GetBoolString(expExp.Body, whereSql);
|
||
if (isEmpty == false)
|
||
sb.Append(" AND ");
|
||
else
|
||
isEmpty = false;
|
||
sb.Append("(").Append(whereSql).Append(")");
|
||
}
|
||
catch
|
||
{
|
||
dicSqlError.TryAdd(errorKey, true);
|
||
continue;
|
||
}
|
||
}
|
||
|
||
if (isEmpty == false)
|
||
return sb.ToString();
|
||
}
|
||
return null;
|
||
}
|
||
public class ReplaceVisitor : ExpressionVisitor
|
||
{
|
||
private Expression _oldexp;
|
||
private Expression _newexp;
|
||
public Expression Modify(Expression find, Expression oldexp, Expression newexp)
|
||
{
|
||
this._oldexp = oldexp;
|
||
this._newexp = newexp;
|
||
return Visit(find);
|
||
}
|
||
protected override Expression VisitMember(MemberExpression node)
|
||
{
|
||
if (node.Expression == _oldexp)
|
||
return Expression.Property(_newexp, node.Member.Name);
|
||
if (node == _oldexp)
|
||
return _newexp;
|
||
return base.VisitMember(node);
|
||
}
|
||
}
|
||
public class ReplaceParameterVisitor : ExpressionVisitor
|
||
{
|
||
private Expression _replaceExp;
|
||
private ParameterExpression oldParameter;
|
||
public Expression Modify(LambdaExpression lambda, Expression replaceExp)
|
||
{
|
||
this._replaceExp = replaceExp;
|
||
this.oldParameter = lambda.Parameters.FirstOrDefault();
|
||
return Visit(lambda.Body);
|
||
}
|
||
protected override Expression VisitMember(MemberExpression node)
|
||
{
|
||
if (node.Expression?.NodeType == ExpressionType.Parameter && node.Expression == oldParameter)
|
||
return Expression.Property(_replaceExp, node.Member.Name);
|
||
return base.VisitMember(node);
|
||
}
|
||
protected override Expression VisitParameter(ParameterExpression node)
|
||
{
|
||
if (node == oldParameter)
|
||
return _replaceExp;
|
||
return base.VisitParameter(node);
|
||
}
|
||
}
|
||
|
||
public class ReplaceHzyTupleToMultiParam : ExpressionVisitor
|
||
{
|
||
private List<SelectTableInfo> tables;
|
||
private ParameterExpression[] parameters;
|
||
private ParameterExpression lambdaHzyParameter;
|
||
public LambdaExpression Modify(LambdaExpression lambda, List<SelectTableInfo> tables)
|
||
{
|
||
this.tables = tables.Where(a => a.Type != SelectTableInfoType.Parent).ToList();
|
||
parameters = this.tables.Select(a => a.Parameter ??
|
||
Expression.Parameter(a.Table.Type,
|
||
a.Alias.StartsWith("SP10") ? a.Alias.Replace("SP10", "ht") : a.Alias)).ToArray();
|
||
lambdaHzyParameter = lambda.Parameters.FirstOrDefault();
|
||
var exp = Visit(lambda.Body);
|
||
return Expression.Lambda(exp, parameters);
|
||
}
|
||
|
||
protected override Expression VisitMember(MemberExpression node)
|
||
{
|
||
int widx;
|
||
if (node.Expression?.NodeType == ExpressionType.MemberAccess)
|
||
{
|
||
var parent = node.Expression as MemberExpression;
|
||
if (parent.Expression?.NodeType == ExpressionType.Parameter &&
|
||
parent.Expression.Type.Name.StartsWith("HzyTuple`") == true &&
|
||
parent.Expression == lambdaHzyParameter &&
|
||
int.TryParse(parent.Member.Name.Replace("t", ""), out widx) && widx > 0 && widx <= tables.Count)
|
||
{
|
||
if (parameters[widx - 1].Type != parent.Type) //解决 BaseEntity + AsTable 时报错
|
||
parameters[widx - 1] = Expression.Parameter(parent.Type, parameters[widx - 1].Name);
|
||
return Expression.Property(parameters[widx - 1], node.Member.Name);
|
||
}
|
||
}
|
||
|
||
if (node.Expression?.NodeType == ExpressionType.Parameter &&
|
||
node.Expression.Type.Name.StartsWith("HzyTuple`") == true &&
|
||
node.Expression == lambdaHzyParameter &&
|
||
int.TryParse(node.Member.Name.Replace("t", ""), out widx) && widx > 0 && widx <= tables.Count)
|
||
{
|
||
if (parameters[widx - 1].Type != node.Type) //解决 BaseEntity + AsTable 时报错
|
||
parameters[widx - 1] = Expression.Parameter(node.Type, parameters[widx - 1].Name);
|
||
return parameters[widx - 1];
|
||
}
|
||
|
||
return base.VisitMember(node);
|
||
}
|
||
}
|
||
|
||
public string formatSql(object obj, Type mapType, ColumnInfo mapColumn, List<DbParameter> dbParams)
|
||
{
|
||
//参数化设置,日后优化
|
||
if (_common.CodeFirst.IsGenerateCommandParameterWithLambda && dbParams != null)
|
||
{
|
||
if (obj == null) return "NULL";
|
||
if (mapColumn != null)
|
||
{
|
||
var objType = obj.GetType();
|
||
if (obj is ICollection && objType.GetGenericArguments().FirstOrDefault()?.NullableTypeOrThis() == mapColumn.CsType?.NullableTypeOrThis())
|
||
return string.Format(CultureInfo.InvariantCulture, "{0}", _ado.AddslashesProcessParam(obj, mapType, mapColumn));
|
||
if (obj is Array && objType.GetElementType()?.NullableTypeOrThis() == mapColumn.CsType?.NullableTypeOrThis())
|
||
return string.Format(CultureInfo.InvariantCulture, "{0}", _ado.AddslashesProcessParam(obj, mapType, mapColumn));
|
||
}
|
||
var type = mapType ?? mapColumn?.Attribute.MapType ?? obj?.GetType();
|
||
if (_common.CodeFirst.GetDbInfo(type) != null)
|
||
{
|
||
var paramName = $"exp_{dbParams.Count}";
|
||
if (_common._orm?.Ado.DataType == DataType.GBase) paramName = "?";
|
||
var parm = _common.AppendParamter(dbParams, paramName, mapColumn, type, mapType == null ? obj : Utils.GetDataReaderValue(mapType, obj));
|
||
return _common.QuoteParamterName(paramName);
|
||
}
|
||
}
|
||
return string.Format(CultureInfo.InvariantCulture, "{0}", _ado.AddslashesProcessParam(obj, mapType, mapColumn));
|
||
//return string.Concat(_ado.AddslashesProcessParam(obj, mapType, mapColumn));
|
||
}
|
||
|
||
public static void ParseExpressionNoAsSelect(object sender, Aop.ParseExpressionEventArgs e, Func<Type, string, string> tableRule, List<GlobalFilter.Item> whereGlobalFilter)
|
||
{
|
||
if (e.Expression.NodeType != ExpressionType.Call &&
|
||
(e.Expression as MemberExpression)?.Member.Name != "Count") return;
|
||
var exp3Stack = new Stack<Expression>();
|
||
var exp3tmp = e.Expression;
|
||
while (exp3tmp != null)
|
||
{
|
||
exp3Stack.Push(exp3tmp);
|
||
switch (exp3tmp.NodeType)
|
||
{
|
||
case ExpressionType.Call:
|
||
var exp3tmpCall = (exp3tmp as MethodCallExpression);
|
||
if (exp3tmpCall.Type.FullName.StartsWith("FreeSql.ISelect`") && exp3tmpCall.Method.Name == "AsSelect" && exp3tmpCall.Object == null) return;
|
||
exp3tmp = exp3tmpCall.Object == null ? exp3tmpCall.Arguments.FirstOrDefault() : exp3tmpCall.Object;
|
||
continue;
|
||
case ExpressionType.MemberAccess:
|
||
exp3tmp = (exp3tmp as MemberExpression).Expression;
|
||
continue;
|
||
case ExpressionType.Parameter:
|
||
exp3tmp = null;
|
||
continue;
|
||
}
|
||
return;
|
||
}
|
||
exp3tmp = exp3Stack.Pop();
|
||
if (exp3tmp.NodeType != ExpressionType.Parameter)
|
||
{
|
||
//if (e.Expression.NodeType == ExpressionType.Call)
|
||
//{
|
||
// var rootExpCall = e.Expression as MethodCallExpression;
|
||
// if (rootExpCall.Object == null && rootExpCall.Method.Name == "Any" &&
|
||
// rootExpCall.Arguments.Count == 2 &&
|
||
// rootExpCall.Arguments[0].Type.GetGenericTypeDefinition() == typeof(IEnumerable<>) &&
|
||
// rootExpCall.Arguments[1].Type == typeof(Func<,>).MakeGenericType(rootExpCall.Arguments[0].Type.GetGenericArguments()[0], typeof(bool)))
|
||
// {
|
||
// //e.Tables[0].Parameter
|
||
// var anyExp = rootExpCall.Arguments[1];
|
||
// while(anyExp.NodeType == ExpressionType.AndAlso)
|
||
// {
|
||
|
||
// }
|
||
// if (anyExp.NodeType != ExpressionType.AndAlso && anyExp.NodeType != ExpressionType.Equal) return;
|
||
|
||
|
||
// var array = Expression.Lambda(rootExpCall.Arguments[0]).Compile().DynamicInvoke() as IEnumerable;
|
||
// foreach (var arritem in array)
|
||
// {
|
||
// }
|
||
// }
|
||
//}
|
||
return;
|
||
}
|
||
if (exp3tmp.Type.FullName.StartsWith("FreeSql.ISelectGroupingAggregate")) return;
|
||
var commonExp = sender as FreeSql.Internal.CommonExpression;
|
||
if (commonExp == null) return;
|
||
var exp3Tb = commonExp._common.GetTableByEntity(exp3tmp.Type);
|
||
if (exp3Tb == null) return;
|
||
var paramExp = exp3tmp as ParameterExpression;
|
||
Select1Provider<object> select = null;
|
||
TableRef memberTbref = null;
|
||
MemberExpression memberExp = null;
|
||
bool selectSetAliased = false;
|
||
|
||
void LocalSetSelectProviderAlias(string alias)
|
||
{
|
||
if (selectSetAliased) return;
|
||
if (alias.Length == 1 && alias[0] >= 'a' && alias[0] <= 'd')
|
||
alias = $"sub_{alias}";
|
||
selectSetAliased = true;
|
||
select._tables[0].Alias = alias;
|
||
select._tables[0].AliasInit = alias;
|
||
switch (memberTbref.RefType)
|
||
{
|
||
case TableRefType.ManyToMany:
|
||
var mtmReftbname = e.FreeParse(Expression.MakeMemberAccess(memberExp.Expression, exp3Tb.Properties[exp3Tb.ColumnsByPosition[0].CsName]));
|
||
mtmReftbname = mtmReftbname.Substring(0, mtmReftbname.Length - commonExp._common.QuoteSqlName(exp3Tb.ColumnsByPosition[0].Attribute.Name).Length - 1);
|
||
var midSelect = commonExp._common._orm.Select<object>().As($"M{select._tables[0].Alias}_M{mtmReftbname}").AsType(memberTbref.RefMiddleEntityType) as Select1Provider<object>;
|
||
if (tableRule != null) midSelect._tableRules.Add(tableRule);
|
||
if (whereGlobalFilter != null)
|
||
{
|
||
midSelect._whereGlobalFilter.Clear();
|
||
if (whereGlobalFilter.Any()) midSelect._whereGlobalFilter.AddRange(whereGlobalFilter);
|
||
}
|
||
switch (commonExp._ado.DataType)
|
||
{
|
||
case DataType.Oracle:
|
||
case DataType.OdbcOracle:
|
||
case DataType.Dameng:
|
||
case DataType.OdbcDameng:
|
||
case DataType.GBase:
|
||
break;
|
||
default:
|
||
midSelect.Limit(1); //#462 ORACLE rownum <= 2 会影响索引变慢
|
||
break;
|
||
}
|
||
for (var tidx = 0; tidx < memberTbref.RefColumns.Count; tidx++)
|
||
midSelect.Where($"{midSelect._tables[0].Alias}.{commonExp._common.QuoteSqlName(memberTbref.MiddleColumns[memberTbref.Columns.Count + tidx].Attribute.Name)} = {select._tables[0].Alias}.{commonExp._common.QuoteSqlName(memberTbref.RefColumns[tidx].Attribute.Name)}");
|
||
for (var tidx = 0; tidx < memberTbref.Columns.Count; tidx++)
|
||
midSelect.Where($"{midSelect._tables[0].Alias}.{commonExp._common.QuoteSqlName(memberTbref.MiddleColumns[tidx].Attribute.Name)} = {mtmReftbname}.{commonExp._common.QuoteSqlName(memberTbref.Columns[tidx].Attribute.Name)}");
|
||
select.Where($"exists({midSelect.ToSql("1").Replace(" \r\n", " \r\n ")})");
|
||
break;
|
||
case TableRefType.OneToMany:
|
||
var omtReftbname = e.FreeParse(Expression.MakeMemberAccess(memberExp.Expression, exp3Tb.Properties[exp3Tb.ColumnsByPosition[0].CsName]));
|
||
omtReftbname = omtReftbname.Substring(0, omtReftbname.Length - commonExp._common.QuoteSqlName(exp3Tb.ColumnsByPosition[0].Attribute.Name).Length - 1);
|
||
for (var tidx = 0; tidx < memberTbref.Columns.Count; tidx++)
|
||
select.Where($"{select._tables[0].Alias}.{commonExp._common.QuoteSqlName(memberTbref.RefColumns[tidx].Attribute.Name)} = {omtReftbname}.{commonExp._common.QuoteSqlName(memberTbref.Columns[tidx].Attribute.Name)}");
|
||
break;
|
||
case TableRefType.PgArrayToMany:
|
||
var amtReftbname = e.FreeParse(Expression.MakeMemberAccess(memberExp.Expression, exp3Tb.Properties[exp3Tb.ColumnsByPosition[0].CsName]));
|
||
amtReftbname = amtReftbname.Substring(0, amtReftbname.Length - commonExp._common.QuoteSqlName(exp3Tb.ColumnsByPosition[0].Attribute.Name).Length - 1);
|
||
if (memberTbref.RefColumns[0] == select._tables[0].Table.Primarys[0])
|
||
{
|
||
var dbinfo = commonExp._common._orm.CodeFirst.GetDbInfo(memberTbref.Columns[0].CsType);
|
||
select.Where($"{amtReftbname}.{commonExp._common.QuoteSqlName(memberTbref.Columns[0].Attribute.Name)} @> ARRAY[{select._tables[0].Alias}.{commonExp._common.QuoteSqlName(memberTbref.RefColumns[0].Attribute.Name)}]::{dbinfo?.dbtype}");
|
||
}
|
||
else if (memberTbref.Columns[0] == exp3Tb.Primarys[0])
|
||
{
|
||
var dbinfo = commonExp._common._orm.CodeFirst.GetDbInfo(memberTbref.RefColumns[0].CsType);
|
||
select.Where($"{select._tables[0].Alias}.{commonExp._common.QuoteSqlName(memberTbref.RefColumns[0].Attribute.Name)} @> ARRAY[{amtReftbname}.{commonExp._common.QuoteSqlName(memberTbref.Columns[0].Attribute.Name)}]::{dbinfo?.dbtype}");
|
||
}
|
||
else
|
||
{
|
||
;
|
||
}
|
||
break;
|
||
}
|
||
|
||
}
|
||
void LocalInitSelectProvider()
|
||
{
|
||
select = commonExp._common._orm.Select<object>().AsType(memberTbref.RefEntityType) as Select1Provider<object>;
|
||
select._tables.AddRange(e.Tables.Select(a => new SelectTableInfo
|
||
{
|
||
Alias = a.Alias,
|
||
On = "1=1",
|
||
Table = a.Table,
|
||
Type = SelectTableInfoType.Parent,
|
||
Parameter = a.Parameter
|
||
}));
|
||
if (tableRule != null) select._tableRules.Add(tableRule);
|
||
if (whereGlobalFilter != null)
|
||
{
|
||
select._whereGlobalFilter.Clear();
|
||
if (whereGlobalFilter.Any()) select._whereGlobalFilter.AddRange(whereGlobalFilter);
|
||
}
|
||
}
|
||
while (exp3Stack.Any())
|
||
{
|
||
var exp4 = exp3Stack.Pop();
|
||
if (exp4.NodeType == ExpressionType.MemberAccess)
|
||
{
|
||
var tmpExp = exp4 as MemberExpression;
|
||
if (tmpExp.Member.Name == "Count" && select != null)
|
||
{
|
||
if (exp3Stack.Any()) return;
|
||
LocalSetSelectProviderAlias("tbcou");
|
||
e.Result = $"({select.ToSql("count(1)").Replace(" \r\n", " \r\n ")})";
|
||
return;
|
||
}
|
||
if (select != null) return;
|
||
memberExp = tmpExp;
|
||
memberTbref = exp3Tb.GetTableRef(memberExp.Member.Name, false);
|
||
if (memberTbref == null) return;
|
||
switch (memberTbref.RefType)
|
||
{
|
||
case TableRefType.ManyToOne:
|
||
case TableRefType.OneToOne:
|
||
exp3Tb = commonExp._common.GetTableByEntity(memberExp.Type);
|
||
if (exp3Tb == null) return;
|
||
continue;
|
||
case TableRefType.ManyToMany:
|
||
if (select != null) return;
|
||
LocalInitSelectProvider();
|
||
continue;
|
||
case TableRefType.OneToMany:
|
||
if (select != null) return;
|
||
LocalInitSelectProvider();
|
||
continue;
|
||
case TableRefType.PgArrayToMany:
|
||
if (select != null) return;
|
||
LocalInitSelectProvider();
|
||
continue;
|
||
}
|
||
}
|
||
if (exp4.NodeType == ExpressionType.Call)
|
||
{
|
||
if (select == null) return;
|
||
var callExp = exp4 as MethodCallExpression;
|
||
switch (callExp.Method.Name)
|
||
{
|
||
case "Exists":
|
||
case "Any":
|
||
if (callExp.Method.Name == "Exists" && callExp.Arguments.Count == 1 && callExp.Type.IsGenericType && callExp.Type.GetGenericTypeDefinition().IsAssignableFrom(typeof(IList<>)))
|
||
{
|
||
select._tables[0].Parameter = (callExp.Arguments[0] as LambdaExpression)?.Parameters.FirstOrDefault();
|
||
LocalSetSelectProviderAlias(select._tables[0].Parameter.Name);
|
||
select.InternalWhere(callExp.Arguments[0]);
|
||
}
|
||
if (callExp.Method.Name == "Any" && callExp.Arguments.Count == 2)
|
||
{
|
||
select._tables[0].Parameter = (callExp.Arguments[1] as LambdaExpression)?.Parameters.FirstOrDefault();
|
||
LocalSetSelectProviderAlias(select._tables[0].Parameter.Name);
|
||
select.InternalWhere(callExp.Arguments[1]);
|
||
}
|
||
switch (commonExp._ado.DataType)
|
||
{
|
||
case DataType.Oracle:
|
||
case DataType.OdbcOracle:
|
||
case DataType.Dameng:
|
||
case DataType.OdbcDameng:
|
||
case DataType.GBase:
|
||
break;
|
||
default:
|
||
select._limit = 1; //#462 ORACLE rownum <= 2 会影响索引变慢
|
||
break;
|
||
}
|
||
if (exp3Stack.Any()) return;
|
||
LocalSetSelectProviderAlias("tbany");
|
||
e.Result = $"exists({select.ToSql("1").Replace(" \r\n", " \r\n ")})";
|
||
return;
|
||
case "Max":
|
||
case "Min":
|
||
case "Sum":
|
||
case "Average":
|
||
if (callExp.Arguments.Count == 2)
|
||
{
|
||
var aggregateMethodName = callExp.Method.Name == "Average" ? "avg" : callExp.Method.Name.ToLower();
|
||
if (exp3Stack.Any()) return;
|
||
select._tables[0].Parameter = (callExp.Arguments[1] as LambdaExpression)?.Parameters.FirstOrDefault();
|
||
LocalSetSelectProviderAlias(select._tables[0].Parameter.Name);
|
||
|
||
var map = new ReadAnonymousTypeInfo();
|
||
var field = new StringBuilder();
|
||
var index = -1;
|
||
|
||
commonExp.ReadAnonymousField(select._tables, select._tableRule, field, map, ref index, callExp.Arguments[1], null, null, null, null, null, false);
|
||
var fieldSql = field.Length > 0 ? field.Remove(0, 2).ToString() : null;
|
||
|
||
e.Result = commonExp._common.IsNull($"({select.ToSql($"{aggregateMethodName}({fieldSql})").Replace(" \r\n", " \r\n ")})", commonExp.formatSql(callExp.Method.ReturnType.CreateInstanceGetDefaultValue(), callExp.Method.ReturnType, null, null));
|
||
return;
|
||
}
|
||
throw throwCallExp($"不支持 {callExp.Arguments.Count}个参数的方法");
|
||
case "Count":
|
||
if (callExp.Arguments.Count == 2)
|
||
{
|
||
select._tables[0].Parameter = (callExp.Arguments[1] as LambdaExpression)?.Parameters.FirstOrDefault();
|
||
LocalSetSelectProviderAlias(select._tables[0].Parameter.Name);
|
||
select.InternalWhere(callExp.Arguments[1]);
|
||
}
|
||
if (exp3Stack.Any()) return;
|
||
LocalSetSelectProviderAlias("tbcou");
|
||
e.Result = $"({select.ToSql("count(1)").Replace(" \r\n", " \r\n ")})";
|
||
return;
|
||
|
||
case "First":
|
||
select.Limit(1);
|
||
if (callExp.Arguments.Count == 1)
|
||
{
|
||
if (exp3Stack.Any()) return;
|
||
LocalSetSelectProviderAlias("tbfirst");
|
||
e.Result = $"({select.ToSql().Replace(" \r\n", " \r\n ")})";
|
||
return;
|
||
}
|
||
throw throwCallExp(CoreStrings.Not_Support);
|
||
case "ToList":
|
||
if (callExp.Arguments.Count == 1)
|
||
{
|
||
if (exp3Stack.Any()) return;
|
||
LocalSetSelectProviderAlias("tbtolist");
|
||
e.Result = $"({select.ToSql().Replace(" \r\n", " \r\n ")})";
|
||
return;
|
||
}
|
||
throw throwCallExp(CoreStrings.Not_Support);
|
||
case "Contains":
|
||
if (callExp.Arguments.Count == 2)
|
||
{
|
||
if (exp3Stack.Any()) return;
|
||
select._tables[0].Parameter = (callExp.Arguments[1] as LambdaExpression)?.Parameters.FirstOrDefault();
|
||
LocalSetSelectProviderAlias(select._tables[0].Parameter.Name);
|
||
e.Result = $"({e.FreeParse(callExp.Arguments[1])}) in {select.ToSql().Replace(" \r\n", " \r\n ")})";
|
||
return;
|
||
}
|
||
throw throwCallExp($" 不支持 {callExp.Arguments.Count}个参数的方法");
|
||
|
||
case "Distinct":
|
||
if (callExp.Arguments.Count == 1)
|
||
{
|
||
select.Distinct();
|
||
break;
|
||
}
|
||
throw throwCallExp(CoreStrings.Not_Support);
|
||
case "OrderBy":
|
||
select._tables[0].Parameter = (callExp.Arguments[1] as LambdaExpression)?.Parameters.FirstOrDefault();
|
||
LocalSetSelectProviderAlias(select._tables[0].Parameter.Name);
|
||
select.OrderByReflection(callExp.Arguments[1] as LambdaExpression as LambdaExpression, false);
|
||
break;
|
||
case "OrderByDescending":
|
||
select._tables[0].Parameter = (callExp.Arguments[1] as LambdaExpression)?.Parameters.FirstOrDefault();
|
||
LocalSetSelectProviderAlias(select._tables[0].Parameter.Name);
|
||
select.OrderByReflection(callExp.Arguments[1] as LambdaExpression as LambdaExpression, true);
|
||
break;
|
||
case "ThenBy":
|
||
select._tables[0].Parameter = (callExp.Arguments[1] as LambdaExpression)?.Parameters.FirstOrDefault();
|
||
LocalSetSelectProviderAlias(select._tables[0].Parameter.Name);
|
||
select.OrderByReflection(callExp.Arguments[1] as LambdaExpression as LambdaExpression, false);
|
||
break;
|
||
case "ThenByDescending":
|
||
select._tables[0].Parameter = (callExp.Arguments[1] as LambdaExpression)?.Parameters.FirstOrDefault();
|
||
LocalSetSelectProviderAlias(select._tables[0].Parameter.Name);
|
||
select.OrderByReflection(callExp.Arguments[1] as LambdaExpression as LambdaExpression, true);
|
||
break;
|
||
|
||
case "Where":
|
||
var whereParam = callExp.Arguments[1] as LambdaExpression;
|
||
if (whereParam?.Parameters.Count == 1)
|
||
{
|
||
select._tables[0].Parameter = whereParam.Parameters.FirstOrDefault();
|
||
LocalSetSelectProviderAlias(select._tables[0].Parameter.Name);
|
||
select.InternalWhere(whereParam);
|
||
break;
|
||
}
|
||
throw throwCallExp(CoreStrings.Not_Support);
|
||
|
||
case "Skip":
|
||
select.Offset((int)callExp.Arguments[1].GetConstExprValue());
|
||
break;
|
||
case "Take":
|
||
select.Limit((int)callExp.Arguments[1].GetConstExprValue());
|
||
break;
|
||
|
||
case "Select":
|
||
var selectParam = callExp.Arguments[1] as LambdaExpression;
|
||
if (selectParam?.Parameters.Count == 1)
|
||
{
|
||
select._tables[0].Parameter = selectParam.Parameters.FirstOrDefault();
|
||
LocalSetSelectProviderAlias(select._tables[0].Parameter.Name);
|
||
select._selectExpression = selectParam;
|
||
break;
|
||
}
|
||
throw throwCallExp(CoreStrings.Not_Support);
|
||
}
|
||
Exception throwCallExp(string message) => new Exception(CoreStrings.Parsing_Failed(callExp.Method.Name, message));
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|