我正在尝试创建一组新的测试来测试我正在处理的遗留网站.该站点在后端使用数据库.我打算使用SpecFlow和Selenium,但是我对处理清理数据的最佳方法感到有点困惑.
目前,我有一组数据库备份,其中包含一组样本数据,我会在每次测试运行之前将其恢复.然而,这很麻烦,所以我只想在发布之前对关键测试运行执行此操作,并使持续集成运行在两者之间的同一数据库上运行.
目前我有大量的测试,如下所示:
Secenario: Test Item Creation Given I am logged in When I create an item with a unique name Then an item exists with the unique name
when步骤使用GUID来确保名称是唯一的,然后步骤可以通过模块变量访问它以检查它是否存在.
就像我说的那样,我有很多类似的测试,我在同一个数据库上多次运行它们,所以测试系统充满了物品,这会减慢搜索速度等.
我的问题是,解决这个问题的最佳方法是什么?我应该在测试中创建另一个步骤,再次删除该项目,如下所示:
Secenario: Test Item Creation Given I am logged in When I create an item with a unique name Then an item exists with the unique name Then delete the item with the unique name
或者我的测试框架应该以某种方式处理这个?如果是这样,人们做什么?鉴于SpecFlow步骤的全局特性我想象如果具有父子关系的多个项目可能成为问题,则以正确的顺序获取拆除步骤.
解决方法
一个好的测试应该没有依赖关系,因此创建测试步骤然后“拆除”测试数据是一个好主意.
When I create an item with a unique name
在ScenarioContext对象中,例如
ScenarioContext.Current["testItem"] = "testItemName";
这将允许您在方案的持续时间内保留此值.
然后,您可以创建与特定标记关联的SpecFlow hook,以在场景完成时拆除此数据,例如场景将是(注意标签):
@deleteTestItem Secenario: Test Item Creation Given I am logged in When I create an item with a unique name Then an item has exists with the unique name
清理代码将是:
[AfterScenario("deleteTestItem")] public void DeleteTestItem() { var testItemName = ScenarioContext.Current["testItemName"]; // Use testItemName to clean-up your database }
然后,只有具有此标记的方案才会运行此代码.请注意,如果您的所有测试都涉及创建测试项,那么您可以省略该标记并只使用AfterScenario挂钩.
或者,您可以在FeatureContext中保留所有测试项名称,然后在AfterFeature挂钩中删除这些项.这将导致更少的数据库调用(即,您不会在每个方案之后调用数据库进行清理).
我更喜欢ScenarioContext方法,因为我觉得如果一个Scenario创建数据,那么该方案应该负责清理它自己.