AutoMapper无法将TestDbAsyncEnumerable强制转换为IQueryable

Dyl*_*sil 12 c# entity-framework automapper

我已经从https://msdn.microsoft.com/en-us/library/dn314429(v=vs.113).aspx实现了TestDbAsync假货,我希望能够使用AutoMapper投影到其他类型之前调用Async EF方法(ToListAsync,CountAsync等).

我在ProjectionExpression.To中得到了一个强制转换异常

抛出异常的示例代码.

_userRepository.GetAll().OrderBy(x => x.Id).ProjectTo<User>.ToListAsync();
Run Code Online (Sandbox Code Playgroud)

这在非测试场景中工作正常,但是当我使用TestDbAsyncEnumerable模拟DbSet时,我得到了

: Unable to cast object of type 'Namespace.TestDbAsyncEnumerable`1[UserEntity]' to type 'System.Linq.IQueryable`1[User]'.
Run Code Online (Sandbox Code Playgroud)

现在为了解决这个问题,我在调用Async EF扩展后必须使用ProjectTo.有没有办法在EF扩展之前保持ProjectTo调用?

参考代码:

public class TestDbAsyncEnumerable<T> : EnumerableQuery<T>, IDbAsyncEnumerable<T>, IQueryable<T>
{
    public TestDbAsyncEnumerable(IEnumerable<T> enumerable)
        : base(enumerable)
    { }

    public TestDbAsyncEnumerable(Expression expression)
        : base(expression)
    { }

    public IDbAsyncEnumerator<T> GetAsyncEnumerator()
    {
        return new TestDbAsyncEnumerator<T>(this.AsEnumerable().GetEnumerator());
    }

    IDbAsyncEnumerator IDbAsyncEnumerable.GetAsyncEnumerator()
    {
        return GetAsyncEnumerator();
    }

    IQueryProvider IQueryable.Provider => new TestDbAsyncQueryProvider<T>(this);
}

public static Mock<DbSet<T>> ToAsyncDbSetMock<T>(this IEnumerable<T> source)
        where T : class
    {

        var data = source.AsQueryable();

        var mockSet = new Mock<DbSet<T>>();

        mockSet.As<IDbAsyncEnumerable<T>>()
            .Setup(m => m.GetAsyncEnumerator())
            .Returns(new TestDbAsyncEnumerator<T>(data.GetEnumerator()));

        mockSet.As<IQueryable<T>>()
            .Setup(m => m.Provider)
            .Returns(new TestDbAsyncQueryProvider<T>(data.Provider));

        mockSet.As<IQueryable<T>>().Setup(m => m.Expression).Returns(data.Expression);
        mockSet.As<IQueryable<T>>().Setup(m => m.ElementType).Returns(data.ElementType);
        mockSet.As<IQueryable<T>>().Setup(m => m.GetEnumerator()).Returns(data.GetEnumerator());

        return mockSet;
    }
Run Code Online (Sandbox Code Playgroud)

Mas*_*uto 19

编辑您的内容TestDbAsyncQueryProvider<>.CreateQuery(),使其重新运用所传递的表达式的正确类型ProjectTo<>.

这是我的示例实现.

public IQueryable CreateQuery(Expression expression)
{
    switch (expression)
    {
        case MethodCallExpression m:
            {
                var resultType = m.Method.ReturnType; // it shoud be IQueryable<T>
                var tElement = resultType.GetGenericArguments()[0];
                var queryType = typeof(TestDbAsyncEnumerable<>).MakeGenericType(tElement);
                return (IQueryable)Activator.CreateInstance(queryType, expression);
            }
    }
    return new TestDbAsyncEnumerable<TEntity>(expression);
}
Run Code Online (Sandbox Code Playgroud)

https://gist.github.com/masaedw/95ab972f8181de6bbe48a20ffe9be113

我也写了单元测试.它正在发挥作用.

https://github.com/masaedw/AutoMapper/blob/TestDbAsync/src/IntegrationTests/MockedContextTests.cs


Ian*_*son 5

我遇到了同样的问题,除了可接受的答案之外,您可能还像我一样拥有CreateQuery的通用版本-我也这样修复:

public IQueryable<TElement> CreateQuery<TElement>(Expression expression)
{
    var queryType = typeof(TestDbAsyncEnumerable<>).MakeGenericType(typeof(TElement));
    return (IQueryable<TElement>)Activator.CreateInstance(queryType, expression);
}
Run Code Online (Sandbox Code Playgroud)

该类型由TElement提供,因此其在通用版本上的实现更为简单。