13

Autofixture によって生成された古い文字列 (Guid-y のような値) では満足できない文字列コンストラクター パラメーターがクラスにあるというシナリオを克服しようとしています。

Mark Seemann の Ploeh ブログ エントリ on Convention-based Customizations へのリンクで簡単に答えたくなる前に、私はこのテストのためにそれと彼の他のブログ エントリを参照していると言っておきましょう。合格。

デバッグをステップ実行すると、ある時点でコンストラクター パラメーターが有効な値で渡されていることがわかりますが、テストは Guid-y Color 値で失敗します。これは、「色」パラメーター値と、Autofixture によって入力される「色」プロパティの両方があるという事実と関係があると思います。コンストラクター パラメーターに対処する ISpecimenBuilder を作成しましたが、パブリック プロパティの値をテストしているのでしょうか (2 つの異なること)?

この例ではこれがやり過ぎであることはわかっていますが、Build<T>().With()メソッドの使用が DRY ではない、より複雑なシナリオを想定しています。

失敗したテスト

    [Fact]
    public void Leaf_Color_Is_Brown()
    {
        // arrange
        var fixture = new Fixture().Customize(new LeafColorCustomization());

        // act
        var leaf = fixture.Create<Leaf>();

        // using .Build<>.With(), test passes
        //var leaf = fixture.Build<Leaf>().With(l => l.Color, "brown").CreateAnonymous();

        // assert
        Assert.True(leaf.Color == "brown");
    }

SUT

    public class Leaf
    {
        public Leaf(string color)
        {
            if (color != "brown")
                throw new ArgumentException(@"NO LEAF FOR YOU!");

            this.Color = color;
        }
        public string Color { get; set; }
    }

CompositeCustomization の実装(この例では AutoMoqCustomization() は必要ないことはわかっています)

    public class LeafCustomization : CompositeCustomization
    {
        public LeafCustomization()
            : base(
            new LeafColorCustomization(),
            new AutoMoqCustomization()) { }
    }

リーフ固有の ICustomization

    public class LeafColorCustomization : ICustomization
    {
        public void Customize(IFixture fixture)
        {
            if (fixture == null)
                throw new ArgumentNullException("fixture");

            fixture.Customizations.Add(new LeafBuilder());
        }
    }

色固有の ISpecimenBuilder を持つ文字列コンストラクター

    public class LeafBuilder : ISpecimenBuilder
    {
        public object Create(object request, ISpecimenContext context)
        {
            var pi = request as ParameterInfo;
            if (pi == null)
                return new NoSpecimen(request);

            if (pi.ParameterType != typeof(string) || pi.Name != "color")
                return new NoSpecimen(request);

            return "brown";
        }
    }
4

3 に答える 3

9

解決策1

Color後処理の一部として、書き込み可能なプロパティに自動値を割り当てないように登録します。

internal class LeafColorCustomization : ICustomization
{
    public void Customize(IFixture fixture)
    {
        fixture.Customize<Leaf>(c => c
            .Without(x => x.Color));

        fixture.Customizations.Add(new LeafBuilder());
    }
}

解決策2

プロパティColorを読み取り専用にします。

public class Leaf
{
    private readonly string color;

    public Leaf(string color)
    {
        if (color != "brown")
            throw new ArgumentException(@"NO LEAF FOR YOU!");

        this.color = color;
    }

    public string Color
    {
        get { return this.color; }
    }
}

プロパティは読み取り専用であるため、ColorAutoFixtureはプロパティに値を割り当てません。

上記のソリューションは、AutoFixture2にも適用されます。

于 2013-03-20T21:06:54.170 に答える
5

プロパティの設定を個別に処理すると仮定すると、コンストラクターの引数の制限Customizationは次のようになります。

class BrownLeavesCustomization : ICustomization
{
    void ICustomization.Customize( IFixture fixture )
    {
        Func<string> notBrownGenerator = fixture.Create<Generator<string>>()
            .SkipWhile( x => x == "Brown" )
            .First;
        fixture.Customizations.Add( 
            ArgumentGeneratorCustomization<Leaf>.ForConstructorArgument(
                "color", 
                notBrownGenerator ) );
    }

    static class ArgumentGeneratorCustomization<T>
    {
        public static ISpecimenBuilder ForConstructorArgument<TArg>( string argumentName, Func<TArg> generator )
        {
            return new ConstructorArgumentGenerator<TArg>( argumentName, generator );
        }

        class ConstructorArgumentGenerator<TArg> : ISpecimenBuilder
        {
            readonly string _argumentName;
            readonly Func<TArg> _generator;

            public ConstructorArgumentGenerator( string argumentName, Func<TArg> generator )
            {
                Assert.Contains( argumentName, from ctor in typeof( T ).GetConstructors() from param in ctor.GetParameters() select param.Name );
                _argumentName = argumentName;
                _generator = generator;
            }

            object ISpecimenBuilder.Create( object request, ISpecimenContext context )
            {
                var pi = request as ParameterInfo;
                if ( pi == null )
                    return new NoSpecimen( request );
                if ( pi.Member.DeclaringType != typeof( T ) )
                    return new NoSpecimen( request );
                if ( pi.Member.MemberType != MemberTypes.Constructor )
                    return new NoSpecimen( request );
                if ( pi.ParameterType != typeof( TArg ) )
                    return new NoSpecimen( request );
                if ( pi.Name != _argumentName )
                    return new NoSpecimen( request );

                return _generator();
            }
        }
    }
}
于 2013-03-21T14:37:44.240 に答える
3

解決策:(この回答に関するMark Seemannのコメントに基づく)

コンストラクター パラメーターと書き込み可能なプロパティの両方を ISpecimenBuilder 実装に収容し、LeafColorCustomization に LeafBuilder インスタンスを追加する以外は何もしません。

public class LeafBuilder : ISpecimenBuilder
{
    public object Create(object request, ISpecimenContext context)
    {
        var paramInfo = request as ParameterInfo;
        if (paramInfo != null
            && paramInfo.ParameterType == typeof(string)
            && paramInfo.Name == "color")
        { return "brown"; }

        var propInfo = request as PropertyInfo;
        if (propInfo != null
            && propInfo.PropertyType == typeof(string)
            && propInfo.Name == "Color")
        { return "brown"; }

        return new NoSpecimen(request);
    }
}

internal class LeafColorCustomization : ICustomization
{
    public void Customize(IFixture fixture)
    {
        fixture.Customizations.Add(new LeafBuilder());
    }
}
于 2013-03-22T02:32:13.210 に答える