zoukankan      html  css  js  c++  java
  • 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 est esources目录下创建),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自动化测试框架。
  • 相关阅读:
    Vim+Vundle+YouCompleteMe 安装
    TortoiseSVN 的分支合并操作
    JSP数据交互一
    Jquery操作DOM
    Jquery
    JQuery选择器
    第五章初始JQuery
    JavaScript对象及初识面向对象
    JavaScript第三章操作DOM
    JavaScript第二章操作BOM
  • 原文地址:https://www.cnblogs.com/ceshi2016/p/8444510.html
Copyright © 2011-2022 走看看