模拟 IDbSet GetEnumerator 时 LINQ 失败

Hol*_*roe 2 c# linq entity-framework moq

我试图模拟 System.Data.Entity.IDbSet 以使其返回一些数据(在这种情况下只是一个空集合):

var mock = new Mock<IDbSet<Setting>>();
mock.Setup(x => x.GetEnumerator()).Returns(Enumerable.Empty<Setting>().GetEnumerator());
var myEnumerator = mock.Object.GetEnumerator();
var count = mock.Object.Count();
Run Code Online (Sandbox Code Playgroud)

这里 myEnumerator 将有一个非空值,正如预期的那样,但 mock.Object.Count() 将抛出一个 ArgumentNullException

System.ArgumentNullException : Value cannot be null.
Parameter name: arguments
   at System.Linq.Expressions.Expression.RequiresCanRead(Expression expression, String paramName)
   at System.Linq.Expressions.Expression.ValidateOneArgument(MethodBase method, ExpressionType nodeKind, Expression arg, ParameterInfo pi)
   at System.Linq.Expressions.Expression.ValidateArgumentTypes(MethodBase method, ExpressionType nodeKind, ref ReadOnlyCollection`1 arguments)
   at System.Linq.Expressions.Expression.Call(Expression instance, MethodInfo method, IEnumerable`1 arguments)
   at System.Linq.Queryable.Count(IQueryable`1 source)
Run Code Online (Sandbox Code Playgroud)

如果我用 IEnumerable 替换 IDbSet,它会按预期工作(计数为零)。为什么没有像我期望的那样模拟 IDbSet?

编辑

看起来 LINQ 尝试使用 IQueryable 接口,因为它有效:

var mock = new Mock<IDbSet<Setting>>();
var myQueryable = Enumerable.Empty<Setting>().AsQueryable();
mock.Setup(m => m.Provider).Returns(myQueryable.Provider);
mock.Setup(m => m.Expression).Returns(myQueryable.Expression);
mock.Setup(m => m.GetEnumerator()).Returns(myQueryable.GetEnumerator());
var count = mock.Object.Count();
Run Code Online (Sandbox Code Playgroud)

Hol*_*roe 5

事实证明,LINQ 具有 IQueryable 的显式实现:

    public static int Count<TSource>(this IQueryable<TSource> source) { 
        if (source == null)
            throw Error.ArgumentNull("source"); 
        return source.Provider.Execute<int>( 
            Expression.Call(
                null, 
                ((MethodInfo)MethodBase.GetCurrentMethod()).MakeGenericMethod(typeof(TSource)),
                new Expression[] { source.Expression }
                ));
    }
Run Code Online (Sandbox Code Playgroud)

实际上,LINQ 表达式如何将表达式转换为实体框架查询,这很有意义。

所以我还必须模拟提供者和表达式:

var mock = new Mock<IDbSet<Setting>>();
var myQueryable = Enumerable.Empty<Setting>().AsQueryable();
mock.Setup(m => m.Provider).Returns(myQueryable.Provider);
mock.Setup(m => m.Expression).Returns(myQueryable.Expression);
mock.Setup(m => m.GetEnumerator()).Returns(myQueryable.GetEnumerator());
var count = mock.Object.Count();
Run Code Online (Sandbox Code Playgroud)