如何使用SpecFlow中的table.CreateSet<>(myClass)()生成缺失值?

时间:2016-09-20 12:16:27

标签: c# bdd specflow

我第一次使用SpecFlow为我的项目编写测试时遇到了一个小问题。

我有下一堂课:

public class FancyName
{
    [DataMember]
    public Guid Guid { get; set; }

    [DataMember]
    public string Name { get; set; }

    [DataMember]
    public List <Country> Countries { get; set; }
}

我想使用SpecFlow助手在我的测试中生成这个类。

以下是场景的一部分:

[...]
When i add some names
    | Name | Countries |
    | UK   | 1         |
    | US   | 2         |
[...]

我尝试在这样的步骤定义中解析它:

[When(@"I add some names")]
public void AddNames(Table table)
{
    var names = table.CreateSet<FancyName>();
    [...]
}

我遇到了两个问题:

  1. 我没有传递Guid,因为想要像Guid.NewGuid()那样生成它,所以创建的对象包含null
  2. 我将国家作为排序通过,但我需要创建List<Country>()
  3. 我曾经尝试迭代表并手动创建FancyName个对象,但据我所知,它不是SpecFlow方式。我试图查看文档并且很难找到合适的解决方案。

    可能有人知道解决这个问题的好方法吗? 提前谢谢。

3 个答案:

答案 0 :(得分:9)

这里主要的SpecFlow.Assist作者......

我同意@ sam-holder上面Table.CreateSet<T>并不神奇,在这种情况下,转换可能更容易解决。但是,Assist确实具有完成所要求的功能所需的功能。 :)

我想解释一下。我看到两个问题呈现为:

1)如何在每条记录上设置Guid?

2)如何将字符串值转换为数组?

对于(1),答案很简单。您可以将函数传递给CreateSet,告诉库如何实例化您要创建的对象。既然您只想设置Guid,就可以这样做:

    table.CreateSet<FancyName>(() => new FancyName { Guid = Guid.NewGuid()});
    // or simpler, table.CreateSet(() => new FancyName { Guid = Guid.NewGuid()});

对于(2),你必须做更多的编程。您希望Assist知道如何将字符串转换为列表。要做到这一点,您必须创建一个&#34;值检索器&#34;并在Assist中注册。我能够使用以下代码执行此操作:

public class CountryRetriever : IValueRetriever
{
    public bool CanRetrieve(KeyValuePair<string, string> keyValuePair, Type targetType, Type propertyType)
    {
        return propertyType.FullName.StartsWith("System.Collections.Generic.List`1[[SpecFlowExample.Country");
    }

    public object Retrieve(KeyValuePair<string, string> keyValuePair, Type targetType, Type propertyType)
    {
        return keyValuePair.Value.Split(',')
            .Select(x => new Country {Name = x})
            .ToList();
    }
}

[Binding]
public class Steps
{
    [BeforeTestRun]
    public static void Setup()
    {
        TechTalk.SpecFlow.Assist.Service.Instance.RegisterValueRetriever(new CountryRetriever());
    }

    [When(@"i add some names")]
    public void WhenIAddSomeNames(Table table)
    {
        var things = table.CreateSet<FancyName>(() => new FancyName { Guid = Guid.NewGuid()});

    }
}

直接指向这样的列表,特别是使用字符串,非常hacky,但这段代码确实有用。

国家/地区检索器会宣布它可以在遇到List<Country>类型时处理字符串的转换。然后它会拆分字符串并创建一个国家/地区列表。

SpecFlow使用大多数基本.Net类型的值检索器开始每次测试运行,但不是List<Country>。为了适应您的类型,您需要在每次测试运行之前注册新的值检索器。

答案 1 :(得分:2)

Table.CreateSet<>无法执行魔法。它不能知道它应该为你的对象创建一个新的Guid,或者它应该创建一个包含2个国家的列表。我想你必须自己创建这个对象。

解决此问题的最佳方法是使用[StepArgumentTransformation]

类似的东西:

[StepArgumentTransformation]
public List<FancyName> TransformToFancyName(Table table)
{
    //create the list from the table contents
}

[When(@"I add some names")]
public void AddNames(List<FancyName> names)
{
    .. use your FancyNames here
}

specflow会针对任何参数为StepArgumentTransformation的步骤作为最后一个参数和功能中的相应表格来调用您的List<FancyName>

答案 2 :(得分:0)

你可以考虑类似Nested Tables?之类的东西,但根据这篇文章,这是一个坏主意。它建议引入额外的步骤来填充复杂的对象。