使用ExtentReports生成Testng测试报告

本文中使用的extentreports的版本为4.1.7,对网上v3版本的Listener进行了改造。
XML支持单suite文件或者多个suite文件的编写形式。
单个suite文件时,以test名称为一级标题。
多个suite文件时,以suite名称为一级标题。
 
依赖包

        <dependency>
            <groupId>com.aventstack</groupId>
            <artifactId>extentreports</artifactId>
            <version>4.1.7</version>
            <scope>test</scope>
        </dependency>

 

配置Testng.xml
    <listeners>
        <!--testng的XML配置文件中添加这些内容-->
        <listener class-name="org.uncommons.reportng.HTMLReporter"/>
        <listener class-name="org.uncommons.reportng.JUnitXMLReporter"/>
        <listener class-name="com.shein.dms.config.ExtentTestNGIReporterListener"/>
    </listeners>

 

配置pom.xml

<plugin>
                <artifactId>maven-surefire-plugin</artifactId>
                <!--                <version>2.22.1</version>-->
                <configuration>
                    <!--<testFailureIgnore>true</testFailureIgnore>-->
                    <forkMode>never</forkMode>
                    <argLine>-Dfile.encoding=UTF-8</argLine>
                    <suiteXmlFiles>
                        <suiteXmlFile>src/test/resources/testng.xml</suiteXmlFile>
                    </suiteXmlFiles>
                    <properties>
                        <property>
                            <name>usedefaultlisteners</name>
                            <value>true</value>
                        </property>
                        <property>
                            <name>listener</name>
                            <value>
                                org.uncommons.reportng.HTMLReporter,
                                org.uncommons.reportng.JUnitXMLReporter,
                                com.shein.dms.config.ExtentTestNGIReporterListener
                            </value>
                        </property>
                    </properties>
                    <workingDirectory>target/</workingDirectory>
                    <forkMode>always</forkMode>
                    <!-- 解决报告中中文乱码 -->
                    <argLine>-Dfile.encoding=UTF-8</argLine>
                </configuration>
            </plugin>
        </plugins>

 

生成报告源码:
package com.shein.dms.config;

import com.aventstack.extentreports.ExtentReports;
import com.aventstack.extentreports.ExtentTest;
import com.aventstack.extentreports.Status;
import com.aventstack.extentreports.reporter.ExtentSparkReporter;
import com.aventstack.extentreports.reporter.configuration.Theme;
import org.springframework.util.StringUtils;
import org.testng.*;
import org.testng.xml.XmlSuite;

import java.io.File;
import java.util.*;

/**
 * testng报告监听
 * 生成extentreport测试报告
 */
public class ExtentTestNGIReporterListener implements IReporter {
    //生成的路径以及文件名
    private static final String OUTPUT_FOLDER = "test-output/";
    private static final String FILE_NAME = "extent.html";
    private static final String REPORT_TITLE = "DMS自动化测试报告";

    private ExtentReports extent;
    private ExtentSparkReporter sparkReporter;

    @Override
    public void generateReport(List<XmlSuite> xmlSuites, List<ISuite> suites, String outputDirectory) {
        init();
        boolean createSuiteNode = false;
        if (suites.size() > 1) {
            createSuiteNode = true;
        }
//        for (ISuite suite : suites) {
        // 解决报告一级标题排序问题
        for (int i = suites.size(); i > 0; i--) {
            ISuite suite = suites.get(i - 1);

            Map<String, ISuiteResult> result = suite.getResults();
            //如果suite里面没有任何用例,直接跳过,不在报告里生成
            if (result.size() == 0) {
                continue;
            }
            //统计suite下的成功、失败、跳过的总用例数
            int suiteFailSize = 0;
            int suitePassSize = 0;
            int suiteSkipSize = 0;
            ExtentTest suiteTest = null;
            //存在多个suite的情况下,在报告中将同一个suite的测试结果归为一类,创建一级节点。
            if (createSuiteNode) {
//                suiteTest = extent.createTest(suite.getName()).assignCategory(suite.getName());
                suiteTest = extent.createTest(suite.getName());  // 删除一级标签
            }
            boolean createSuiteResultNode = false;
            if (result.size() > 1) {
                createSuiteResultNode = true;
            }
            for (ISuiteResult r : result.values()) {
                ExtentTest resultNode;
                ITestContext context = r.getTestContext();
                if (createSuiteResultNode) {
                    //没有创建suite的情况下,将在SuiteResult的创建为一级节点,否则创建为suite的一个子节点。
                    if (null == suiteTest) {
                        resultNode = extent.createTest(r.getTestContext().getName());
                    } else {
                        resultNode = suiteTest.createNode(r.getTestContext().getName());
                    }
                } else {
                    resultNode = suiteTest;
                }
                if (resultNode != null) {
                    // 设置功能名称 e.g. 测试用例集 : demo测试用例 / suite标签名:test标签名
                    resultNode.getModel().setName(suite.getName() + " : " + r.getTestContext().getName());

//                    功能名打标签,影响数据统计
//                    resultNode.assignCategory(suite.getName(), r.getTestContext().getName());

                    // 设置用例执行开始和结束时间
                    resultNode.getModel().setStartTime(r.getTestContext().getStartDate());
                    resultNode.getModel().setEndTime(r.getTestContext().getEndDate());

                    //统计SuiteResult下的数据
                    int passSize = r.getTestContext().getPassedTests().size();
                    int failSize = r.getTestContext().getFailedTests().size();
                    int skipSize = r.getTestContext().getSkippedTests().size();
                    suitePassSize += passSize;
                    suiteFailSize += failSize;
                    suiteSkipSize += skipSize;
                    if (failSize > 0) {
                        resultNode.getModel().setStatus(Status.FAIL);
                    }
                    resultNode.getModel().setDescription(String.format("Pass: %s ; Fail: %s ; Skip: %s ;", passSize, failSize, skipSize));
                }
                // 设置标签信息
                String[] categories = {suite.getName(), r.getTestContext().getName()};

                // 生成子节点,测试用例
                buildTestNodes(resultNode, context.getFailedTests(), Status.FAIL, categories);
                buildTestNodes(resultNode, context.getSkippedTests(), Status.SKIP, categories);
                buildTestNodes(resultNode, context.getPassedTests(), Status.PASS, categories);

                // 如果是三级标题,从子节点中获取开始和结束时间
                if (createSuiteNode) {
                    resultNode.getModel().computeEndTimeFromChildren();
                }
            }
            if (suiteTest != null) {
                suiteTest.getModel().setDescription(String.format("Pass: %s ; Fail: %s ; Skip: %s ;", suitePassSize, suiteFailSize, suiteSkipSize));
                if (suiteFailSize > 0) {
                    suiteTest.getModel().setStatus(Status.FAIL);
                }
            }
        }
//        for (String s : Reporter.getOutput()) {
//            extent.setTestRunnerOutput(s);
//        }

        extent.flush();
    }

