6

我正在寻找自定义AutoFixture的创建时行为,以便我可以在生成和分配固定装置的属性后设置一些依赖对象。

例如,假设我有一个自定义 a 的方法,User因为它的IsDeleted属性对于某些测试集总是必须为 false:

public class User
{
   public int Id { get; set; }
   public string Name { get; set; }
   public bool IsDeleted { get; set; }
}

public static ObjectBuilder<User> BuildUser(this Fixture f)
{
   return f.Build<User>().With(u => u.IsDeleted, false);
}

(我将ObjectBuilder返回给测试,以便它可以在必要时进一步定制夹具。)

我想做的是Id在创建时自动将该用户与匿名集合相关联,但我不能按原样执行此操作,因为Id在我将返回值交回单元测试时尚未生成恰当的。这是我正在尝试做的事情:

public static ObjectBuilder<User> BuildUserIn(this Fixture f, UserCollection uc)
{
   return f.Build<User>()
           .With(u => u.IsDeleted, false);
           .AfterCreation(u =>
            {
               var relation = f.Build<UserCollectionMembership>()
                               .With(ucm => ucm.UserCollectionId, uc.Id)
                               .With(ucm => ucm.UserId, u.Id)
                               .CreateAnonymous();
               Repository.Install(relation);
            }
}

这样的事情可能吗?或者也许有更好的方法来实现我创建匿名对象图的目标?

4

2 回答 2

6

对于该Build方法,这是不可能的,而且可能永远不会,因为有更好的选择。

首先,永远不需要围绕方法编写静态辅助方法Build。该Build方法用于真正的一次性初始化,其中需要在事前定义属性或字段值。

即想象这样一个类:

public class MyClass
{
    private string txt;

    public string SomeWeirdText
    {
        get { return this.txt; }
        set
        {
            if (value != "bar")
                throw new ArgumentException();
            this.txt = value;
        }
    }
}

在这个(人为的)示例中,顺子fixture.CreateAnonymous<MyClass>会抛出,因为它会尝试将“bar”以外的东西分配给属性。

在一次性场景中,可以使用该Build方法来逃避这个问题。一个示例是简单地将值显式设置为“bar”:

var mc =
    fixture.Build<MyClass>().With(x => x.SomeWeirdText, "bar").CreateAnonymous();

但是,更简单的方法是省略该属性:

var mc =
    fixture.Build<MyClass>().Without(x => x.SomeWeirdText).CreateAnonymous();

但是,一旦您开始想要反复执行此操作,就有更好的选择。AutoFixture 有一个非常复杂且可定制的引擎,用于定义事物的创建方式。

首先,可以从将属性的省略移到自定义中开始,如下所示:

fixture.Customize<MyClass>(c => c.Without(x => x.SomeWeirdText));

现在,只要夹具创建了 MyClass 的实例,它就会完全跳过该属性。之后您仍然可以分配一个值:

var mc = fixture.CreateAnonymous<MyClass>();
my.SomeWeirdText = "bar";

如果你想要更复杂的东西,你可以实现一个自定义的 ISpecimenBuilder。如果您想在创建实例后运行一些自定义代码,您可以使用后处理器装饰您自己的 ISpecimenBuilder 并提供一个委托。这可能看起来像这样:

fixture.Customizations.Add(
    new Postprocessor(yourCustomSpecimenBuilder, obj =>
        { */ do something to obj here */ }));

(顺便说一句,你还在使用 AutoFixture 1.0 吗?IIRC,ObjectBuilder<T>从那以后就没有了……)

于 2012-04-12T19:44:28.853 回答
3

AutoFixture CodePlex 网站上有关于此主题的有用讨论

我相信我在那里链接的后处理器定制应该对您有所帮助。示例用法:

class AutoControllerDataAttribute : AutoDataAttribute
{  
    public AutoControllerDataAttribute()
        : this( new Fixture() )
    {
    }

    public AutoControllerDataAttribute( IFixture fixture )
        : base( fixture )
    {
        fixture.Customize( new AutoMoqCustomization() );
        fixture.Customize( new ApplyControllerContextCustomization() );
    }

    class ApplyControllerContextCustomization : PostProcessWhereIsACustomization<Controller>
    {
        public ApplyControllerContextCustomization()
            : base( PostProcess )
        {
        }

        static void PostProcess( Controller controller )
        {
            controller.FakeControllerContext();
            // etc. - add stuff you want to happen after the instance has been created
于 2012-07-25T20:14:03.667 回答