/QueryInterceptor

Provides a generic way to intercept IQueryable expression trees using custom expression visitors.

Primary LanguageC#MIT LicenseMIT

The problem

Normally when you're trying to modify expression trees you need to write an expression visitor. When trying to plug your expression visitor into an IQuerable's expression tree, you need to write a linq provider (yikes).

Implementation based on http://stackoverflow.com/questions/1839901/how-to-wrap-entity-framework-to-intercept-the-linq-expression-just-before-executi

The solution

QueryInterceptor introduces one extension method on IQueryable (InterceptWith) that lets you plug in arbitrary expression visitors.

namespace QueryInterceptor {
    public static class QueryableExtensions {
        public static IQueryable<T> InterceptWith<T>(this IQueryable<T> source, params ExpressionVisitor[] visitors);
    }
}

Basic Example

The example below uses an expression visitor that changes == to != anywhere in the expression tree.

public class EqualsToNotEqualsVisitor : ExpressionVisitor {
    protected override Expression VisitBinary(BinaryExpression node) {
        if (node.NodeType == ExpressionType.Equal) {
            // Change == to !=
            return Expression.NotEqual(node.Left, node.Right);
        }
        return base.VisitBinary(node);
    }
}

class Program {
    static void Main(string[] args) {
        var query = from n in Enumerable.Range(0, 10).AsQueryable()
                    where n % 2 == 0
                    select n;

        // Print even numbers
        foreach (var item in query) {
            Console.WriteLine(item);
        }

        Console.WriteLine();

        // Print odd numbers
        var visitor = new EqualsToNotEqualsVisitor();
        foreach (var item in query.InterceptWith(visitor)) {
            Console.WriteLine(item);
        }
    }
}