在存储库中模拟DbSet <tentity>

时间:2017-06-29 13:57:55

标签: c# entity-framework unit-testing asp.net-core-mvc moq

我想从具有模拟DbContext的存储库对检索方法进行单元测试,但我无法将模拟的DbSet值设置为存储库。

存储库如下所示:

public class ChangeLogRepository : Repository<ChangeLog>, IChangeLogRepository
{
    public ChangeLogRepository(IDbContext context, long tenantId) : base(context, tenantId)
    {
    }
}

基类:

public class Repository<TEntity> where TEntity : class {
    protected readonly IDbContext Context;
    protected DbSet<TEntity> Entities { get; set; }
    public long TenantId { get; set; }

    protected Repository(IDbContext context, long tenant)
    {
        Context = context;
        TenantId = tenant;
        Entities = Context.Set<TEntity>();
    }
    public List<TEntity> GetAll()
    {
        return Entities.ToList();
    }
    //..
}

最后但并非最不重要的是,测试类:

[TestClass]
public class ChangeLogRepository_Test
{
    private ChangeLogRepository repository;
    private List<ChangeLog> allTestData;

    [TestInitialize]
    public void TestInitialize()
    {
        var dbContext = new Mock<IDbContext>();
        allTestData = new List<ChangeLog>() {
            new ChangeLog { Id = 10, EntityName = "User",PropertyName = "UserName",PrimaryKeyValue = 1,OldValue = "Max",NewValue = "Moritz",DateChanged = DateTime.Now,FieldType = ChangeLogFieldType.Default },
            new ChangeLog { Id = 10, EntityName = "User",PropertyName = "CreatedAt",PrimaryKeyValue =2,OldValue = "15/06/2017",NewValue = "15/06/2017",DateChanged = DateTime.Now,FieldType = ChangeLogFieldType.Date },
            new ChangeLog { Id = 10, EntityName = "Role",PropertyName = "RoleName",PrimaryKeyValue = 56,OldValue = "Admin",NewValue = "Administrator",DateChanged = DateTime.Now,FieldType = ChangeLogFieldType.Default },
        };
        var changelogs = MockDbSet(allTestData);
        dbContext.Setup(m => m.Set<ChangeLog>()).Returns(() => changelogs);
        repository = new ChangeLogRepository(dbContext.Object, 10);
    }

    [TestMethod]
    public void Setup_Test()
    {
        Assert.AreEqual(repository.GetAll(), allTestData);
    }
    private static DbSet<T> MockDbSet<T>(IEnumerable<T> list) where T : class, new()
    {
        IQueryable<T> queryableList = list.AsQueryable();
        Mock<DbSet<T>> dbSetMock = new Mock<DbSet<T>>();
        dbSetMock.As<IQueryable<T>>().Setup(x => x.Provider).Returns(queryableList.Provider);
        dbSetMock.As<IQueryable<T>>().Setup(x => x.Expression).Returns(queryableList.Expression);
        dbSetMock.As<IQueryable<T>>().Setup(x => x.ElementType).Returns(queryableList.ElementType);
        dbSetMock.As<IQueryable<T>>().Setup(x => x.GetEnumerator()).Returns(queryableList.GetEnumerator());
        return dbSetMock.Object;
     }
}

如果我运行它,测试失败,因为getAll()方法返回null。接缝时,模拟的Set()方法未正确初始化属性“实体”。

当我在repository-constructor中设置断点时 并检查“表达式”下的实体属性。价值&gt;结果视图'出现三个条目。在第一个结果视图下有一个“枚举产生无结果”消息和两行?在它(Visual Studio 2017)。

如何正确模拟存储库中的实体?我做错了什么?

2 个答案:

答案 0 :(得分:2)

我完全根据原始问题中提供的示例重新创建了测试,但无法重现null问题。模拟返回了一个填充的集合,就像配置它一样。

但是在比较两个集合时出现了问题

Assert.AreEqual(repository.GetAll(), allTestData);
他们被认为不平等。预计,因为ToList将创建一个新列表,该列表显然是对用作模拟数据源的原始列表的不同引用。

使用CollectionAssert.AreEquivalent代替

比较两个馆藏
[TestMethod]
public void Setup_Test() {
    var actual = repository.GetAll();
    CollectionAssert.AreEquivalent(allTestData, actual);
}

并且测试通过。

答案 1 :(得分:0)

EF文档涵盖了这一点。

  1. 在您的上下文中使用IDbSet<T>
  2. 为您的测试创建IDbSet<T>的虚假实现。
  3. 请参阅https://msdn.microsoft.com/en-gb/data/dn314431

    请注意,还有一个关于使用模拟框架的页面,但我从未这样做过:https://msdn.microsoft.com/en-gb/data/dn314429

    另请注意,如果使用EF Core(又名EF7),则内存提供商会避免需要双打。