C# 如何 Moq entityframework DbSet 添加方法

C# How to Moq entityframework DbSet Add method

我正在尝试创建一个测试来测试 entity framework Add 方法。任何人都可以帮助如何模拟 DbSet.Add 方法。我试过如下但没有工作。我做错了什么?

我得到的结果是 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;
    }
}

根据被测方法中Add方法的使用情况...

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

...设置中还应该有一个 Returns,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);  
}