    private void init() {
        //文件夹不存在的话进行创建
        File reportDir = new File(OUTPUT_FOLDER);
        if (!reportDir.exists() && !reportDir.isDirectory()) {
            reportDir.mkdir();
        }

        // 生成SparkReporter报告,v4版本废弃了原htmlreport,改为sparkReporter
//        ExtentSparkReporter sparkReporter = new ExtentSparkReporter(OUTPUT_FOLDER + FILE_NAME);
        sparkReporter = new ExtentSparkReporter(OUTPUT_FOLDER + FILE_NAME);

        sparkReporter.config().setDocumentTitle(REPORT_TITLE);
        sparkReporter.config().setReportName(REPORT_TITLE);
        sparkReporter.config().setEncoding("UTF-8");
        sparkReporter.config().setTheme(Theme.STANDARD);
        sparkReporter.config().setCSS("css-string");
        sparkReporter.config().setJS("js-string");
        sparkReporter.config().setTimeStampFormat("yyyy-MM-dd  HH:mm:ss");
        sparkReporter.config().setTheme(Theme.STANDARD);

        extent = new ExtentReports();
        extent.attachReporter(sparkReporter);
        extent.setReportUsesManualConfiguration(true);
    }

    private void buildTestNodes(ExtentTest extenttest, IResultMap tests, Status status, String[] categories) {
        //存在父节点时,获取父节点的标签
//        String[] categories = new String[0];
//        if (extenttest != null) {
//            List<Attribute> categoryList = extenttest.getModel().getCategoryContext().getAll();
//            categories = new String[categoryList.size()];
//            for (int index = 0; index < categoryList.size(); index++) {
//                categories[index] = categoryList.get(index).getName();
//            }
//        }

        ExtentTest testNode;

        if (tests.size() > 0) {
            //调整用例排序,按时间排序
            Set<ITestResult> treeSet = new TreeSet<ITestResult>(new Comparator<ITestResult>() {
                @Override
                public int compare(ITestResult o1, ITestResult o2) {
                    return o1.getStartMillis() < o2.getStartMillis() ? -1 : 1;
                }
            });
            treeSet.addAll(tests.getAllResults());
            for (ITestResult result : treeSet) {
                String name = "";
                String testName = result.getTestName(); // 获取testName

                //如果有参数,则使用参数的toString组合代替报告中的name
//                Object[] parameters = result.getParameters();
//                if (parameters.length > 0) {
//                    for (Object param : parameters) {
//                        name += param.toString();
//                    }
//                    if (name.length() > 50) {
//                        name = name.substring(0, 49) + "...";
//                    }
//                }

                if (!StringUtils.isEmpty(testName)) {
                    name = testName;
                } else {
                    // 使用方法名称当做用例名称
                    name = result.getMethod().getMethodName();
                }

                if (extenttest == null) {
                    testNode = extent.createTest(name);
                } else {
                    //作为子节点进行创建时,设置标签,便于报告检索。
                    testNode = extenttest.createNode(name).assignCategory(categories);
                }

                // 设置用例的描述信息
                String description = result.getMethod().getDescription();
                if (!StringUtils.isEmpty(description)) {
                    testNode.getModel().setDescription(description);
                }

                for (String group : result.getMethod().getGroups())
                    testNode.assignCategory(group);

                List<String> outputList = Reporter.getOutput(result);
                for (String output : outputList) {
                    //将用例的log输出报告中
                    testNode.debug(output);
                }
                // 有异常则报错,没有则显示Test passed
                if (result.getThrowable() != null) {
                    testNode.log(status, result.getThrowable());
                } else {
                    testNode.log(status, "Test " + status.toString().toLowerCase() + "ed!" + testNode.getModel().getDescription());
                }

                testNode.getModel().setStartTime(getTime(result.getStartMillis()));
                testNode.getModel().setEndTime(getTime(result.getEndMillis()));
            }
        }
    }

    private Date getTime(long millis) {
        Calendar calendar = Calendar.getInstance();
        calendar.setTimeInMillis(millis);
        return calendar.getTime();
    }
}

 

 
 报告

 

 

 

 

 

posted @ 2020-12-22 10:31  星瑞  阅读(523)  评论(0编辑  收藏  举报