4

我正在使用 FakeItEasy 库为我的单元测试创​​建假货。

我有一个ClassUnderTest我想测试该方法的MethodToTest(Data dataObject)。此方法正在调用我想伪造的接口的方法:

public interface IFoo
{
  void Execute(Action<IDataAccess> action);
}

public class ClassUnderTest
{
  private IFoo _foo;

  public ClassUnderTest(IFoo foo)
  {
    _foo = foo;
  }

  public void MethodToTest(Data dataObject)
  {
    _foo.Execute(dataAccess => dataAccess.Update(dataObject));
  }
}

public interface IDataAccess
{
  void Update(Data data);
}

public class Data
{
  public int Property { get; set; }
}

在我的单元测试中,我想检查测试方法是否正确调用了接口(使用正确的属性值):

[TestClass]
public class UnitTest1
{
  [TestMethod]
  public void TestMethod1()
  {
    var foo = A.Fake<IFoo>(x => x.Strict());
    A.CallTo(() => foo.Execute(dataAccess => dataAccess.Update(A<Data>.That.Matches(d => d.Property == 20))));      

    var cut = new ClassUnderTest(foo);

    cut.MethodToTest(new Data { Property = 20 });      
  }
}

但是在这个测试中配置有问题。我得到了例外:

测试方法TestProject1.UnitTest1.TestMethod1抛出异常:FakeItEasy.ExpectationException:调用严格假的未配置方法“Execute”。

有人知道我必须如何CallTo()正确配置语句吗?

4

2 回答 2

4

更新后的示例确实很有帮助,@rhe1980。

首先是关于您提供的测试的一些注释:

  1. A.CallTo方法没有做任何事情 - 它没有设置行为(使用 a.Invokes或 a.Returns甚至 a .DoesNothing)或验证该方法是否已被调用(例如使用.MustHaveHappened)。
  2. 比较Actions 似乎很难。我确实在Compare Delegates Action<T>找到了一些建议,但如果是我,我会采取稍微不同的策略。

与其尝试将Action委托与参考模型进行比较,我想我可以通过捕获提供给的动作Execute然后在 an 上运行它IDataAccess并查看动作的作用来模拟这一点。幸运的是,我们有 FakeItEasy 提供帮助!

我在这个测试中取得了成功:

[TestMethod]
public void TestMethod1()
{
    // Arrange
    var foo = A.Fake<IFoo>(x => x.Strict());

    var fakeDataAccess = A.Fake<IDataAccess>();

    A.CallTo(() => foo.Execute(A<Action<IDataAccess>>.Ignored))
                    .Invokes((Action<IDataAccess> action)=>action(fakeDataAccess));

    var cut = new ClassUnderTest(foo);

    // Act
    cut.MethodToTest(new Data { Property = 20 });

    // Assert
    A.CallTo(() => fakeDataAccess.Update(A<Data>.That.Matches(d => d.Property == 20)))
        .MustHaveHappened();
}

我希望它有所帮助。

于 2013-11-15T03:09:13.530 回答
1

如果我正确理解您的意图,您需要使用以下内容:

A.CallTo(() => foo.Execute(A<Action<IDataAccess>>.Ignored).MustHaveHappened();
于 2013-11-14T10:37:33.963 回答