a . callto (..). returnlazy(..)抛出“指定对象不被识别为伪对象”

本文关键字:对象 识别 returnlazy callto 抛出 | 更新日期: 2023-09-27 18:08:23

在c#中我曾经写过

var provider = A.Fake<ITimeProvider>();
A.CallTo(() => provider.Fetch()).ReturnsLazily(call => data[0]);
container.Register(() => provider);

捕获对Fetch()的调用。

当我尝试同样的f#

let provider = A.Fake<ITimeProvider>()
A.CallTo(fun () -> provider.Fetch()).ReturnsLazily(fun call -> data.[0]) |> ignore
container.Register(fun () -> provider)

测试失败
Test Error : ....Test
System.ArgumentException : The specified object is not recognized as a fake object.
at Microsoft.FSharp.Control.AsyncBuilderImpl.commit[a](Result`1 res)
at Microsoft.FSharp.Control.CancellationTokenOps.RunSynchronously[a](CancellationToken token, FSharpAsync`1 computation, FSharpOption`1 timeout)
at Microsoft.FSharp.Control.FSharpAsync.RunSynchronously[T](FSharpAsync`1 computation, FSharpOption`1 timeout, FSharpOption`1 cancellationToken)
...

对于f#,它的定义应该有所不同。你知道怎么做吗?

a . callto (..). returnlazy(..)抛出“指定对象不被识别为伪对象”

FakeItEasy使用f# 3支持的LINQ表达式,但是在使用静态API时似乎存在不兼容性。根据错误信息"对象不被识别为假对象",我怀疑在这种情况下,对象的分辨率是c#/VB。净具体。

FakeItEasy的基于实例成员的设置似乎有效:

let fake = Fake<ITimeProvider>()
fake.CallsTo(fun x -> x.Fetch()).ReturnsLazily(fun () -> data.[0]) |> ignore
let provider = fake.FakedObject

另一个选择是使用f#和Moq(这里我使用的是Moq. fsharp . extensions):

let mock = Mock<ITimeProvider>()
mock.SetupFunc(fun x -> x.Fetch()).Returns(data.[0]).End
let provider = mock.Object

或者是专门为f#设计的Foq:

let provider = 
    Mock<ITimerProvider>.Method(fun x -> <@ x.Fetch @>).Returns(data.[0])