如何使用excel文件中的不同测试数据集在testng中运行多个测试用例?

时间:2017-07-21 13:36:02

标签: java selenium testng data-driven-tests testng-dataprovider

我有以下testng.xml,我的项目包含多个类,每个类都有一个@Test testNG方法及其相关的数据提供者(意味着类I1_DoLoginTest包含一个方法及其数据提供者,类I2_CreateScenarioTest包含一个方法及其数据提供者和类I3_RunSimulationTest包含一种方法及其数据提供者)

参考这三类非常相似的结构:

public class I1 {

@Test(priority = 1,dataProvider="dp_i1_Login")
public void I1_LoginTestCase(){
//Processing data from dataprovider given below
}
@DataProvider(name="dp_i1_Login")
public Object[][] dp_i1_Login() throws Exception{
//return fetching single row data in the form object array from excelsheet 
//to be processed by test case
}
}

所以基本上我有一个带有dataprovider的类,它一次提供一行数据,每个单元格将作为参数传递给同一个类的@Test方法,然后控制移动到下一个类,即I2_CreateScenarioTest,然后是I3_RunSimulationTest(两个) I2_CreateScenarioTest和I3_RunSimulationTest具有与I1_LoginTestCase相同的结构)
testng.xml结构如下:

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE suite SYSTEM "http://testng.org/testng-1.0.dtd">
<suite guice-stage="DEVELOPMENT" name="Default suite">
  <test verbose="2" name="Default test">
    <classes>
      <class name="com.als.I1"/>
      <class name="com.als.I2"/>
      <class name="com.als.I3"/>
    </classes>
  </test> <!-- Default test -->
</suite> <!-- Default suite -->

我的问题是所有这三个类的执行顺序,我从不同的Excel工作表中选择了一行(参考excel表格的结构Excel Sheet

例如,如果每个工作表中有2行提供数据,则当前执行testng suit(testng.xml)就像执行类I1-I1-I2-I2-I3-I3但我想执行类像I1-I2-I3-I1-I2-I3。

单个方法在其各自的工作表中将其数据提供者的数据执行两次2行,但是我需要在执行I1时将执行控制移动到第二类,一旦I1类从Excel工作表读取并处理第一行,类似于I2类,I2也处理一行并移动到I3,整个执行顺序(I1-I2-I3)应重复其各自表单的第二行,每个单独的excel表中的每一行共同代表一个测试套装数据,第二行在每个单独的excel表中共同代表第二个测试诉讼数据。我希望执行testng类为I1-I2-I3-(对于所有行1)-I1-I2-I3-(对于所有行2)而不是I1-I1-I2-I2-I3-I3(行1行) 2,第1行第2行,第1行第2行

1 个答案:

答案 0 :(得分:1)

目前您无法在TestNG中执行此操作。在获取@Test标记中的下一个类之前,TestNG基本上会运行一个由数据提供程序提供<test>测试的类,并迭代所有数据提供程序提供的数据集。如果你已经启用了并行性,那么这将并行发生,而不是你期望的方式。

但是,您可以尝试执行以下操作:

构建一个由@Factory带注释的构造函数驱动的测试类,并将此构造函数的@Factory注释绑定到@DataProvider带注释的数据提供程序。

在您的测试类中,所有@Test方法目前分散在多个类中。

这种安排会导致TestNG从您的数据提供者中挑选一行数据,将其提供给构造函数以实例化您的测试类,之后,您的@Test方法基本上可以处理注入的数据它通过构造函数。因此,对于数据提供程序提供的每一行数据,都会创建一个测试类实例。这是您实现所需目标的唯一途径。

以下是一个示例:

import static org.testng.Assert.assertTrue;

import org.testng.annotations.DataProvider;
import org.testng.annotations.Factory;
import org.testng.annotations.Test;

import java.util.HashMap;
import java.util.Map;

public class SampleTestClass {
    private String name;
    private int age;
    private Map<String, Integer> marks;

    @Factory(dataProvider = "getData")
    public SampleTestClass(String name, int age, Map<String, Integer> marks) {
        this.name = name;
        this.age = age;
        this.marks = marks;
    }

    @DataProvider(name = "getData")
    public static Object[][] testData() {
        return new Object[][]{
                {"Amar", 16, marksInSubjects(45, 55, 65)},
                {"Akbar", 16, marksInSubjects(55, 65, 75)},
                {"Antony", 16, marksInSubjects(35, 45, 55)},
        };
    }

    @Test
    public void testName() {
        assertTrue(name != null && !name.trim().isEmpty(), "Should have received a valid name");
    }

    @Test
    public void testAge() {
        assertTrue(age > 0 && age < 30, "Should have received a valid age.");
    }

    @Test(dataProvider = "marks")
    public void testMarks(String subject, int marks) {
        boolean validSubject = subject != null && !"sports".equalsIgnoreCase(subject.trim());
        assertTrue(validSubject, "Should have received a valid subject");
        assertTrue(marks >= 40, this.name + " didn't pass in " + subject);
    }

    @DataProvider(name = "marks")
    public Object[][] getMarks() {
        Object[][] marks = new Object[this.marks.size()][1];
        int index = 0;
        for (Map.Entry<String, Integer> mark : this.marks.entrySet()) {
            marks[index++] = new Object[]{mark.getKey(), mark.getValue()};
        }
        return marks;
    }

    private static Map<String, Integer> marksInSubjects(int m1, int m2, int m3) {
        Map<String, Integer> marks = new HashMap<>();
        marks.put("english", m1);
        marks.put("science", m2);
        marks.put("mathematics", m3);
        return marks;
    }
}