2016-11-07 101 views
13

我正在嘗試爲調用異步存儲庫的類創建單元測試。我正在使用ASP.NET Core和Entity Framework Core。我的通用資源庫看起來像這樣。如何使用實體框架模擬異步存儲庫核心

public class EntityRepository<TEntity> : IEntityRepository<TEntity> where TEntity : class 
{ 
    private readonly SaasDispatcherDbContext _dbContext; 
    private readonly DbSet<TEntity> _dbSet; 

    public EntityRepository(SaasDispatcherDbContext dbContext) 
    { 
     _dbContext = dbContext; 
     _dbSet = dbContext.Set<TEntity>(); 
    } 

    public virtual IQueryable<TEntity> GetAll() 
    { 
     return _dbSet; 
    } 

    public virtual async Task<TEntity> FindByIdAsync(int id) 
    { 
     return await _dbSet.FindAsync(id); 
    } 

    public virtual IQueryable<TEntity> FindBy(Expression<Func<TEntity, bool>> predicate) 
    { 
     return _dbSet.Where(predicate); 
    } 

    public virtual void Add(TEntity entity) 
    { 
     _dbSet.Add(entity); 
    } 
    public virtual void Delete(TEntity entity) 
    { 
     _dbSet.Remove(entity); 
    } 

    public virtual void Update(TEntity entity) 
    { 
     _dbContext.Entry(entity).State = EntityState.Modified; 
    } 

    public virtual async Task SaveChangesAsync() 
    { 
     await _dbContext.SaveChangesAsync(); 
    } 
} 

然後,我有一個在倉庫的一個實例調用FindBy和FirstOrDefaultAsync服務類:

public async Task<Uri> GetCompanyProductURLAsync(Guid externalCompanyID, string productCode, Guid loginToken) 
    {    
     CompanyProductUrl companyProductUrl = await _Repository.FindBy(u => u.Company.ExternalCompanyID == externalCompanyID && u.Product.Code == productCode.Trim()).FirstOrDefaultAsync(); 

     if (companyProductUrl == null) 
     { 
      return null; 
     } 

     var builder = new UriBuilder(companyProductUrl.Url); 
     builder.Query = $"-s{loginToken.ToString()}"; 

     return builder.Uri; 
    } 

我試圖嘲弄庫調用下面我的測試:

[Fact] 
    public async Task GetCompanyProductURLAsync_ReturnsNullForInvalidCompanyProduct() 
    { 
     var companyProducts = Enumerable.Empty<CompanyProductUrl>().AsQueryable(); 

     var mockRepository = new Mock<IEntityRepository<CompanyProductUrl>>(); 
     mockRepository.Setup(r => r.FindBy(It.IsAny<Expression<Func<CompanyProductUrl, bool>>>())).Returns(companyProducts); 

     var service = new CompanyProductService(mockRepository.Object); 

     var result = await service.GetCompanyProductURLAsync(Guid.NewGuid(), "wot", Guid.NewGuid()); 

     Assert.Null(result); 
    } 

但是,當測試執行對存儲庫的調用時,出現以下錯誤:

The provider for the source IQueryable doesn't implement IAsyncQueryProvider. Only providers that implement IEntityQueryProvider can be used for Entity Framework asynchronous operations. 

我該如何正確地模擬知識庫以使其發揮作用?

+2

可能這可以幫助https://msdn.microsoft.com/en-us/library/dn314429.aspx – Nkosi

+0

閱讀「異步查詢測試」一節 – Nkosi

+0

您需要同時模擬'IQueryable '和'IAsyncEnumerableAccessor '接口以及 – Dealdiane

回答

23

感謝@Nkosi爲我指出了一個鏈接,在EF 6中做了同樣的事情:https://msdn.microsoft.com/en-us/library/dn314429.aspx。這與EF Core完全不同,但我可以從它開始並進行修改以使其正常工作。下面是我創建的「模擬」 IAsyncQueryProvider測試類:

internal class TestAsyncQueryProvider<TEntity> : IAsyncQueryProvider 
{ 
    private readonly IQueryProvider _inner; 

    internal TestAsyncQueryProvider(IQueryProvider inner) 
    { 
     _inner = inner; 
    } 

    public IQueryable CreateQuery(Expression expression) 
    { 
     return new TestAsyncEnumerable<TEntity>(expression); 
    } 

