2016-11-08 59 views
-1

我想创建一个测试来测试实体框架Add方法。任何人都可以帮助如何嘲笑DbSet.Add方法。我尝试了以下但不工作。我究竟做错了什么?C#如何Moq entityframework DbSet添加方法

我得到的结果是nullrepository.Insert后...

test.cs中:

var productToCreate = new Product { Name = "Added", Description = "Added" };   

var result = repository.InsertAsync(objToCreate, userContext).Result; 
Assert.AreEqual(result.Name, "Added"); 

Mock.cs

internal static DbSet<T> GetMockedDataSet<T>(IEnumerable<T> data) where T : class 
{ 
    // Create a mocked data set that contains the data 
    var set = new Mock<DbSet<T>>(); 
    set.As<IDbAsyncEnumerable<T>>() 
     .Setup(m => m.GetAsyncEnumerator()) 
     .Returns(new TestDbAsyncEnumerator<T>(data.GetEnumerator())); 
    set.As<IQueryable<T>>() 
     .Setup(m => m.Provider) 
     .Returns(new TestDbAsyncQueryProvider<T>(data.AsQueryable().Provider)); 
    set.As<IQueryable<T>>().Setup(m => m.Expression).Returns(data.AsQueryable().Expression); 
    set.As<IQueryable<T>>().Setup(m => m.ElementType).Returns(data.AsQueryable().ElementType); 
    set.As<IQueryable<T>>().Setup(m => m.GetEnumerator()).Returns(data.GetEnumerator()); 

    set.Setup(x => x.AsNoTracking()).Returns(set.Object); 
    set.Setup(x => x.Add(It.IsAny<T>())).Callback<T>((s) => data.Concat(new[] { s })); 

    // Return the mock 
    return set.Object; 
} 

库:

public async Task<Product> InsertAsync(Product input) 
{ 
    using (var ctx = .....)) 
    { 
     var added = ctx.Set<Product>().Add(input); 

     await ctx.ValidateAndSaveAsync(); 

     return added; 
    } 
} 
+1

提供一个能够再现问题的[mcve]。 – Nkosi

+0

@Nkosi,对不起,我不明白你是什么意思?我提供了我的代码块并解释了我期待的内容?谢谢 –

+1

这个例子不完整。能够重新创建问题的因变量太多,测试中的方法与测试中显示的不同,显示的设置不完整....需要继续吗? – Nkosi

回答

2

根据如何Add方法在测试所采用的方法...

var added = ctx.Set<Product>().Add(input); 

...也应该有在返回已输入的参数设置一个Returns,如果是这样的期望功能。

set.Setup(x => x.Add(It.IsAny<T>())) 
    .Returns<T>(arg => arg) 
    .Callback<T>((s) => data.Concat(new[] { s })); 

但考虑到有关的上下文相关性的信息是未知...

using (var ctx = .....)) 

这是不确定的,如果所提供的解决方案将收到预期的效果。

此外,如果测试异步方法,请不要混合使用异步和同步调用。以下行...

var result = repository.InsertAsync(objToCreate, userContext).Result; 

...可能导致死锁。

使测试方法一路异步。

[TestMethod] 
public async Task InsertAsync_Should_Return_Product() { 
    //...other code 

    var expected = new Product { Name = "Added", Description = "Added" };   

    var actual = await repository.InsertAsync(expected, userContext); 

    Assert.AreEqual(expected.Name, actual.Name); 
} 
+0

哇!它像魅力一样工作。谢谢。我改变了testmethod以使用异步。再次感谢 –

+0

附注 - 您对add方法的回调没有做任何事情。它对提供的原始收集没有影响。 – Nkosi