6

我在工作中遇到了这样一个类:

public class MyObject
{
  public int? A {get; set;}
  public int? B {get; set;}
  public int? C {get; set;}
  public virtual int? GetSomeValue()
  {
    //simplified behavior:
    return A ?? B ?? C;
  }  
}

问题是我有一些代码可以访问 A、B 和 C 并调用 GetSomeValue() 方法(现在,我会说这不是一个好的设计,但有时我的双手被束缚了 ;-))。我想创建这个对象的模拟,同时将 A、B 和 C 设置为一些值。所以,当我这样使用起订量时:

var m = new Mock<MyObject>() { DefaultValue = DefaultValue.Mock };

让我在 GetSomeValue() 方法上设置结果,但是所有属性都设置为 null (并且使用 Setup() 设置所有属性非常麻烦,因为真实对象是一个讨厌的数据对象并且具有比上面更多的属性简化示例)。

所以另一方面,像这样使用 AutoFixture:

var fixture = new Fixture();
var anyMyObject = fixture.CreateAnonymous<MyObject>();

让我无法调用 GetSomeValue() 方法。

有没有办法将两者结合起来,拥有匿名值和设置调用结果的能力?

编辑

根据 nemesv 的回答,我得出了以下实用方法(希望我做对了):

public static Mock<T> AnonymousMock<T>() where T : class
{
  var mock = new Mock<T>();
  fixture.Customize<T>(c => c.FromFactory(() => mock.Object));
  fixture.CreateAnonymous<T>();
  fixture.Customizations.RemoveAt(0);
  return mock;
}
4

3 回答 3

5

实际上可以用 AutoFixture 来完成,但它确实需要一些调整。可扩展点都在那里,但我承认在这种情况下,解决方案并不是特别容易发现的。

如果您希望它与嵌套/复杂类型一起使用,则变得更加困难。

给定MyObject上面的类,以及这个MyParent类:

public class MyParent
{
    public MyObject Object { get; set; }

    public string Text { get; set; }
}

这些单元测试全部通过:

public class Scenario
{
    [Fact]
    public void CreateMyObject()
    {
        var fixture = new Fixture().Customize(new MockHybridCustomization());

        var actual = fixture.CreateAnonymous<MyObject>();

        Assert.NotNull(actual.A);
        Assert.NotNull(actual.B);
        Assert.NotNull(actual.C);
    }

    [Fact]
    public void MyObjectIsMock()
    {
        var fixture = new Fixture().Customize(new MockHybridCustomization());

        var actual = fixture.CreateAnonymous<MyObject>();

        Assert.NotNull(Mock.Get(actual));
    }

    [Fact]
    public void CreateMyParent()
    {
        var fixture = new Fixture().Customize(new MockHybridCustomization());

        var actual = fixture.CreateAnonymous<MyParent>();

        Assert.NotNull(actual.Object);
        Assert.NotNull(actual.Text);
        Assert.NotNull(Mock.Get(actual.Object));
    }

    [Fact]
    public void MyParentIsMock()
    {
        var fixture = new Fixture().Customize(new MockHybridCustomization());

        var actual = fixture.CreateAnonymous<MyParent>();

        Assert.NotNull(Mock.Get(actual));
    }
}

MockHybridCustomization 中有什么?这个:

public class MockHybridCustomization : ICustomization
{
    public void Customize(IFixture fixture)
    {
        fixture.Customizations.Add(
            new MockPostprocessor(
                new MethodInvoker(
                    new MockConstructorQuery())));
        fixture.Customizations.Add(
            new Postprocessor(
                new MockRelay(t =>
                    t == typeof(MyObject) || t == typeof(MyParent)),
                new AutoExceptMoqPropertiesCommand().Execute,
                new AnyTypeSpecification()));
    }
}

和类在 AutoMoq 对AutoFixture 的扩展MockPostprocessor中定义,因此您需要添加对这个库的引用。但是,请注意,不需要添加.MockConstructorQueryMockRelayAutoMoqCustomization

AutoExceptMoqPropertiesCommand课程也是为这种场合定制的:

public class AutoExceptMoqPropertiesCommand : AutoPropertiesCommand<object>
{
    public AutoExceptMoqPropertiesCommand()
        : base(new NoInterceptorsSpecification())
    {
    }

    protected override Type GetSpecimenType(object specimen)
    {
        return specimen.GetType();
    }

    private class NoInterceptorsSpecification : IRequestSpecification
    {
        public bool IsSatisfiedBy(object request)
        {
            var fi = request as FieldInfo;
            if (fi != null)
            {
                if (fi.Name == "__interceptors")
                    return false;
            }

            return true;
        }
    }
}

该解决方案提供了该问题的一般解决方案。然而,它还没有经过广泛的测试,所以我很想得到关于它的反馈。

于 2012-02-19T04:29:53.127 回答
4

可能有更好的原因,但这有效:

var fixture = new Fixture();
var moq = new Mock<MyObject>() { DefaultValue = DefaultValue.Mock };
moq.Setup(m => m.GetSomeValue()).Returns(3);

fixture.Customize<MyObject>(c => c.FromFactory(() => moq.Object));

var anyMyObject = fixture.CreateAnonymous<MyObject>();

Assert.AreEqual(3, anyMyObject.GetSomeValue());
Assert.IsNotNull(anyMyObject.A);
//...

最初我尝试使用fixture.Register(() => moq.Object);而不是,fixture.Customize但它注册了创建者功能,OmitAutoProperties()因此它不适用于您的情况。

于 2012-02-18T14:10:37.293 回答
2

从 3.20.0 开始,您可以使用AutoConfiguredMoqCustomization. 这将自动配置所有模拟,以便其成员的返回值由 AutoFixture 生成。

var fixture = new Fixture().Customize(new AutoConfiguredMoqCustomization());

var mock = fixture.Create<Mock<MyObject>>();

Assert.NotNull(mock.Object.A);
Assert.NotNull(mock.Object.B);
Assert.NotNull(mock.Object.C);
于 2014-08-21T17:49:25.737 回答