3

我想知道是否有人对 FakeItEasy的这篇文章有类似的示例(原始帖子在这里。我一直在尝试找到正确的设置,但无法正确设置。我在网上找不到任何示例。我开始教我自己进行了单元测试,我决定使用 FakeItEasy。到目前为止,我所有的项目都使用实体框架并且不使用 Repository/UOW 模式,因为我认为 DbSet 和 DbContext 对于我的小型应用程序来说已经足够好了。我知道有优点/缺点EF和单元测试,但我仍然想弄清楚这一点。我很确定我的尝试已经完成(见下文),因为我得到的错误是

System.NotImplementedException:成员“IQueryable.Provider”尚未在“DbSet 1Proxy' which inherits from 'DbSet1”类型上实现。'DbSet`1' 的测试替身必须提供所使用的方法和属性的实现。

任何方向都会非常有帮助。谢谢你。

var data = new List<Request> 
{ 
request1, 
request2, 
request3
}.AsQueryable();

var fakeDbSet = A.Fake<DbSet<Request>>();

A.CallTo(() => ((IQueryable<Request>)fakeDbSet).Provider).Returns(data.Provider);
A.CallTo(() => ((IQueryable<Request>)fakeDbSet).Expression).Returns(data.Expression);
A.CallTo(() => ((IQueryable<Request>)fakeDbSet).ElementType).Returns(data.ElementType);
A.CallTo(() => ((IQueryable<Request>)fakeDbSet).GetEnumerator()).Returns(data.GetEnumerator());

var fakeContext = A.Fake<RequestPortalContext>();
A.CallTo(() => fakeContext.Requests).Returns(fakeDbSet);

var service = new RequestReadService(fakeContext);
var requests = service.GetAllRequests();

Assert.AreEqual(3, requests.Count);
Assert.AreEqual("Test1", requests[0].Name);
Assert.AreEqual("Test2", requests[1].Name);
Assert.AreEqual("Test3", requests[2].Name);
4

1 回答 1

7

我根本不熟悉 EntityFramework,也不能确定发生了什么,因为 NotImplementedException 没有堆栈跟踪,但这是我的猜测:

您会看到从定义的行service或调用的行引发的异常,而GetAllRequests不是从该行引发的异常A.Fake<DbSet<Request>>()

在我看来,这听起来像是RequestReadService(那是你的课,我在其他地方找不到吗?)的内部结构,或者它所调用的东西依赖于 DbSet 也实现了IQueryable.Provider. 现在。我不确定为什么它还没有出现。也许是显式的实现把它扔掉了。我可以建议的最好的(而且我没有尝试过,因为我完全缺乏 EF6 知识),也许是尝试明确地将接口添加到假的:

A.Fake<DbSet<Request>>(builder => 
                         builder.Implements(typeof (IQueryable<Result>)));

虽然在我输入之后,我阅读了 FakeItEasy issue 31,我担心这可能适用,所以我的想法可能行不通。再说一次,如果你愿意,试试吧。可能不会花很长时间。

或者,在短期内,我不确定要建议哪些其他解决方法。也许伪造一些DbSet实现而不是伪造DbSet自己的接口?

更新:在关于这种事情的另一个问题之后,实际上抓住了 EF6 以便我可以尝试一下,我创建了问题 477来跟踪这里似乎是潜在的问题。

于 2014-03-17T15:32:26.657 回答