Moq和Equals()的组合使MS测试框架崩溃

rpm*_*lly 5 c# unit-testing moq

我认为应该是一个非常简单的测试用例,但每次运行它都会导致QTAgent32死机.在调试模式下运行测试用例显示System.StackOverflowException在"未知模块"中抛出.我已经将它缩小到展示这种行为的最基本的实现(.NET 4和VS 2010 Ultimate):

[TestClass]
public class StackOverflow
{
    [TestMethod]
    public void CreateStackOverflow()
    {
        var mockMyType1 = new Mock<MyType>();
        mockMyType1.Setup(m => m.Equals(mockMyType1.Object)).Returns(true);

        var mockMyType2 = new Mock<MyType>();

        // Real test is for a filtering routine and the Assert is using
        // Contains(), but it uses Equals() internally so it has the same problem
        Assert.IsTrue(mockMyType1.Object.Equals(mockMyType1.Object)); // returns true
        Assert.IsFalse(mockMyType1.Object.Equals(mockMyType2.Object)); // explodes
    }
}

public class MyType
{
    public virtual bool IsActive { get; set; }
    public override bool Equals(object obj)
    {
        return false;  // Not the real implementation but irrelevant to this issue
    }
}
Run Code Online (Sandbox Code Playgroud)

我觉得我错过了一些关于闭包或Moq的重要信息,但似乎这应该有效.我试过的事情,试图理解这个问题,但只是让我更加困惑:

  • 我尝试使用替换Equals()设置,mockMyType.Setup(m => m.Equals(m)).Returns(true);但这会导致Moq抛出NotSupportedException
  • 如果我将CallBase设为true而不是设置Equals(),那么一切正常
  • 最后,如果MyType类没有覆盖Equals(),那么一切正常.

谁能指出我可能发生的事情的方向?我完全不知所措.

编辑:我相信我有几个选项来完成这项工作(包括Lanorkin的回复),但我真的很想知道为什么会这样.我做错了什么,或者我应该提交Moq或Visual Studio中的错误?

更新:我最终使用下面的Denys解决方案版本并向Moq提交错误报告.我的设置现在看起来像:

mockMyType1.Setup(m => m.Equals(It.Is<MyType>(x => ReferenceEquals(x, mockMyType1.Object)))).Returns(true);
Run Code Online (Sandbox Code Playgroud)

Den*_*nko 1

这个问题从昨天就一直困扰着我,今天终于找到了答案。您必须在设置方法中使用一个函数,并且它应该断言模拟对象的真正相等性。我的意思是ReferenceEquals。所以我修改了你的GetMockMyTypes代码。当然它不能用作参考,但到目前为止意图很明确:

public static class MyTypeHelper
{
   public static IList<MyType> GetMockMyTypes()
   {
      var myTypes = new List<MyType>();

      var myMock1 = new Mock<MyType>().Object;
      Mock.Get(myMock1)
          .Setup(m => m.Equals(It.Is<MyType>(x => ReferenceEquals(x, myMock1))))
          .Returns(true);
      Mock.Get(myMock1).Setup(m => m.IsActive).Returns(false);
      myTypes.Add(myMock1);

      var myMock2 = new Mock<MyType>().Object;
      Mock.Get(myMock2)
          .Setup(m => m.Equals(It.Is<MyType>(x => ReferenceEquals(x, myMock2))))
          .Returns(true);
      Mock.Get(myMock2).Setup(m => m.IsActive).Returns(true);
      myTypes.Add(myMock2);

      return myTypes;
   }
}
Run Code Online (Sandbox Code Playgroud)