Maven+TestNG+ReportNG/Allure接口自动化测试框架初探(上)

转载:http://www.51testing.com/html/58/n-3721258.html
由于一直忙于功能和性能测试,接口自动化测试框架改造的工作被耽搁了好久。近期闲暇一些,可以来做点有意思的事情。
  先前的自动化测试框架完全是用Java纯手工编写,核心交易接口代码、测试脚本、测试数据都进行了抽象和分离,测试报告也是自己设计的html模版输出,如果项目仅仅本地实施运行,也完全能满足目前的自动化测试需求。
  但为了自动化测试变得更加高大上,并配合公司实施持续集成的工作开展,决定将现有的接口自动化测试框架改造成Maven+TestNG方式,代码由SVN进行版本管理,项目由Jenkins构建运行。
  听说TestNG已很久,遗憾一直未尝试过,但最近学习实践了一把,这体验那叫一个字:超爽。单元测试、注解、组概念、套件、异常、参数化、依赖等等测试思想的加入,让TestNG服务于接口自动化测试大放异彩。
  本篇文章分5部分介绍:
  --1 Maven+TestNG的测试框架搭建
  --2 使用ReportNG来优化测试报告
  --3 测试案例的数据调度设计
  --4 使用Jenkins来调度构建运行
  --5 让报告更高大上—Allure报告插件使用
  由于篇幅问题,后两节会放在下一篇。
  1、Maven+TestNG的测试框架搭建
  准备条件
  1、Eclipse及其Maven插件、TestNG插件的安装请自行找度娘;
  2、本地需安装Maven,并配置好环境变量;
  3、Eclipse中的Maven-settings配置。
  创建Maven项目
  为了便于管理自动化项目中依赖的jar包、项目构建运行等,我们选择Maven来创建项目。由于对Maven研究不多,在自动化项目中遇到无法构建引入本地jar包的尴尬,但没找到解决方法,只好通过避免引入本地jar包来搞定。留待以后再研究怎么引入本地jar包的方法。
 
  Maven项目创建完后,如下目录结构:
  POM文件配置
  1、添加testNG的依赖包
  有两种方式:
  一种使用下面截图的方式,当然这种方式依赖于你的Maven插件,有时候搜索不到你想要的依赖包;
  还有一种是直接用下面的配置代码。
  <dependency>
  <groupId>org.testng</groupId>
  <artifactId>testng</artifactId>
  <version>6.11</version>
  </dependency>
 
  添加完成后查看pom文件内容
  创建TestNG-Class类
  选中 src/test/java目录,右键创建包,
  然后在创建好的包上创建TestNG类, 类名:IdAuth,同时添加上@BeforeClass 和 @AfterClass:
 
 
  代码中通过@Test注解的就是一个测试案例。
  注:如果没有在pom.xml文件中配置testng的依赖jar包,代码中会有报错。
  当前对于BeforeClass和AfterClass代码为空,在后面的内容会使用到。
  此时可以使用TestNG来运行类,邮件Run As —> TestNG Test 方式运行,结果如下:
  以上就完成了TestNG类创建并使用TestNG 运行测试类。
  接下来学习使用Maven来运行项目。
  使用Maven运行项目
  首先创建testng.xml配置文件(src\test\resources目录下创建),testng.xml中配置需要运行的TestNG类名(可以配置多个),配置内容如下:
  然后用Maven运行项目很简单,选中项目然后右键Run As,会出现很多Maven ***的选项,使用Maven Test选项来运行TestNG类。
  当然也可以在DOS窗口中,进入到项目来执行 mvn test。
  运行完成后,在项目根目录中,会出现2个新文件夹:
  其中:target是maven生成一些类文件,暂时不用管
  test-output是生成testNG类执行的结果:
  提供了2种查看执行结果的方式,1个是xml查看,1个是html查看,因为xml友好性不高,咱们直接看html的方式,结果报告中会显示执行了哪些测试案例,成功数和失败数,以及每个测试案例执行的时长。
  但是,大家看到这个报告的第一感觉是什么?是不是觉得很Low?
  我的感觉是简直Low爆了!!!宇宙第二Low的报告!!!
  接下来的内容咱们就来优化报告。