    public IQueryable<TElement> CreateQuery<TElement>(Expression expression) 
    { 
     return new TestAsyncEnumerable<TElement>(expression); 
    } 

    public object Execute(Expression expression) 
    { 
     return _inner.Execute(expression); 
    } 

    public TResult Execute<TResult>(Expression expression) 
    { 
     return _inner.Execute<TResult>(expression); 
    } 

    public IAsyncEnumerable<TResult> ExecuteAsync<TResult>(Expression expression) 
    { 
     return new TestAsyncEnumerable<TResult>(expression); 
    } 

    public Task<TResult> ExecuteAsync<TResult>(Expression expression, CancellationToken cancellationToken) 
    { 
     return Task.FromResult(Execute<TResult>(expression)); 
    } 
} 

internal class TestAsyncEnumerable<T> : EnumerableQuery<T>, IAsyncEnumerable<T>, IQueryable<T> 
{ 
    public TestAsyncEnumerable(IEnumerable<T> enumerable) 
     : base(enumerable) 
    { } 

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

    public IAsyncEnumerator<T> GetEnumerator() 
    { 
     return new TestAsyncEnumerator<T>(this.AsEnumerable().GetEnumerator()); 
    } 

    IQueryProvider IQueryable.Provider 
    { 
     get { return new TestAsyncQueryProvider<T>(this); } 
    } 
} 

internal class TestAsyncEnumerator<T> : IAsyncEnumerator<T> 
{ 
    private readonly IEnumerator<T> _inner; 

    public TestAsyncEnumerator(IEnumerator<T> inner) 
    { 
     _inner = inner; 
    } 

    public void Dispose() 
    { 
     _inner.Dispose(); 
    } 

    public T Current 
    { 
     get 
     { 
      return _inner.Current; 
     } 
    } 

    public Task<bool> MoveNext(CancellationToken cancellationToken) 
    { 
     return Task.FromResult(_inner.MoveNext()); 
    } 
} 

這裏是我使用這些類更新測試用例:

[Fact] 
public async Task GetCompanyProductURLAsync_ReturnsNullForInvalidCompanyProduct() 
{ 
    var companyProducts = Enumerable.Empty<CompanyProductUrl>().AsQueryable(); 

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

    mockSet.As<IAsyncEnumerable<CompanyProductUrl>>() 
     .Setup(m => m.GetEnumerator()) 
     .Returns(new TestAsyncEnumerator<CompanyProductUrl>(companyProducts.GetEnumerator())); 

    mockSet.As<IQueryable<CompanyProductUrl>>() 
     .Setup(m => m.Provider) 
     .Returns(new TestAsyncQueryProvider<CompanyProductUrl>(companyProducts.Provider)); 

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

    var contextOptions = new DbContextOptions<SaasDispatcherDbContext>(); 
    var mockContext = new Mock<SaasDispatcherDbContext>(contextOptions); 
    mockContext.Setup(c => c.Set<CompanyProductUrl>()).Returns(mockSet.Object); 

    var entityRepository = new EntityRepository<CompanyProductUrl>(mockContext.Object); 

    var service = new CompanyProductService(entityRepository); 

    var result = await service.GetCompanyProductURLAsync(Guid.NewGuid(), "wot", Guid.NewGuid()); 

    Assert.Null(result); 
} 

感謝這麼多的幫助!

+0

很高興您最終弄清楚了。我正在挖掘github上的源代碼,看看他們是否有任何moq示例。有趣的是,當我檢查並看到你來到你自己的解決方案時,我正在調查一個。涼。 – Nkosi

+0

我會將其轉換爲擴展方法,以便您可以在測試中重複使用它。 – Nkosi

+1

檢查給出[這裏](http://stackoverflow.com/a/40500030/5233410)給出的答案,其中使用了擴展方法。快樂編碼! – Nkosi

0

嘗試使用我的起訂量擴展MockQueryable:https://github.com/romantitov/MockQueryable 支持的所有同步/異步操作

//1 - create a List<T> with test items 
var users = new List<UserEntity>() 
{ 
new UserEntity, 
... 
}; 

//2 - build mock by extension 
var mock = users.AsQueryable().BuildMock(); 

//3 - setup the mock as Queryable 
_userRepository.Setup(x => x.GetQueryable()).Returns(mock.Object);