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

本文关键字:myClass CreateSet SpecFlow table | 更新日期: 2023-09-27 18:07:47

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

我有下一节课:

public class FancyName
{
    [DataMember]
    public Guid Guid { get; set; }
    [DataMember]
    public string Name { get; set; }
    [DataMember]
    public List <Country> Countries { get; set; }
}

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

下面是Scenario的部分:

[...]
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>()

我曾经尝试通过表迭代并手动创建FancyName对象,但据我所知,这不是SpecFlow方式。我试着翻阅文档,但没能幸运地找到合适的解决方案。

可能有人知道解决这个问题的真正好方法吗?

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

主SpecFlow。协助作者…

我同意上面@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知道如何将字符串转换为列表。为此,您必须创建一个"值检索器"并将其注册到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()});
    }
}

像这样直接指向一个列表,特别是使用一个字符串,这是相当粗糙的,但这段代码确实可以工作。

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

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

Table.CreateSet<>不会施魔法。它不知道它应该为你的对象创建一个新的Guid,或者它应该创建一个包含两个国家的列表。我觉得你得自己做这个东西。

最好的解决方法是使用[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>,并且在

特性中有相应的表。

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