Testng(三):加载外部数据

1 概述

对于一个函数或者一个流程,给定一个输入,应当返回一个结果

输入指参数或条件的组合,也就构成不同的测试场景

结果即程序对输入的处理,将其与预期结果比对,便可知当前测试场景功能的正确性

当这些数据以外部文件的形式存储时,就可以很方便地修改追加,而不用重新调整代码

2 @DataProvider

在Testng中,@DataProvider负责向测试方法提供数据,返回Object[][]对象

于是可以考虑,在@DataProvider方法中,实现读取外部文件,并将文件的内容以Object[][]形式返回

方案:

  1. 每个方法对应各自的dataProvider,读取各自的数据文件;易于实现,但代码过于冗余,并且数据文件太零散,读频繁
  2. 所有方法对应一个dataProvider,dataProvider根据调用的方法名选择对应的测试数据;实现复杂,且需要考虑并行,但代码精简
public class DataDriver {
    public static Map<String, Object[][]> data = null;

    @DataProvider(name = "dp")
    public static Object[][] getData(Method method) {
        if (data == null || data.get(method.getName()) == null) {
            return new Object[][] {};
        }
        return data.get(method.getName());
    }
}

这里选择第2种方式,但是并没有读外部文件,只提供按方法名选取数据

考虑到测试用例过多,如果一次性将所有用例数据加载到内存,会过于臃肿,所以决定只加载当前case的数据

3 TestCase

@BeforeClass阶段初始化数据,@AfterClass阶段回收

读取数据时,封装了工具类;根据不同的数据载体,可以编写多套工具类

测试方法均指向同一个dataProvider,不要忘记加上dataProviderClass属性

public class IpCase {
    @BeforeClass
    public void setup() {
        String caseFullName = this.getClass().getName();
        String caseName = caseFullName.substring(caseFullName.lastIndexOf('.') + 1);
        DataDriver.data = ExcelUtil.readExcel(caseName);
    }

    @AfterClass
    public void teardown() {
        DataDriver.data.clear();
    }

    @Test(dataProvider = "dp", dataProviderClass = DataDriver.class)
    public void isIpv41(String ip, boolean expected) {
        boolean actual = ip != null;
        Assert.assertEquals(actual, expected);
    }

    @Test(dataProvider = "dp", dataProviderClass = DataDriver.class)
    public void isIpv6(String ip, boolean expected) {
        boolean actual = ip != null;
        Assert.assertEquals(actual, expected);
    }
}

4 处理Excel文件

数据存储,可以使用Excel,TXT,数据库也行,挑选一个易于维护、易于阅读的载体即可

这里我们暂定为excel,使用poi包进行解析

excel的名称要求与TestCase一致(IpCase.xlsx),且sheet名称对应各个测试方法(isIpv4, isIpv6),这样便可以根据方法名获取相应的测试数据

isIpv4对应数据如下,首行为标题,解析时注意跳过

IP Expected
1.1.1.1 TRUE
2.2.2.256 FALSE

isIpv6对应数据如下

IP Expected
::1 TRUE
FF01::1101 TRUE
public class ExcelUtil {
    private static final String SUFFIX_2007 = ".xlsx";

    public static Map<String, Object[][]> readExcel(String caseName) {
        Map<String, Object[][]> result = new HashMap<>();
        InputStream is = null;

        try {
            is = ExcelUtil.class.getClassLoader().getResourceAsStream(String.format("TestCase/%s%s", caseName, SUFFIX_2007));
            XSSFWorkbook workbook = new XSSFWorkbook(is);

            for (int sheetNum = 0; sheetNum < workbook.getNumberOfSheets(); sheetNum++) {
                Sheet sheet = workbook.getSheetAt(sheetNum);

                int rows = sheet.getLastRowNum();
                if (rows <= 0) continue;
                int cols = sheet.getRow(1).getLastCellNum();
                Object[][] data = new Object[rows][cols];

                for (int rowNum = 1; rowNum <= rows; rowNum++) {
                    Row row = sheet.getRow(rowNum);
                    for (int celNum = 0; celNum < cols; celNum++) {
                        Cell cell = row.getCell(celNum);

                        Object cellValue = null;
                        CellType cellType = cell.getCellTypeEnum();
                        if (cellType == CellType.BOOLEAN) {
                            cellValue = cell.getBooleanCellValue();
                        } else if (cellType == CellType.STRING) {
                            cellValue = cell.getStringCellValue();
                        } else if (cellType == CellType.NUMERIC) {
                            cellValue = cell.getNumericCellValue();
                        } else {
                            cellValue = "";
                        }

                        data[rowNum - 1][celNum] = cellValue;
                    }
                }
                result.put(sheet.getSheetName(), data);
            }
        } catch (IOException e) {
            System.out.println("File not found: " + caseName);
        } finally {
            if (is != null) {
                try {
                    is.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
        return result;
    }
}

5 @Test注解转换

当测试方法需要加载数据时,都不得不指定属性dataProvider = "dp", dataProviderClass = DataDriver.class

为了省事,可以使用IAnnotationTransformer监听器

  • 约定方法名包含“Dp”、“_dp”或其他你喜欢的后缀时,自动给@Test注解添加dataProvider属性
  • 指定group名,假设group = dp的测试方法均自动补充属性
public class TestAnnotationTransformer implements IAnnotationTransformer {
    @Override
    public void transform(ITestAnnotation annotation, Class testClass, Constructor testConstructor, Method testMethod) {
        String[] groups = annotation.getGroups();
        for(String group: groups) {
            if (group.equals("dp")) {
                annotation.setDataProvider("dp");
                annotation.setDataProviderClass(DataDriver.class);
                break;
            }
        }
    }
}

6 后续优化

  • 目前只有excel解析工具类,且只支持2007版,可以补充2007以下版本支持
  • Excel数据类型判断,浮点数、日期等类型需要加强
  • 并行执行时,DataDriver.data可能冲突,需要将引用迁移到Case类下
上一篇:TestNG用例执行顺序


下一篇:API判断网站IP地址,国家区域