Geh*_*net 5 c# unit-testing entity-framework moq asp.net-core-mvc
我想对具有模拟的存储库的检索方法进行单元测试DbContext,但我无法将模拟DbSet值设置到存储库。
存储库看起来像这样:
public class ChangeLogRepository : Repository<ChangeLog>, IChangeLogRepository
{
public ChangeLogRepository(IDbContext context, long tenantId) : base(context, tenantId)
{
}
}
Run Code Online (Sandbox Code Playgroud)
基类:
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();
}
//..
}
Run Code Online (Sandbox Code Playgroud)
最后但并非最不重要的一点是测试类:
[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;
}
}
Run Code Online (Sandbox Code Playgroud)
如果我运行它,测试就会失败,因为该getAll()方法返回 null。看来,模拟Set()方法未正确初始化属性“Entities”。
当我在存储库构造函数中设置断点并检查实体属性时,在“表达式 > 值 > 结果视图”下会出现三个条目。在第一个结果视图下,有一个“枚举未产生结果”消息和两行带有?在其中(Visual Studio 2017)。
如何正确模拟存储库中的实体?我究竟做错了什么?
EF 文档对此进行了介绍。
IDbSet<T>。IDbSet<T>。请参阅https://msdn.microsoft.com/en-gb/data/dn314431。
请注意,还有一个关于使用模拟框架的页面,但我从未这样做过: https: //msdn.microsoft.com/en-gb/data/dn314429。
另请注意,如果使用 EF Core(又名 EF7),内存中提供程序将完全避免需要双精度。