Ошибка, когда тестируемый метод вызывает IQueryable.SingleOrDefault

Я использую Entity Framework и FakeItEasy для модульного тестирования.

Есть многочисленные методы модульного тестирования, все они в порядке - DbSet-ы насмехаются, все хорошо. В частности, этот сбой происходит из-за вызова конкретного метода IQueryable.SingleOrDefault, что приводит к следующей ошибке:

Result Message: 
Test method BLL.Tests.TrackerBLLTests.GetTracker_NoCache_No_User_Success threw exception: 
System.ArgumentException: Expression of type '' cannot be used for parameter of type 'System.Linq.IQueryable`1[Model.Tracker]' of method 'Model.Tracker SingleOrDefault[Tracker](System.Linq.IQueryable`1[Mobiltracker.Model.Tracker])'
Result StackTrace:  
at System.Linq.Expressions.Expression.ValidateOneArgument(MethodBase method, ExpressionType nodeKind, Expression arg, ParameterInfo pi)
   at System.Linq.Expressions.Expression.ValidateArgumentTypes(MethodBase method, ExpressionType nodeKind, ReadOnlyCollection`1& arguments)
   at System.Linq.Expressions.Expression.Call(Expression instance, MethodInfo method, IEnumerable`1 arguments)
   at System.Linq.Expressions.Expression.Call(Expression instance, MethodInfo method, Expression[] arguments)
   at System.Linq.Queryable.SingleOrDefault[TSource](IQueryable`1 source)
   at BLL.TrackerBLL.GetTracker(Int32 trackerId, Nullable`1 _userId) in ...\TrackerBLL.cs:line 655
   at BLL.Tests.TrackerBLLTests.GetTracker_NoCache_No_User_Success() in ...\TrackerBLLTests.cs:line 145

Метод испытания (адаптированный):

[TestMethod]
public void GetTracker_NoCache_No_User_Success()
{
    var trackers = new List<Tracker> { new Tracker { TrackerId = 1 } };
    A.CallTo(() => m_iUnitOfWork.GetModelContainer().Trackers).Returns(PrepareAndGenerateFakeDbSet<Tracker>(trackers));
    var trackerBLL = A.Fake<TrackerBLL>(...);
    var returnedTracker = trackerBLL.GetTracker(1);

    Assert.IsNotNull(returnedTracker);
    Assert.AreSame(returnedTracker, m_trackerList[0]);
}

Метод PrepareAndGenerateFakeDbSet (полный):

public static DbSet<T> PrepareAndGenerateFakeDbSet<T>(List<T> _dataForDbSet) where T : class
{
    var queryableList = _dataForDbSet.AsQueryable<T>();
    var fakeDbSet = A.Fake<DbSet<T>>(builder => builder.Implements(typeof(IQueryable<T>)));

    A.CallTo(() => ((IQueryable<T>)fakeDbSet).Expression).Returns(queryableList.Expression);
    A.CallTo(() => ((IQueryable<T>)fakeDbSet).ElementType).Returns(queryableList.ElementType);
    A.CallTo(() => ((IQueryable<T>)fakeDbSet).GetEnumerator()).Returns(queryableList.GetEnumerator());

    return fakeDbSet;
}

Метод, который тестируется (адаптируется):

(...)
Tracker returnData = null;
var query = from t in Model.Trackers
            where t.TrackerId == trackerId
            select t;
returnData = query.SingleOrDefault(); //The error happens here and happens with Single(), First() or FirstOrDefault() extensions methods...
(...)
return returnData;

Есть идеи? Заранее спасибо!

1 ответ

Решение

Просто выяснил, в моих процедурах проб и неудач, что использование приведенного ниже класса для проверки моего DbSet работает, как и ожидалось:
http://pastebin.com/wQwbxhHr

using System;
using System.Collections.Generic;
using System.Data.Entity;
using System.Linq;
using System.Text;

public class MockDbSet<TEntity> : DbSet<TEntity>, IQueryable<TEntity> where TEntity : class
{
    private List<TEntity> list = null;

    /// <summary>Initializes a new instance of the MockDbSet class.</summary>
    public MockDbSet(IEnumerable<TEntity> collection)
    {
        this.list = new List<TEntity>(collection);
    }

    public override IEnumerable<TEntity> AddRange(IEnumerable<TEntity> entities)
    {
        list.AddRange(entities);

        return list;
    }

    public override TEntity Add(TEntity entity)
    {
        list.Add(entity);

        return entity;
    }

    public override TEntity Attach(TEntity entity)
    {
        return entity;
    }

    public new TDerivedEntity Create<TDerivedEntity>() where TDerivedEntity : class, TEntity
    {
        return (TDerivedEntity)list.FirstOrDefault();
    }

    public override TEntity Create()
    {
        return list.FirstOrDefault();
    }

    public override TEntity Find(params object[] keyValues)
    {
        return null;
    }

    public override System.Collections.ObjectModel.ObservableCollection<TEntity> Local
    {
        get { return new System.Collections.ObjectModel.ObservableCollection<TEntity>(this.list); }
    }

    public override TEntity Remove(TEntity entity)
    {
        list.Remove(entity);
        return entity;
    }

    public IEnumerator<TEntity> GetEnumerator()
    {
        return list.GetEnumerator();
    }

    System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator()
    {
        return list.GetEnumerator();
    }

    public Type ElementType
    {
        get { return this.list.AsQueryable().ElementType; }
    }

    public System.Linq.Expressions.Expression Expression
    {
        get { return this.list.AsQueryable().Expression; }
    }

    public IQueryProvider Provider
    {
        get { return this.list.AsQueryable().Provider; }
    }
}
Другие вопросы по тегам