private LambdaExpression Parameterize(Expression query, out object[] arguments) { IQueryProvider provider = this.FindProvider(query); if (provider == null) { throw new ArgumentException("Cannot deduce query provider from query"); } var ep = provider as IEntityProvider; Func <Expression, bool> fn = ep != null ? (Func <Expression, bool>)ep.CanBeEvaluatedLocally : null; List <ParameterExpression> parameters = new List <ParameterExpression>(); List <object> values = new List <object>(); var body = PartialEvaluator.Eval(query, fn, c => { bool isQueryRoot = c.Value is IQueryable; if (!isQueryRoot && ep != null && !ep.CanBeParameter(c)) { return(c); } var p = Expression.Parameter(c.Type, "p" + parameters.Count); parameters.Add(p); values.Add(c.Value); // if query root then parameterize but don't replace in the tree if (isQueryRoot) { return(c); } return(p); }); if (body.Type != typeof(object)) { body = Expression.Convert(body, typeof(object)); } arguments = values.ToArray(); if (arguments.Length < 5) { return(Expression.Lambda(body, parameters.ToArray())); } else { arguments = new object[] { arguments }; return(ParameterToObjectArrayAccessorRewriter.Rewrite(body, parameters)); } }
public static LambdaExpression Rewrite(Expression body, IReadOnlyList <ParameterExpression> parameters) { var visitor = new ParameterToObjectArrayAccessorRewriter(parameters); return(Expression.Lambda(visitor.Visit(body), visitor.array)); }