我如何在 SpecFlow 中使用 table.CreateSet<>(myClass)() 生成缺失值?
How can i generate missing values using table.CreateSet<>(myClass)() in SpecFlow?
我第一次使用 SpecFlow 为我的项目编写测试,我 运行 遇到了一个小问题。
我有下一个class:
public class FancyName
{
[DataMember]
public Guid Guid { get; set; }
[DataMember]
public string Name { get; set; }
[DataMember]
public List <Country> Countries { get; set; }
}
我想在我的测试中使用 SpecFlow 助手生成这个 class。
这是场景的一部分:
[...]
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>();
[...]
}
我 运行 遇到了 2 个问题:
- 我没有传递 Guid,因为想像
Guid.NewGuid()
那样生成它,所以创建的对象包含 null
- 我通过国家排序,但我需要创建
List<Country>()
。
我曾经尝试遍历 Table 并手动创建 FancyName
对象,但据我所知,这不是 SpecFlow 方式。我试图查看文档,但没能找到合适的解决方案。
可能有人知道解决这个问题的真正好方法吗?
提前致谢。
你可以考虑 Nested Tables?,但根据这个 post,这是个坏主意。它建议引入额外的步骤来填充复杂对象。
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>
作为最后一个参数并且在特征
中有相应的 table 的步骤
这里主要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 知道如何将字符串转换为列表。为此,您必须创建一个 "value retriever" 并将其注册到 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>
。为了适应您的类型,您需要在每次测试前注册您的新值检索器 运行。
我第一次使用 SpecFlow 为我的项目编写测试,我 运行 遇到了一个小问题。
我有下一个class:
public class FancyName
{
[DataMember]
public Guid Guid { get; set; }
[DataMember]
public string Name { get; set; }
[DataMember]
public List <Country> Countries { get; set; }
}
我想在我的测试中使用 SpecFlow 助手生成这个 class。
这是场景的一部分:
[...]
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>();
[...]
}
我 运行 遇到了 2 个问题:
- 我没有传递 Guid,因为想像
Guid.NewGuid()
那样生成它,所以创建的对象包含null
- 我通过国家排序,但我需要创建
List<Country>()
。
我曾经尝试遍历 Table 并手动创建 FancyName
对象,但据我所知,这不是 SpecFlow 方式。我试图查看文档,但没能找到合适的解决方案。
可能有人知道解决这个问题的真正好方法吗? 提前致谢。
你可以考虑 Nested Tables?,但根据这个 post,这是个坏主意。它建议引入额外的步骤来填充复杂对象。
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>
作为最后一个参数并且在特征
这里主要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 知道如何将字符串转换为列表。为此,您必须创建一个 "value retriever" 并将其注册到 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>
。为了适应您的类型,您需要在每次测试前注册您的新值检索器 运行。