2、使用ReportNG插件来优化测试报告
  Pom.xml文件配置更新
  1、属性配置
 <properties>
  <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
  <xmlFileName>testng.xml</xmlFileName>
  </properties>
  2、依赖包管理配置
  <!-- testNG依赖管理 -->
  <dependency>
  <groupId>org.testng</groupId>
  <artifactId>testng</artifactId>
  <version>6.11</version>
  </dependency>
  <dependency>
  <groupId>org.apache.maven.reporting</groupId>
  <artifactId>maven-reporting-api</artifactId>
  <version>2.0.9</version>
  <scope>test</scope>
  <exclusions>
  <exclusion>
  <groupId>org.testng</groupId>
  <artifactId>testng</artifactId>
  </exclusion>
  </exclusions>
  </dependency>
  <!-- 依赖Guice -->
  <dependency>
  <groupId>com.google.inject</groupId>
  <artifactId>guice</artifactId>
  <version>3.0</version>
  <scope>test</scope>
  </dependency>
  <dependency>
  <groupId>org.uncommons</groupId>
  <artifactId>reportng</artifactId>
  <version>1.1.4</version>
  </dependency>
  <dependency>
  <groupId>org.apache.maven.plugins</groupId>
  <artifactId>maven-surefire-plugin</artifactId>
  <version>2.18.1</version>
  <type>maven-plugin</type>
  </dependency>
  <dependency>
  <groupId>com.beust</groupId>
  <artifactId>jcommander</artifactId>
  <version>1.64</version>
  </dependency>
  <dependency>
  <groupId>net.sourceforge.jexcelapi</groupId>
  <artifactId>jxl</artifactId>
  <version>2.6.10</version>
  </dependency>
  3、Build及插件配置
  <build>
  <plugins>
  <plugin>
  <groupId>org.apache.maven.plugins</groupId>
  <artifactId>maven-surefire-plugin</artifactId>
  <version>2.18.1</version>
  <configuration>
  <suiteXmlFiles>
  <suiteXmlFile>
  src/test/resources/${xmlFileName}
  </suiteXmlFile>
  </suiteXmlFiles>
  </configuration>
  </plugin>
  <plugin>
  <groupId>org.apache.maven.plugins</groupId>
  <artifactId>maven-surefire-plugin</artifactId>
  <version>2.18.1</version>
  <configuration>
  <properties>
  <property>
  <name>usedefaultlisteners</name>
  <value>false</value>
  </property>
  <property>
  <name>listener</name>
  <value>org.uncommons.reportng.HTMLReporter, org.uncommons.reportng.JUnitXMLReporter</value>
  </property>
  </properties>
  <workingDirectory>target/</workingDirectory>
  <forkMode>always</forkMode>
  <!-- 解决报告中中文乱码 -->
  <argLine>-Dfile.encoding=UTF-8</argLine>
  </configuration>
  </plugin>
  </plugins>
  </build>
  运行查看报告
  用Maven Test运行完成后,可以发现target/目录下出现新的文件夹
  进入这个目录,打开html/index.html文件可以查看测试报告,如下图:
  这样的页面是不是比刚才的美观了很多。
  如果就这样的报告,肯定不饱满,那我们需要给测试案例添加测试数据获取,业务逻辑,断言以及多个测试类等等。
  3、测试案例的参数化设计
  参数化设计逻辑图
  项目详细的目录结构
  ExcelUtils处理类
  (代码未经优化,有些地方可能写得不合理,请见谅!)
  package com.test.pub;
  import java.io.File;
  import java.util.HashMap;
  import java.util.Map;
  import jxl.Workbook;
  import jxl.Sheet;
  import jxl.Cell;
  public class ExcelUtils {
  public static Map <String, HashMap> excelToHashMapByFirstSheet(String fPath, String sheetName){
  try{
  File fExcel = getExcelFileObj(fPath);
  Workbook wb = Workbook.getWorkbook(fExcel);
  Sheet sht = wb.getSheet(sheetName);
  int rowCount = sht.getRows();
  int colCount = sht.getColumns();
  Cell cel = null;
  Map <String, HashMap> excelContents = new HashMap();
  if(rowCount<=2){
  System.out.println("无测试案例");
  return null;
  }else{
  //检查是否存在空行
  String rowContents = "";
  for(int i=2;i<rowCount;i++){
  if(sht.getCell(0, i).getContents().toString().length()==0){
  System.out.println("测试案例文件中存在空行");
  return null;
  }else{
  for(int j=0;j<colCount;j++){
  rowContents = rowContents + sht.getCell(j, i).getContents().toString();
  }
  if (rowContents.length()<20){
  System.out.println("测试案例文件中存在空行");
  return null;
  }
  }
  }
  }
  //开始读取内容
  for(int rowIndex=2;rowIndex<rowCount;rowIndex++){
  HashMap<String, String> rowMap = new HashMap();
  String testCaseCode = sht.getCell(0, rowIndex).getContents().toString();
  for(int colIndex=1;colIndex<colCount;colIndex++){
  rowMap.put(sht.getCell(colIndex, 1).getContents().toString(), sht.getCell(colIndex, rowIndex).getContents().toString());
  }
  excelContents.put(testCaseCode, rowMap);
  }
  wb.close();
  //HashMap<String, String> tmpMap = new HashMap();
  //tmpMap.put("count", "" + (rowCount-2));
  //excelContents.put("testsCount", tmpMap);
  return excelContents;
  }catch (Exception e){
  System.out.println("发生异常:" + e);
  }
  return null;
  }
  public  static File getExcelFileObj(String fPath){
  try{
  File fExl = new File(fPath);
  return fExl;
  }catch (Exception e){
  System.out.println(e);
  }
  return null;
  }
  }
  BeforeClass的处理
  (代码未经优化,有些地方可能写得不合理,请见谅!)
  //所有测试案例变量
  public Map<String, HashMap> allIdAuthTestCases;
  //测试案例文件路径--excel作为存储
  public String testCasePath = new TestDirUtils().getTestCasesDir() + "/IdAuthTestCases.xls";
  @BeforeClass
  public void beforeClass() {
  try{
  allIdAuthTestCases = ExcelUtil.excelToHashMapByFirstSheet(testCasePath, "身份证鉴权");
  }catch(Exception e){
  e.printStackTrace();
  }
  }
  @Test测试方法的处理
  IdAuth.java类代码
  (代码未经优化,有些地方可能写得不合理,请见谅!)
  package com.test.api;
  import static org.testng.Assert.assertFalse;
  import java.util.HashMap;
  import java.util.Map;
  import org.testng.Assert;
  import org.testng.annotations.Test;
  import org.testng.annotations.BeforeClass;
  import org.testng.annotations.AfterClass;
  import com.test.pub.*;
  public class IdAuth {
  //所有测试案例变量
  public Map<String, HashMap> allIdAuthTestCases;
  //测试案例文件路径--excel作为存储
  public String testCasePath = new TestDirUtils().getTestCasesDir() + "/IdAuthTestCases.xls";
  @BeforeClass
  public void beforeClass() {
  try{
  allIdAuthTestCases = ExcelUtils.excelToHashMapByFirstSheet(testCasePath, "身份证鉴权");
  }catch(Exception e){
  e.printStackTrace();
  }
  System.out.println("allIdAuthTestCases:" + allIdAuthTestCases);
  }
  /*
  * *****************************  身份证鉴权   *******************************
  */
  /*
  * 姓名为空
  */
  @Test(timeOut=10000, description="姓名为空")
  public void testById_NameIsNull() {
  //测试案例编号
  String testCaseCode = "ById_NameIsNull";
  //测试数据
  HashMap<String, String> testCaseData = new HashMap();
  System.out.println("allIdAuthTestCases——>" + allIdAuthTestCases);
  System.out.println(allIdAuthTestCases.containsKey(testCaseCode));
  //判断是否存在当前案例数据
  try {
  if (allIdAuthTestCases.containsKey(testCaseCode)) {
  testCaseData = allIdAuthTestCases.get(testCaseCode);
  //执行测试案例
  //断言测试结果
  Assert.assertTrue(true);
  } else {
  Assert.fail("不存在当前测试方法的案例,请检查测试案例文件!");
  }
  } catch(Exception e) {
  e.printStackTrace();
  Assert.fail("测试案例获取失败");
  }
  }
  /*
  * 证件号为空
  */
  @Test(timeOut=10000, description="证件号为空")
  public void testById_IdNoIsNull() {
  //测试案例编号
  String testCaseCode = "ById_IdNoIsNull";
  //测试数据
  HashMap<String, String> testCaseData = new HashMap();
  //判断是否存在当前案例数据
  try {
  if (allIdAuthTestCases.containsKey(testCaseCode)) {
  testCaseData = allIdAuthTestCases.get(testCaseCode);
  //执行测试案例
  //断言测试结果
  Assert.assertTrue(false);
  } else {
  Assert.fail("不存在当前测试方法的案例,请检查测试案例文件!");
  }
  } catch(Exception e) {
  e.printStackTrace();
  Assert.fail("测试案例获取失败");
  }
  }
  /*
  * 姓名与证件号均正确
  */
  @Test(timeOut=10000, description="姓名与证件号均正确")
  public void testById_IdNoAndNameRight() {
  //测试案例编号
  String testCaseCode = "ById_IdNoAndNameRight";
  //测试数据
  HashMap<String, String> testCaseData = new HashMap();
  //判断是否存在当前案例数据
  try {
  if (allIdAuthTestCases.containsKey(testCaseCode)) {
  testCaseData = allIdAuthTestCases.get(testCaseCode);
  //执行测试案例
  //断言测试结果
  Assert.assertTrue(true);
  } else {
  Assert.fail("不存在当前测试方法的案例,请检查测试案例文件!");
  }
  } catch(Exception e) {
  e.printStackTrace();
  Assert.fail("测试案例获取失败");
  }
  }
  @AfterClass
  public void afterClass() {
  }
  }
  包com.test.pub下的TestDirUtils类代码
  (代码未经优化,有些地方可能写得不合理,请见谅!)
  package com.test.pub;
  public class TestDirUtils {
  // 获取主目录
  public String getMainDir() {
  String userDir = System.getProperty("user.dir");
  if (userDir.indexOf("target")>0) {
  userDir = userDir.split("target")[0];
  }
  int userDirLen = userDir.length();
  String subUserDir = userDir.substring(userDirLen-1, userDirLen);
  if (subUserDir.equals("/")) {
  userDir = userDir.substring(0, userDirLen-1);
  }
  return userDir;
  }
  // 获取用例目录路径
  public String getTestCasesDir() {
  String testCasesDir = getMainDir();
  if (testCasesDir.indexOf("target")>0) {
  testCasesDir = testCasesDir.split("target")[0];
  }
  int userDirLen = testCasesDir.length();
  String subUserDir = testCasesDir.substring(userDirLen-1, userDirLen);
  if (subUserDir.equals("/")) {
  testCasesDir = testCasesDir.substring(0, userDirLen-1);
  }
  return  testCasesDir + "/src/test/resources/testcases";
  }
  }
  运行结果:
  OK, 大功告成,基本实现一个简单的Maven+TestNG自动化测试框架。
posted @ 2018-02-12 14:33  小学生II  阅读(2933)  评论(0编辑  收藏  举报