1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369 |
- using System;
- using System.Collections.Generic;
- using System.Collections.ObjectModel;
- using System.IO;
- using System.Linq;
- using System.Linq.Expressions;
- using System.Reflection;
- using System.Text;
- using System.Text.RegularExpressions;
- namespace Common.LambdaToSQL
- {
- internal enum SqlType : byte
- {
- Where,
- Order
- }
- /// <summary>
- /// 输出一个基于C#分析器的表达示树
- /// </summary>
- internal class ExpressionWriterSql : ExpressionVisitor
- {
- private readonly TextWriter _writer;
- private int depth = 0;
- private string _aiWhereResult = string.Empty;
- private string _aiOrdeRsult = string.Empty;
- private int AiOrderTime { get; set; } = 0;
- private int AiWhereTime { get; set; } = 0;
- private SqlType _bizRead = SqlType.Where;
- private ExpressionWriterSql(TextWriter writer)
- {
- this._writer = writer;
- }
- private static void Write(TextWriter writer, Expression expression)
- {
- new ExpressionWriterSql(writer).Visit(expression);
- }
- private static string Write(TextWriter writer, Expression expression, SqlType bizSql)
- {
- expression = PartialEvaluator.Eval(expression);
- ExpressionWriterSql bizR = new ExpressionWriterSql(writer)
- {
- _bizRead = bizSql
- };
- bizR.Visit(expression);
- string result;
- switch (bizSql)
- {
- case SqlType.Order:
- result = Regex.Replace(bizR._aiOrdeRsult, @",\s?$", "");
- return result;
- case SqlType.Where:
- result = Regex.Replace(bizR._aiWhereResult, @"And\s?$", "");
- return result; ;
- default: return string.Empty;
- }
- }
- private static string WriteToString(Expression expression)
- {
- StringWriter sw = new StringWriter();
- Write(sw, expression);
- return sw.ToString();
- }
- public static string BizWhereWriteToString(Expression expression, SqlType bizSql)
- {
- StringWriter sw = new StringWriter();
- return Write(sw, expression, bizSql);
- }
- protected enum Indentation
- {
- Same,
- Inner,
- Outer
- }
- protected int IndentationWidth { get; set; } = 2;
- protected void WriteLine(Indentation style)
- {
- this._writer.WriteLine();
- // this.Indent(style);
- }
- private static readonly char[] Splitters = new char[] { '\n', '\r' };
- private void Write(string text)
- {
- switch (_bizRead)
- {
- case SqlType.Order:
- _aiOrdeRsult = _aiOrdeRsult + text;
- break;
- case SqlType.Where:
- _aiWhereResult = _aiWhereResult + text;
- break;
- }
- this._writer.Write(text);
- }
- protected void Indent(Indentation style)
- {
- if (style == Indentation.Inner)
- {
- // this.depth++;
- }
- else if (style == Indentation.Outer)
- {
- // this.depth--;
- System.Diagnostics.Debug.Assert(this.depth >= 0);
- }
- }
- protected virtual string GetOperator(ExpressionType type)
- {
- switch (type)
- {
- case ExpressionType.Not:
- return "!";
- case ExpressionType.Add:
- case ExpressionType.AddChecked:
- return "+";
- case ExpressionType.Negate:
- case ExpressionType.NegateChecked:
- case ExpressionType.Subtract:
- case ExpressionType.SubtractChecked:
- return "-";
- case ExpressionType.Multiply:
- case ExpressionType.MultiplyChecked:
- return "*";
- case ExpressionType.Divide:
- return "/";
- case ExpressionType.Modulo:
- return "%";
- case ExpressionType.And:
- return "&";
- case ExpressionType.AndAlso:
- return "And";
- case ExpressionType.Or:
- return "Or";
- case ExpressionType.OrElse:
- return "Or";
- case ExpressionType.LessThan:
- return "<";
- case ExpressionType.LessThanOrEqual:
- return "<=";
- case ExpressionType.GreaterThan:
- return ">";
- case ExpressionType.GreaterThanOrEqual:
- return ">=";
- case ExpressionType.Equal:
- return "=";
- case ExpressionType.NotEqual:
- return "!=";
- case ExpressionType.Coalesce:
- return "??";
- case ExpressionType.RightShift:
- return ">>";
- case ExpressionType.LeftShift:
- return "<<";
- case ExpressionType.ExclusiveOr:
- return "^";
- default:
- return null;
- }
- }
- protected override Expression VisitBinary(BinaryExpression b)
- {
- this.Write("(");
- if (b.NodeType == ExpressionType.Power)
- {
- this.Write("POWER(");
- this.VisitValue(b.Left);
- this.Write(", ");
- this.VisitValue(b.Right);
- this.Write(")");
- // return b;
- }
- else if (b.NodeType == ExpressionType.Coalesce)
- {
- this.Write("COALESCE(");
- this.VisitValue(b.Left);
- this.Write(", ");
- Expression right = b.Right;
- while (right.NodeType == ExpressionType.Coalesce)
- {
- BinaryExpression rb = (BinaryExpression)right;
- this.VisitValue(rb.Left);
- this.Write(", ");
- right = rb.Right;
- }
- this.VisitValue(right);
- this.Write(")");
- // return b;
- }
- else if (b.NodeType == ExpressionType.LeftShift)
- {
- this.Write("(");
- this.VisitValue(b.Left);
- this.Write(" * POWER(2, ");
- this.VisitValue(b.Right);
- this.Write("))");
- // return b;
- }
- else if (b.NodeType == ExpressionType.RightShift)
- {
- this.Write("(");
- this.VisitValue(b.Left);
- this.Write(" / POWER(2, ");
- this.VisitValue(b.Right);
- this.Write("))");
- //return b;
- }
- else
- {
- //if (b.Left is MemberExpression)
- //{
- // if (((MemberExpression)b.Left).Type == typeof(bool) && (_bizRead == SqlType.Where))
- // {
- // this.Write("(" + ((MemberExpression)b.Left).Member.Name + " = 1)");
- // }
- // else
- // {
- // this.Visit(b.Left);
- // }
- //}
- //else
- //{
- this.Visit(b.Left);
- //}
- this.Write(" ");
- this.Write(GetOperator(b.NodeType));
- this.Write(" ");
- this.Visit(b.Right);
- }
- this.Write(")");
- return b;
- }
- protected override Expression VisitUnary(UnaryExpression u)
- {
- switch (u.NodeType)
- {
- case ExpressionType.Convert:
- case ExpressionType.ConvertChecked:
- //this.Write("((");
- //this.Write(this.GetTypeName(u.Type));
- // this.Write(")");
- this.Visit(u.Operand);
- //this.Write(")");
- break;
- case ExpressionType.ArrayLength:
- this.Visit(u.Operand);
- this.Write(".Length");
- break;
- case ExpressionType.Quote:
- this.Visit(u.Operand);
- break;
- case ExpressionType.TypeAs:
- this.Visit(u.Operand);
- this.Write(" as ");
- this.Write(this.GetTypeName(u.Type));
- break;
- case ExpressionType.UnaryPlus:
- this.Visit(u.Operand);
- break;
- default:
- this.Write(this.GetOperator(u.NodeType));
- this.Visit(u.Operand);
- break;
- }
- return u;
- }
- protected virtual string GetTypeName(Type type)
- {
- string name = type.Name;
- name = name.Replace('+', '.');
- int iGeneneric = name.IndexOf('`');
- if (iGeneneric > 0)
- {
- name = name.Substring(0, iGeneneric);
- }
- if (type.IsGenericType || type.IsGenericTypeDefinition)
- {
- StringBuilder sb = new StringBuilder();
- sb.Append(name);
- sb.Append("<");
- var args = type.GetGenericArguments();
- for (int i = 0, n = args.Length; i < n; i++)
- {
- if (i > 0)
- {
- sb.Append(",");
- }
- if (type.IsGenericType)
- {
- sb.Append(this.GetTypeName(args[i]));
- }
- }
- sb.Append(">");
- name = sb.ToString();
- }
- return name;
- }
- protected override Expression VisitConditional(ConditionalExpression c)
- {
- this.Visit(c.Test);
- this.WriteLine(Indentation.Inner);
- this.Write("? ");
- this.Visit(c.IfTrue);
- this.WriteLine(Indentation.Same);
- this.Write(": ");
- this.Visit(c.IfFalse);
- this.Indent(Indentation.Outer);
- return c;
- }
- protected override IEnumerable<MemberBinding> VisitBindingList(ReadOnlyCollection<MemberBinding> original)
- {
- for (int i = 0, n = original.Count; i < n; i++)
- {
- this.VisitBinding(original[i]);
- if (i < n - 1)
- {
- this.Write(",");
- this.WriteLine(Indentation.Same);
- }
- }
- return original;
- }
- private static readonly char[] Special = new char[] { '\n', '\n', '\\' };
- protected override Expression VisitConstant(ConstantExpression c)
- {
- if (c.Value == null)
- {
- this.Write("null");
- }
- else if (c.Type == typeof(DateTime) || c.Type == typeof(Guid))
- {
- this.Write("'");//new DateTime(\"
- this.Write(c.Value.ToString());
- this.Write("'");//\"
- }
- else
- {
- switch (Type.GetTypeCode(c.Value.GetType()))
- {
- case TypeCode.Boolean:
- //if (c.NodeType != ExpressionType.Constant)
- this.Write(((bool)c.Value) ? "1" : "0");
- break;
- case TypeCode.Single:
- case TypeCode.Double:
- string str = c.Value.ToString();
- if (!str.Contains('.'))
- {
- str += ".0";
- }
- this.Write(str);
- break;
- case TypeCode.DateTime:
- this.Write("'");//new DateTime(\"
- this.Write(c.Value.ToString());
- this.Write("'");//\"
- break;
- case TypeCode.String:
- this.Write("'");
- this.Write(c.Value.ToString().Replace("'", "\""));
- this.Write("'");
- break;
- default:
- this.Write(c.Value.ToString());
- break;
- }
- }
- return c;
- }
- protected override ElementInit VisitElementInitializer(ElementInit initializer)
- {
- if (initializer.Arguments.Count > 1)
- {
- this.Write("{");
- for (int i = 0, n = initializer.Arguments.Count; i < n; i++)
- {
- this.Visit(initializer.Arguments[i]);
- if (i < n - 1)
- {
- this.Write(", ");
- }
- }
- this.Write("}");
- }
- else
- {
- this.Visit(initializer.Arguments[0]);
- }
- return initializer;
- }
- protected override IEnumerable<ElementInit> VisitElementInitializerList(ReadOnlyCollection<ElementInit> original)
- {
- for (int i = 0, n = original.Count; i < n; i++)
- {
- this.VisitElementInitializer(original[i]);
- if (i < n - 1)
- {
- this.Write(",");
- this.WriteLine(Indentation.Same);
- }
- }
- return original;
- }
- protected override ReadOnlyCollection<Expression> VisitExpressionList(ReadOnlyCollection<Expression> original)
- {
- for (int i = 0, n = original.Count; i < n; i++)
- {
- this.Visit(original[i]);
- //if (i < n - 1)
- //{
- // this.Write(",");
- // this.WriteLine(Indentation.Same);
- //}
- }
- return original;
- }
- protected override Expression VisitInvocation(InvocationExpression iv)
- {
- this.Write("Invoke(");
- this.WriteLine(Indentation.Inner);
- this.VisitExpressionList(iv.Arguments);
- this.Write(", ");
- this.WriteLine(Indentation.Same);
- this.Visit(iv.Expression);
- this.WriteLine(Indentation.Same);
- this.Write(")");
- this.Indent(Indentation.Outer);
- return iv;
- }
- protected override Expression VisitLambda(LambdaExpression lambda)
- {
- if (lambda.Body.NodeType == ExpressionType.MemberAccess)
- {
- if (((MemberExpression)lambda.Body).Type == typeof(bool) && (_bizRead == SqlType.Where))
- {
- this.Write(((MemberExpression)lambda.Body).Member.Name + " = 1");
- }
- else
- {
- this.Visit(lambda.Body);
- return lambda;
- }
- }
- this.Visit(lambda.Body);
- return lambda;
- }
- protected override Expression VisitListInit(ListInitExpression init)
- {
- this.Visit(init.NewExpression);
- this.Write(" {");
- this.WriteLine(Indentation.Inner);
- this.VisitElementInitializerList(init.Initializers);
- this.WriteLine(Indentation.Outer);
- this.Write("}");
- return init;
- }
- protected override Expression VisitMemberAccess(MemberExpression m)
- {
- this.Write(m.Member.Name);
- // string t = m.Update.GetType().Name;
- if (m.Member.DeclaringType == typeof(string))
- {
- switch (m.Member.Name)
- {
- case "Length":
- this.Write("LEN(");
- this.Visit(m.Expression);
- this.Write(")");
- return m;
- }
- }
- else if (m.Member.DeclaringType == typeof(DateTime) || m.Member.DeclaringType == typeof(DateTimeOffset))
- {
- switch (m.Member.Name)
- {
- case "Day":
- this.Write("DAY(");
- this.Visit(m.Expression);
- this.Write(")");
- return m;
- case "Month":
- this.Write("MONTH(");
- this.Visit(m.Expression);
- this.Write(")");
- return m;
- case "Year":
- this.Write("YEAR(");
- this.Visit(m.Expression);
- this.Write(")");
- return m;
- case "Hour":
- this.Write("DATEPART(hour, ");
- this.Visit(m.Expression);
- this.Write(")");
- return m;
- case "Minute":
- this.Write("DATEPART(minute, ");
- this.Visit(m.Expression);
- this.Write(")");
- return m;
- case "Second":
- this.Write("DATEPART(second, ");
- this.Visit(m.Expression);
- this.Write(")");
- return m;
- case "Millisecond":
- this.Write("DATEPART(millisecond, ");
- this.Visit(m.Expression);
- this.Write(")");
- return m;
- case "DayOfWeek":
- this.Write("(DATEPART(weekday, ");
- this.Visit(m.Expression);
- this.Write(") - 1)");
- return m;
- case "DayOfYear":
- this.Write("(DATEPART(dayofyear, ");
- this.Visit(m.Expression);
- this.Write(") - 1)");
- return m;
- }
- }
- return base.VisitMemberAccess(m);
- }
- protected override MemberAssignment VisitMemberAssignment(MemberAssignment assignment)
- {
- this.Write(assignment.Member.Name);
- this.Write(" = ");
- this.Visit(assignment.Expression);
- return assignment;
- }
- protected override Expression VisitMemberInit(MemberInitExpression init)
- {
- this.Visit(init.NewExpression);
- this.Write(" {");
- this.WriteLine(Indentation.Inner);
- this.VisitBindingList(init.Bindings);
- this.WriteLine(Indentation.Outer);
- this.Write("}");
- return init;
- }
- protected override MemberListBinding VisitMemberListBinding(MemberListBinding binding)
- {
- this.Write(binding.Member.Name);
- this.Write(" = {");
- this.WriteLine(Indentation.Inner);
- this.VisitElementInitializerList(binding.Initializers);
- this.WriteLine(Indentation.Outer);
- this.Write("}");
- return binding;
- }
- protected override MemberMemberBinding VisitMemberMemberBinding(MemberMemberBinding binding)
- {
- this.Write(binding.Member.Name);
- this.Write(" = {");
- this.WriteLine(Indentation.Inner);
- this.VisitBindingList(binding.Bindings);
- this.WriteLine(Indentation.Outer);
- this.Write("}");
- return binding;
- }
- protected override Expression VisitMethodCall(MethodCallExpression m)
- {
- //if (m.Object != null)
- //{
- // this.Visit(m.Object);
- //}
- //else
- //{
- // this.Write(this.GetTypeName(m.Method.DeclaringType));
- //}
- string bizname = m.Method.Name.ToLower();
- //if (this.GetTypeName(m.Method.DeclaringType).ToLower() != "queryable")
- //{
- // this.Write(".");
- //}
- // this.Write(m.Method.Name);
- switch (bizname)
- {
- case "where":
- if (_bizRead == SqlType.Where)
- {
- AiWhereTime = AiWhereTime + 1;
- }
- else
- {
- this.Visit(m.Arguments[0]);
- return m;
- }
- break;
- case "orderby":
- case "orderbydescending":
- case "thenbydescending":
- case "thenby":
- if (_bizRead == SqlType.Order)
- {
- AiOrderTime = AiOrderTime + 1;
- }
- else
- {
- this.Visit(m.Arguments[0]);
- return m;
- }
- break;
- }
- //bool bizisw = bizname == "where" || bizname == "orderby" || bizname == "thenby";
- //if (bizisw)
- //{
- // this.Write(m.Method.Name);
- //}
- //this.VisitExpressionList(m.Arguments);
- if (m.Arguments.Count > 1)
- {
- }
- // this.WriteLine(Indentation.Outer);
- //if (bizisw)
- //{
- // this.Write(")");
- //}
- //return m;
- #region
- if (m.Method.DeclaringType == typeof(string))
- {
- switch (m.Method.Name)
- {
- case "StartsWith":
- this.Write("(");
- this.Visit(m.Object);
- this.Write(" LIKE '");
- this.Visit(m.Arguments[0]);
- this.Write("%')");
- return m;
- case "EndsWith":
- this.Write("(");
- this.Visit(m.Object);
- this.Write(" LIKE '%");
- this.Visit(m.Arguments[0]);
- this.Write("')");
- return m;
- case "Contains":
- this.Write("(");
- this.Visit(m.Object);
- this.Write(" LIKE '%");
- this.Visit(m.Arguments[0]);
- this.Write("%')");
- return m;
- case "Concat":
- IList<Expression> args = m.Arguments;
- if (args.Count == 1 && args[0].NodeType == ExpressionType.NewArrayInit)
- {
- args = ((NewArrayExpression)args[0]).Expressions;
- }
- for (int i = 0, n = args.Count; i < n; i++)
- {
- if (i > 0) this.Write(" + ");
- this.Visit(args[i]);
- }
- return m;
- case "IsNullOrEmpty":
- this.Write("(");
- this.Visit(m.Arguments[0]);
- this.Write(" IS NULL OR ");
- this.Visit(m.Arguments[0]);
- this.Write(" = '')");
- return m;
- case "ToUpper":
- this.Write("UPPER(");
- this.Visit(m.Object);
- this.Write(")");
- return m;
- case "ToLower":
- this.Write("LOWER(");
- this.Visit(m.Object);
- this.Write(")");
- return m;
- case "Replace":
- this.Write("REPLACE(");
- this.Visit(m.Object);
- this.Write(", ");
- this.Visit(m.Arguments[0]);
- this.Write(", ");
- this.Visit(m.Arguments[1]);
- this.Write(")");
- return m;
- case "Substring":
- this.Write("SUBSTRING(");
- this.Visit(m.Object);
- this.Write(", ");
- this.Visit(m.Arguments[0]);
- this.Write(" + 1, ");
- if (m.Arguments.Count == 2)
- {
- this.Visit(m.Arguments[1]);
- }
- else
- {
- this.Write("8000");
- }
- this.Write(")");
- return m;
- case "Remove":
- this.Write("STUFF(");
- this.Visit(m.Object);
- this.Write(", ");
- this.Visit(m.Arguments[0]);
- this.Write(" + 1, ");
- if (m.Arguments.Count == 2)
- {
- this.Visit(m.Arguments[1]);
- }
- else
- {
- this.Write("8000");
- }
- this.Write(", '')");
- return m;
- case "IndexOf":
- this.Write("(CHARINDEX(");
- this.Visit(m.Arguments[0]);
- this.Write(", ");
- this.Visit(m.Object);
- if (m.Arguments.Count == 2 && m.Arguments[1].Type == typeof(int))
- {
- this.Write(", ");
- this.Visit(m.Arguments[1]);
- this.Write(" + 1");
- }
- this.Write(") - 1)");
- return m;
- case "Trim":
- this.Write("RTRIM(LTRIM(");
- this.Visit(m.Object);
- this.Write("))");
- return m;
- }
- }
- else if (m.Method.DeclaringType == typeof(DateTime))
- {
- switch (m.Method.Name)
- {
- case "op_Subtract":
- if (m.Arguments[1].Type == typeof(DateTime))
- {
- this.Write("DATEDIFF(");
- this.Visit(m.Arguments[0]);
- this.Write(", ");
- this.Visit(m.Arguments[1]);
- this.Write(")");
- return m;
- }
- break;
- case "AddYears":
- this.Write("DATEADD(YYYY,");
- this.Visit(m.Arguments[0]);
- this.Write(",");
- this.Visit(m.Object);
- this.Write(")");
- return m;
- case "AddMonths":
- this.Write("DATEADD(MM,");
- this.Visit(m.Arguments[0]);
- this.Write(",");
- this.Visit(m.Object);
- this.Write(")");
- return m;
- case "AddDays":
- this.Write("DATEADD(DAY,");
- this.Visit(m.Arguments[0]);
- this.Write(",");
- this.Visit(m.Object);
- this.Write(")");
- return m;
- case "AddHours":
- this.Write("DATEADD(HH,");
- this.Visit(m.Arguments[0]);
- this.Write(",");
- this.Visit(m.Object);
- this.Write(")");
- return m;
- case "AddMinutes":
- this.Write("DATEADD(MI,");
- this.Visit(m.Arguments[0]);
- this.Write(",");
- this.Visit(m.Object);
- this.Write(")");
- return m;
- case "AddSeconds":
- this.Write("DATEADD(SS,");
- this.Visit(m.Arguments[0]);
- this.Write(",");
- this.Visit(m.Object);
- this.Write(")");
- return m;
- case "AddMilliseconds":
- this.Write("DATEADD(MS,");
- this.Visit(m.Arguments[0]);
- this.Write(",");
- this.Visit(m.Object);
- this.Write(")");
- return m;
- }
- }
- else if (m.Method.DeclaringType == typeof(Decimal))
- {
- switch (m.Method.Name)
- {
- case "Add":
- case "Subtract":
- case "Multiply":
- case "Divide":
- case "Remainder":
- this.Write("(");
- this.VisitValue(m.Arguments[0]);
- this.Write(" ");
- this.Write(GetOperator(m.Method.Name));
- this.Write(" ");
- this.VisitValue(m.Arguments[1]);
- this.Write(")");
- return m;
- case "Negate":
- this.Write("-");
- this.Visit(m.Arguments[0]);
- this.Write("");
- return m;
- case "Ceiling":
- case "Floor":
- this.Write(m.Method.Name.ToUpper());
- this.Write("(");
- this.Visit(m.Arguments[0]);
- this.Write(")");
- return m;
- case "Round":
- if (m.Arguments.Count == 1)
- {
- this.Write("ROUND(");
- this.Visit(m.Arguments[0]);
- this.Write(", 0)");
- return m;
- }
- else if (m.Arguments.Count == 2 && m.Arguments[1].Type == typeof(int))
- {
- this.Write("ROUND(");
- this.Visit(m.Arguments[0]);
- this.Write(", ");
- this.Visit(m.Arguments[1]);
- this.Write(")");
- return m;
- }
- break;
- case "Truncate":
- this.Write("ROUND(");
- this.Visit(m.Arguments[0]);
- this.Write(", 0, 1)");
- return m;
- }
- }
- else if (m.Method.DeclaringType == typeof(Math))
- {
- switch (m.Method.Name)
- {
- case "Abs":
- case "Acos":
- case "Asin":
- case "Atan":
- case "Cos":
- case "Exp":
- case "Log10":
- case "Sin":
- case "Tan":
- case "Sqrt":
- case "Sign":
- case "Ceiling":
- case "Floor":
- this.Write(m.Method.Name.ToUpper());
- this.Write("(");
- this.Visit(m.Arguments[0]);
- this.Write(")");
- return m;
- case "Atan2":
- this.Write("ATN2(");
- this.Visit(m.Arguments[0]);
- this.Write(", ");
- this.Visit(m.Arguments[1]);
- this.Write(")");
- return m;
- case "Log":
- if (m.Arguments.Count == 1)
- {
- goto case "Log10";
- }
- break;
- case "Pow":
- this.Write("POWER(");
- this.Visit(m.Arguments[0]);
- this.Write(", ");
- this.Visit(m.Arguments[1]);
- this.Write(")");
- return m;
- case "Round":
- if (m.Arguments.Count == 1)
- {
- this.Write("ROUND(");
- this.Visit(m.Arguments[0]);
- this.Write(", 0)");
- return m;
- }
- else if (m.Arguments.Count == 2 && m.Arguments[1].Type == typeof(int))
- {
- this.Write("ROUND(");
- this.Visit(m.Arguments[0]);
- this.Write(", ");
- this.Visit(m.Arguments[1]);
- this.Write(")");
- return m;
- }
- break;
- case "Truncate":
- this.Write("ROUND(");
- this.Visit(m.Arguments[0]);
- this.Write(", 0, 1)");
- return m;
- }
- }
- if (m.Method.Name == "ToString")
- {
- if (m.Object.Type != typeof(string))
- {
- this.Write("CONVERT(NVARCHAR, ");
- this.Visit(m.Object);
- this.Write(")");
- }
- else
- {
- this.Visit(m.Object);
- }
- return m;
- }
- else if (!m.Method.IsStatic && m.Method.Name == "CompareTo" && m.Method.ReturnType == typeof(int) && m.Arguments.Count == 1)
- {
- this.Write("(CASE WHEN ");
- this.Visit(m.Object);
- this.Write(" = ");
- this.Visit(m.Arguments[0]);
- this.Write(" THEN 0 WHEN ");
- this.Visit(m.Object);
- this.Write(" < ");
- this.Visit(m.Arguments[0]);
- this.Write(" THEN -1 ELSE 1 END)");
- return m;
- }
- else if (m.Method.IsStatic && m.Method.Name == "Compare" && m.Method.ReturnType == typeof(int) && m.Arguments.Count == 2)
- {
- this.Write("(CASE WHEN ");
- this.Visit(m.Arguments[0]);
- this.Write(" = ");
- this.Visit(m.Arguments[1]);
- this.Write(" THEN 0 WHEN ");
- this.Visit(m.Arguments[0]);
- this.Write(" < ");
- this.Visit(m.Arguments[1]);
- this.Write(" THEN -1 ELSE 1 END)");
- return m;
- }
- #endregion
- if (m.Arguments.Count > 1)
- {
- this.WriteLine(Indentation.Outer);
- }
- base.VisitMethodCall(m);
- if (m.Arguments.Count > 1)
- {
- switch (bizname)
- {
- case "orderbydescending":
- case "thenbydescending": _aiOrdeRsult = _aiOrdeRsult + " Desc"; break;
- }
- if (_bizRead == SqlType.Order)
- {
- _aiOrdeRsult = _aiOrdeRsult + ",";
- }
- else
- {
- _aiWhereResult = _aiWhereResult + " And ";
- }
- this.WriteLine(Indentation.Outer);
- }
- return m;
- }
- private Expression VisitValue(Expression expr)
- {
- if (IsPredicate(expr))
- {
- this.Write("CASE WHEN (");
- this.Visit(expr);
- this.Write(") THEN 1 ELSE 0 END");
- return expr;
- }
- return expr;
- }
- protected override NewExpression VisitNew(NewExpression nex)
- {
- //this.Write("new ");
- //this.Write(this.GetTypeName(nex.Constructor.DeclaringType));
- //this.Write("(");
- //if (nex.Arguments.Count > 1)
- // this.WriteLine(Indentation.Inner);
- //this.VisitExpressionList(nex.Arguments);
- //if (nex.Arguments.Count > 1)
- // this.WriteLine(Indentation.Outer);
- //this.Write(")");
- //return nex;
- if (nex.Constructor.DeclaringType == typeof(DateTime))
- {
- if (nex.Arguments.Count == 3)
- {
- this.Write("Convert(DateTime, ");
- this.Write("Convert(nvarchar, ");
- this.Visit(nex.Arguments[0]);
- this.Write(") + '/' + ");
- this.Write("Convert(nvarchar, ");
- this.Visit(nex.Arguments[1]);
- this.Write(") + '/' + ");
- this.Write("Convert(nvarchar, ");
- this.Visit(nex.Arguments[2]);
- this.Write("))");
- return nex;
- }
- else if (nex.Arguments.Count == 6)
- {
- this.Write("Convert(DateTime, ");
- this.Write("Convert(nvarchar, ");
- this.Visit(nex.Arguments[0]);
- this.Write(") + '/' + ");
- this.Write("Convert(nvarchar, ");
- this.Visit(nex.Arguments[1]);
- this.Write(") + '/' + ");
- this.Write("Convert(nvarchar, ");
- this.Visit(nex.Arguments[2]);
- this.Write(") + ' ' + ");
- this.Write("Convert(nvarchar, ");
- this.Visit(nex.Arguments[3]);
- this.Write(") + ':' + ");
- this.Write("Convert(nvarchar, ");
- this.Visit(nex.Arguments[4]);
- this.Write(") + ':' + ");
- this.Write("Convert(nvarchar, ");
- this.Visit(nex.Arguments[5]);
- this.Write("))");
- return nex;
- }
- }
- return base.VisitNew(nex);
- }
- protected override Expression VisitNewArray(NewArrayExpression na)
- {
- this.Write("new ");
- this.Write(this.GetTypeName(TypeHelper.GetElementType(na.Type)));
- this.Write("[] {");
- if (na.Expressions.Count > 1)
- this.WriteLine(Indentation.Inner);
- this.VisitExpressionList(na.Expressions);
- if (na.Expressions.Count > 1)
- this.WriteLine(Indentation.Outer);
- this.Write("}");
- return na;
- }
- protected override Expression VisitParameter(ParameterExpression p)
- {
- // this.Write(p.Name);
- return p;
- }
- protected override Expression VisitTypeIs(TypeBinaryExpression b)
- {
- this.Visit(b.Expression);
- this.Write(" is ");
- this.Write(this.GetTypeName(b.TypeOperand));
- return b;
- }
- protected override Expression VisitUnknown(Expression expression)
- {
- this.Write(expression.ToString());
- return expression;
- }
- protected virtual bool IsBoolean(Type type)
- {
- return type == typeof(bool) || type == typeof(bool?);
- }
- protected virtual bool IsPredicate(Expression expr)
- {
- switch (expr.NodeType)
- {
- case ExpressionType.And:
- case ExpressionType.AndAlso:
- case ExpressionType.Or:
- case ExpressionType.OrElse:
- return IsBoolean(((BinaryExpression)expr).Type);
- case ExpressionType.Not:
- return IsBoolean(((UnaryExpression)expr).Type);
- case ExpressionType.Equal:
- case ExpressionType.NotEqual:
- case ExpressionType.LessThan:
- case ExpressionType.LessThanOrEqual:
- case ExpressionType.GreaterThan:
- case ExpressionType.GreaterThanOrEqual:
- case (ExpressionType)DbExpressionType.IsNull:
- case (ExpressionType)DbExpressionType.Between:
- case (ExpressionType)DbExpressionType.Exists:
- case (ExpressionType)DbExpressionType.In:
- return true;
- case ExpressionType.Call:
- return IsBoolean(((MethodCallExpression)expr).Type);
- default:
- return false;
- }
- }
- internal enum DbExpressionType
- {
- Table = 1000, // make sure these don't overlap with ExpressionType
- ClientJoin,
- Column,
- Select,
- Projection,
- Entity,
- Join,
- Aggregate,
- Scalar,
- Exists,
- In,
- Grouping,
- AggregateSubquery,
- IsNull,
- Between,
- RowCount,
- NamedValue,
- OuterJoined,
- Insert,
- Update,
- Delete,
- Batch,
- Function,
- Block,
- If,
- Declaration,
- Variable
- }
- protected virtual string GetOperator(string methodName)
- {
- switch (methodName)
- {
- case "Add": return "+";
- case "Subtract": return "-";
- case "Multiply": return "*";
- case "Divide": return "/";
- case "Negate": return "-";
- case "Remainder": return "%";
- default: return null;
- }
- }
- protected virtual string GetOperator(UnaryExpression u)
- {
- switch (u.NodeType)
- {
- case ExpressionType.Negate:
- case ExpressionType.NegateChecked:
- return "-";
- case ExpressionType.UnaryPlus:
- return "+";
- case ExpressionType.Not:
- return IsBoolean(u.Operand.Type) ? "NOT" : "~";
- default:
- return "";
- }
- }
- protected virtual string GetOperator(BinaryExpression b)
- {
- switch (b.NodeType)
- {
- case ExpressionType.And:
- case ExpressionType.AndAlso:
- return (IsBoolean(b.Left.Type)) ? "AND" : "&";
- case ExpressionType.Or:
- case ExpressionType.OrElse:
- return (IsBoolean(b.Left.Type) ? "OR" : "|");
- case ExpressionType.Equal:
- return "=";
- case ExpressionType.NotEqual:
- return "<>";
- case ExpressionType.LessThan:
- return "<";
- case ExpressionType.LessThanOrEqual:
- return "<=";
- case ExpressionType.GreaterThan:
- return ">";
- case ExpressionType.GreaterThanOrEqual:
- return ">=";
- case ExpressionType.Add:
- case ExpressionType.AddChecked:
- return "+";
- case ExpressionType.Subtract:
- case ExpressionType.SubtractChecked:
- return "-";
- case ExpressionType.Multiply:
- case ExpressionType.MultiplyChecked:
- return "*";
- case ExpressionType.Divide:
- return "/";
- case ExpressionType.Modulo:
- return "%";
- case ExpressionType.ExclusiveOr:
- return "^";
- case ExpressionType.LeftShift:
- return "<<";
- case ExpressionType.RightShift:
- return ">>";
- default:
- return "";
- }
- }
-
- }
- }
|