jmeter+ant+git+jenkins

打印 上一主题 下一主题

主题 1821|帖子 1821|积分 5463

马上注册,结交更多好友,享用更多功能,让你轻松玩转社区。

您需要 登录 才可以下载或查看,没有账号?立即注册

x
基于工具的接口自动化(jmeter+ant+git+jenkins)

1.1 简介
Jmeter、Ant、Git和Jenkins是一套结合了测试、代码管理和持续集成/持续部署(CI/CD)的工具链,可以帮助开辟者进行高效的接口开辟和测试。这四个工具可以相互配合,共同实现API测试、代码管理和持续集成的目的。
1.2 各工具的作用


  • Jmeter:一个开源的负载测试工具,用于对软件体系进行压力测试和性能测试。它可以模仿多个用户同时访问网站或应用程序,以评估其性能和稳固性。
  • Ant:一个基于Java的构建工具,用于自动化项目的构建过程。它支持多种使命范例,如编译、打包、部署等,可以帮助开辟者快速构建和部署项目。
  • Git:Git是一个版本控制体系,用于跟踪和管理代码的变动。你可以将Jmeter的测试脚本存储在Git仓库中,以便于管理和协作。
  • Jenkins:Jenkins是一个开源的CI/CD服务器,可以自动化各种使命,包括从Git仓库中拉代替码、运行测试、构建和部署应用程序等。你可以在Jenkins中创建一个使命,该使命从Git仓库中拉取Jmeter的集合和测试脚本,然后使用Ant运行这些测试,并将结果输出到Jenkins的构建报告中。
1.3 各工具结合的使用流程
1、编写接口测试计划:


  • 使用Jmeter创建测试计划,包括线程组、HTTP哀求等组件。
  • 在HTTP哀求中设置接口的URL、哀求方法、参数等。
  • 添加断言来验证接口的相应结果。
  • 将测试计划生存为JMeter测试脚本文件(.jmx文件)。
2、配置Ant构建脚本:


  • 创建一个Ant构建文件(build.xml)。
  • 在构建文件中添加Jmeter使命,指定Jmeter测试用例的路径和其他配置。
  • 可以使用Ant的其他使命来执行一些前置或后置操作,如清理环境、天生报告等。
3、提交测试脚本到Git仓库:


  • 在Git中创建一个仓库,用于存储Jmeter测试用例和Ant构建文件。
  • 将Jmeter测试用例和Ant构建文件提交到Git仓库中。
  • 可以使用Git的分支和标签功能来管理差异版本的测试用例和构建文件。
4、配置Jenkins持续集成:


  • Jenkins中创建一个新的项目。
  • 配置项目的源码管理,选择Git,并指定Git仓库的URL和认证信息。
  • 配置构建触发器,可以选择定时触发或者根据代码提交触发构建。
  • 配置构建步骤,选择执行Ant构建,并指定Ant构建文件的路径。
  • 可以配置构建后的操作,如发送邮件、天生测试报告等。

通过上述步骤,可以使用JMeter、Ant、Git和Jenkins进行接口自动化测试,进步测试服从和准确性,低沉人工干预和错误率。同时,通过持续集成和部署,可以确保代码的质量和稳固性。
2.1 Ant简介
Ant,全称Apache Ant,是一个基于Java的构建工具。它的重要目的是将软件工程的编译、测试、部署等步骤自动化,使得用户无需手动执行一系列繁琐的操作。
2.2 Ant结合jmeter的作用
在进行接口测试时,Ant可以帮助我们驱动构建文件中定义的使命,如编译源代码、创建目次、复制文件等。具体来说,Ant可以结合JMeter实现以下功能:


  • 通过JMeter执行接口测试并网络测试结果数据。
  • 使用Ant对JMeter的测试结果数据进行整理和处理。
  • 通过Ant天生定制化的测试报告,包括HTML格式的详细报告和XML格式的结果数据等。
  • 使用Ant将测试报告部署到指定的服务器或者存储位置。
2.3 Ant的构建文件
Ant构建文件,也叫build.xml,是一个基于XML的文件,它是由Apache Ant构建工具使用的。这个文件是Ant运行所需要的,通过调用target树,Ant可以执行各种task,每个task实现了特定接口对象。
JMeter和Ant结合使用时,build.xml文件是一个关键的配置文件,它负责管理和执行所有与JMeter测试相干的使命。在build.xml文件中,你可以定义一系列的目的(targets)以及对应的使命(tasks)。这些使命可以包括:运行JMeter测试、天生测试报告、发送测试结果邮件等。
build.xml文件可参考下方(●标注为需要根据实际情况的修改项):
  1. [/code] [code]
  2. <?xml version="1.0" encoding="UTF-8"?>
  3. <project name="ant-jmeter-test" default="run" basedir=".">
  4.     <tstamp>
  5.         <format property="time" pattern="yyyyMMddHHmm" />
  6.     </tstamp>
  7.     <!-- ●1.需要改成自己本地的Jmeter目录-->
  8.     <property name="jmeter.home" value="xxxx\apache-jmeter-5.5" />
  9.     <!-- ●2.jmeter生成jtl格式的结果报告的路径-->
  10.     <property name="jmeter.result.jtl.dir" value="xxxx\apache-jmeter-x.x\report\jtl" />
  11.     <!-- ●3.jmeter生成html格式的结果报告的路径-->
  12.     <property name="jmeter.result.html.dir" value="xxxx\apache-jmeter-x.x\report\html" />
  13.     <!-- ●4.生成的【报告/日志】的前缀(两个地方):如——>TestReport202303061648-->
  14.     <property name="ReportName" value="TestReport" />
  15.     <property name="jmeter.result.jtlName" value="${jmeter.result.jtl.dir}/${ReportName}${time}.jtl" />
  16.     <property name="jmeter.result.htmlName" value="${jmeter.result.html.dir}/${ReportName}${time}.html" />
  17.     <!-- ●5.输出生成的报告名称和存放路径-->
  18.     <echo message="${jmeter.result.jtlName}"/>
  19.     <echo message="${jmeter.result.htmlName}"/>
  20.     <echo message="${jmeter.result.html.dir}"/>
  21.  
  22.     <target name="run">
  23.         <antcall target="test"/>
  24.         <antcall target="report"/>
  25.     </target>
  26.     <!-- ●6.指定ant-jmeter-x.x.x.jar 的位置 -->
  27.     <path id="jmeter.classpath">
  28.         <fileset dir="${jmeter.home}/extras">
  29.             <include name="ant-jmeter-x.x.x.jar"/>
  30.         </fileset>
  31.  
  32.     </path>
  33.     <!--加载jar包,解决显示时间问题-->
  34.     <path id="xslt.classpath">
  35.         <fileset dir="${jmeter.home}/lib" includes="xalan*.jar"/>
  36.         <fileset dir="${jmeter.home}/lib" includes="serializer*.jar"/>
  37.     </path>
  38.  
  39.     <target name="test">
  40.         <taskdef name="jmeter" classname="org.programmerplanet.ant.taskdefs.jmeter.JMeterTask" />
  41.         <jmeter jmeterhome="${jmeter.home}" resultlog="${jmeter.result.jtlName}">
  42.             <!-- ●7.声明要运行的脚本。"*.jmx"指包含此目录下的所有jmeter脚本,比如test*。jmx表示以test开头的所有.jmx文件-->
  43.             <testplans dir="xxx/apache-jmeter-x.x.x/xxx" includes="*.jmx" />
  44.             <!-- ●8.声明ant执行jmeter时,传入jmeter的属性值,可以自定义必须是xml格式 -->
  45.             <property name="jmeter.save.saveservice.output_format" value="xml"/>
  46.         </jmeter>
  47.     </target>
  48.  
  49.     <target name="report">
  50.     <tstamp> <format property="report.datestamp" pattern="yyyy-MM-dd HH:mm" /></tstamp>
  51.         <xslt
  52.               classpathref="xslt.classpath"
  53.               force="true"
  54.               in="${jmeter.result.jtlName}"
  55.               out="${jmeter.result.htmlName}"
  56.               style="${jmeter.home}/extras/jmeter-results-detail-report_21.xsl">
  57.               <!--jmeter-results-detail-report_21.xsl这里的文件名可以换成你想要的报告效果-->
  58.               <!--显示dateReport的时间-->
  59.               <param name="dateReport" expression="${report.datestamp}"/>
  60.         </xslt>
  61.         <!-- ●9因为上面生成报告的时候,不会将相关的图片也一起拷贝至目标目录,所以,需要手动拷贝 -->
  62.         <copy todir="${jmeter.result.html.dir}">
  63.             <fileset dir="${jmeter.home}/extras">
  64.                 <include name="collapse.png" />
  65.                 <include name="expand.png" />
  66.             </fileset>
  67.         </copy>
  68.     </target>
  69. </project>
复制代码
3.1 准备接口测试脚本


  • 编写并调试好接口测试脚本




  • 生存接口测试脚本为.jmx



  • 当地通过ant运行脚本进行调试

3.2 将测试脚本上传到git长途仓库


  • 在gitee新建一个长途仓库



  • 将当地测试脚本上传推送到长途仓库


3.4 配置jmeter+ant环境(docker)


  • 下载jmeter和ant的安装包,并进行解压



  • 将jmeter和ant移动docker的jenkins容器挂载目次中



  • 启动Jenkins容器并进入容器内



  • 配置环境变量
  1. vim /etc/profile
  2. # 添加下面的内容
  3. ANT_HOME=/var/jenkins_home/apache-ant-1.10.14
  4. export PATH=$PATH:$ANT_HOME/bin
  5. export PATH=$PATH:$ANT_HOME/lib
  6. export JMETER_HOME=/var/jenkins_home/apache-jmeter-5.6.3
  7. export PATH=$JMETER_HOME/bin:$PATH:$HOME/bin
复制代码

  1. # 刷新环境变量
  2. source /etc/profile
  3. # 验证ant环境
  4. ant -version
  5. # 验证jmeter环境
  6. jmeter -v
复制代码



  • 将apache-jmeter-5.6.3\extras目次下的:ant-jmeter-1.1.1.jar 复制到apache-ant-1.10.14\lib\



  • jenkins当前构建使命的工作目次创建html、htl文件夹



  • jenkins当前构建使命的工作目次新增build.xml,并进行配置
  1. [/code] [code]
  2. <?xml version="1.0" encoding="UTF-8"?>
  3. <project name="ant-jmeter-test" default="run" basedir=".">
  4. <tstamp>
  5. <format property="time" pattern="yyyyMMddHHmm" />
  6. </tstamp>
  7. <!-- ●1.需要改成自己本地的Jmeter目录-->
  8. <property name="jmeter.home" value="\var\jenkins_home\apache-jmeter-5.6.3" />
  9. <!-- ●2.jmeter生成jtl格式的结果报告的路径-->
  10. <property name="jmeter.result.jtl.dir" value="\var\jenkins_home\workspace\jmeter+ant+git+jenkins接口自动化案例\jtl" />
  11. <!-- ●3.jmeter生成html格式的结果报告的路径-->
  12. <property name="jmeter.result.html.dir" value="\var\jenkins_home\workspace\jmeter+ant+git+jenkins接口自动化案例\html" />
  13. <!-- ●4.生成的【报告/日志】的前缀(两个地方):如——>TestReport202303061648-->
  14. <property name="ReportName" value="TestReport" />
  15. <property name="jmeter.result.jtlName" value="${jmeter.result.jtl.dir}/${ReportName}${time}.jtl" />
  16. <property name="jmeter.result.htmlName" value="${jmeter.result.html.dir}/${ReportName}${time}.html" />
  17. <!-- ●5.输出生成的报告名称和存放路径-->
  18. <echo message="${jmeter.result.jtlName}"/>
  19. <echo message="${jmeter.result.htmlName}"/>
  20. <echo message="${jmeter.result.html.dir}"/>
  21. <target name="run">
  22. <antcall target="test"/>
  23. <antcall target="report"/>
  24. </target>
  25. <!-- ●6.指定ant-jmeter-1.1.1.jar 的位置 -->
  26. <path id="jmeter.classpath">
  27. <fileset dir="${jmeter.home}/extras">
  28. <include name="ant-jmeter-1.1.1.jar"/>
  29. </fileset>
  30. </path>
  31. <!--加载jar包,解决显示时间问题-->
  32. <path id="xslt.classpath">
  33. <fileset dir="${jmeter.home}/lib" includes="xalan*.jar"/>
  34. <fileset dir="${jmeter.home}/lib" includes="serializer*.jar"/>
  35. </path>
  36. <target name="test">
  37. <taskdef name="jmeter" classname="org.programmerplanet.ant.taskdefs.jmeter.JMeterTask" />
  38. <jmeter jmeterhome="${jmeter.home}" resultlog="${jmeter.result.jtlName}">
  39. <!-- ●7.声明要运行的脚本。"*.jmx"指包含此目录下的所有jmeter脚本,比如test*。jmx表示以test开头的所有.jmx文件-->
  40. <testplans dir="/var/jenkins_home/workspace/jmeter+ant+git+jenkins接口自动化案例" includes="*.jmx" />
  41. <!-- ●8.声明ant执行jmeter时,传入jmeter的属性值,可以自定义必须是xml格式 -->
  42. <property name="jmeter.save.saveservice.output_format" value="xml"/>
  43. </jmeter>
  44. </target>
  45. <target name="report">
  46. <tstamp> <format property="report.datestamp" pattern="yyyy-MM-dd HH:mm" /></tstamp>
  47. <xslt
  48. classpathref="xslt.classpath"
  49. force="true"
  50. in="${jmeter.result.jtlName}"
  51. out="${jmeter.result.htmlName}"
  52. style="${jmeter.home}/extras/jmeter-results-detail-report_21.xsl">
  53. <!--jmeter-results-detail-report_21.xsl这里的文件名可以换成你想要的报告效果-->
  54. <!--显示dateReport的时间-->
  55. <param name="dateReport" expression="${report.datestamp}"/>
  56. </xslt>
  57. <!-- ●9因为上面生成报告的时候,不会将相关的图片也一起拷贝至目标目录,所以,需要手动拷贝 -->
  58. <copy todir="${jmeter.result.html.dir}">
  59. <fileset dir="${jmeter.home}/extras">
  60. <include name="collapse.png" />
  61. <include name="expand.png" />
  62. </fileset>
  63. </copy>
  64. </target>
  65. </project>
复制代码



  • 修改天生的报告,并进行配置
1、在/apache-jmeter-5.6.3/extras中新增jmeter-results-shanhe-me.xsl文件
  1. [/code] [code]
  2. <?xml version="1.0" encoding="UTF-8"?>
  3. <xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0">
  4. <xsl:output method="html" indent="no" encoding="UTF-8" doctype-public="-//W3C//DTD HTML 4.01 Transitional//EN" doctype-system="http://www.w3.org/TR/html4/loose.dtd"/>
  5. <xsl:strip-space elements="*"/>
  6. <xsl:template match="/testResults">
  7. <html lang="en">
  8. <head>
  9. <meta name="Author" content="shanhe.me"/>
  10. <title>JMeter Test Results</title>
  11. <style type="text/css"><![CDATA[
  12. * { margin: 0; padding: 0 }
  13. html, body { width: 100%; height: 100%; background: #b4b4b4; font-size: 12px }
  14. table { border: none; border-collapse: collapse; table-layout: fixed }
  15. td { vertical-align: baseline; font-size: 12px }
  16. #left-panel { position: absolute; left: 0; top: 0; bottom: 0; width: 300px; overflow: auto; background: #dee4ea }
  17. #left-panel li.navigation { font-weight: bold; cursor: default; color: #9da8b2; line-height: 18px; background-position: 12px 5px; background-repeat: no-repeat; padding: 0 0 0 25px; background-image: url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAkAAAAICAYAAAArzdW1AAAAAXNSR0IArs4c6QAAAAZiS0dEAP8A/wD/oL2nkwAAAAlwSFlzAAALEwAACxMBAJqcGAAAAAd0SU1FB9sDEBQqGbO7BEcAAAAdaVRYdENvbW1lbnQAAAAAAENyZWF0ZWQgd2l0aCBHSU1QZC5lBwAAAKRJREFUGNN1zM0KgkAYheF3RvtXSsGyWhRNaILS7bdt11W0KgJvoPwZp0UlBPUtz3nOJw7Hk7necv5dOA2Qaazo2vZP0LEt9olCVtqQROufKNmuqBuBNAYW4QzXGX6B0bDPcjGnMQYJ8Cg12U59oSzaUJQa4IUAXMclDHwAAn/MxPMw765FZd2QRgopBWmsKCrdfhXnS/4ZYElBXdyxewN008Y8AephLAkqz613AAAAAElFTkSuQmCC) }
  18. #left-panel li.success { color: #565b60 }
  19. #left-panel li.failure { color: red }
  20. #left-panel li { list-style: none; color: black; cursor: pointer }
  21. #left-panel li.selected { background-repeat: repeat-x; color: white; background: url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAEAAAAUCAYAAABMDlehAAAAAXNSR0IArs4c6QAAAAZiS0dEAP8A/wD/oL2nkwAAAAlwSFlzAAALEwAACxMBAJqcGAAAAAd0SU1FB9sDEBQxLTs5O2gAAAAdaVRYdENvbW1lbnQAAAAAAENyZWF0ZWQgd2l0aCBHSU1QZC5lBwAAAEdJREFUCNc1y7ERgEAMA0GNUhIyGqM2uqKgtyWZhE9v53A/7/A6D7BkMDNgy2AroB2wHTCZv5UMOgFLG1bvd7XBckBlwCXjA5wMOF5iOX/MAAAAAElFTkSuQmCC) }
  22. #left-panel div { line-height: 20px; background-position: 25px 3px; background-repeat: no-repeat; padding: 0 0 0 45px }
  23. #left-panel div.success { background-image: url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAA8AAAAOCAYAAADwikbvAAAAAXNSR0IArs4c6QAAAAZiS0dEAP8A/wD/oL2nkwAAAAlwSFlzAAALEwAACxMBAJqcGAAAAAd0SU1FB9sDEBULEEc6wzcAAAAdaVRYdENvbW1lbnQAAAAAAENyZWF0ZWQgd2l0aCBHSU1QZC5lBwAAAiNJREFUKM99kktIVGEYhp/jzJl08lI6logp2Y2EFkbtaqlFROsWrlq4ioJWQRs37VoUVItWkYEVRGSBlhleCpywDEWxTEuxcURTZ6YzxzP/5WshCOHUt36f93kXnyMi5Lsnb4clI4s4fhkXzp5w8mWcfHBvfEpUxVdCUUU6lUPNHuD86cYtBQX5GhPrM7hRg7GaSDRg2vuUd90WuOPVsOyqy6FFo2yOQHlU1S9z9dZT+S/8I7GCLlkAN4eyAf56mnT6Fy1HLnGuuYa++MS/4e74qMRqfXLaJ9BpfnsrLC0m2BYuoqwUbj/+274JD43OEqmexwvW8NUKXnaZtVSS1pNtAAyOvyC6v48HnUNb4Z7PH8UtTlIQWA5tb2RhYY7kz3l2FleytJYg/qWb8t2KZ/0PN+1hgI6uEUr2jpHKpGlquExVaS0VbjUZL7WxaqIXK6ADQ0n9GNfv9XCttWnD/O57t0TKFklnF3g5fJ/seoaa2D4O1x0F4PlgO9oIftbgFgYMfLgjACGqj0vlsddoUnj+Kt/mxunq72RP+UGqYjWMTA7R+b6dUCSEGEF5hoJQip6BaFs4HJtCyRrKs6wHCovDip/kys0WWpovMpOYBCtoT2N9B5uzWG0Zid8gnFrVFEQDtBaUrxEgXBimaEeER2/uIiK4roPOaMRYjBKsFly3fOO3G06dETGCWIsYjckprMphtEKMAQtgsMYi1mJMQHJ6xvkDKQoyphCzkl0AAAAASUVORK5CYII=) }
  24. #left-panel div.failure { background-image: url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAA8AAAAOCAYAAADwikbvAAAAAXNSR0IArs4c6QAAAAZiS0dEAP8A/wD/oL2nkwAAAAlwSFlzAAALEwAACxMBAJqcGAAAAAd0SU1FB9sDEBUJOEC5CU8AAAAdaVRYdENvbW1lbnQAAAAAAENyZWF0ZWQgd2l0aCBHSU1QZC5lBwAAAeVJREFUKM+NkDtok2EUhp8vl9ZLo/EyKI6KFgqCKC4OClrBWUQEcRRx1cGpk3WyInWrgoMZKkW8thYaEYQ0i7WC2ngrNDTERHJvkv/L/3//dxwc7F8jeOAsh/c973OOEhG61aPnaen7maXYt4MLZ4+pbppQt+F06jNH3QWOb8pxUs+SmJzjv83hxY8SVy3wNdtVneiHqe54IhLoB4/TUkyMyOrKj5yXoVtPZK02kLyYK7OnlqFWzgcCGtUC/YUJ3n5a/jd28tU7ORTN0myUA6Jms8bpWIa798elqzn1fokjThrpVBC3ETzNbYAuca59j/Hp+b/N869Tsk8tgVMCXQk+RlfQuk1/tMLMwzsSMCcm5zjhvoR2AdpF0GuwO4aqttS05ZSbZHhsBoAIwI83Cdkd/460XDAOG02d24MxvlR8dsUUh3f2UHaEtgdbWCHz4oZwcVCp66PP5FLhKjEc8DXaCMsNy8DYn/SnZ+L0hhWOb/F8yLs9fDtwk8j+VpqwrlC34PrgGEu2bhlYhZ1b8dncq3AMeBaUr/k6NUyk4ChKzu+N2hc6Bqody+WDG8g2fLatD7F3axjPgmvAtYJvIbouhhIRrl0ZktnkBGIt1gqeMXQ8D2MMiCIUCqFEsFhEQMSykCuqX0MzLAUJTzRsAAAAAElFTkSuQmCC) }
  25. #left-panel div.detail { display: none }
  26. #right-panel { position: absolute; right: 0; top: 0; bottom: 0; left: 301px; overflow: auto; background: white }
  27. #right-panel .group { font-size: 12px; font-weight: bold; line-height: 16px; padding: 0 0 0 18px; counter-reset: assertion; background-repeat: repeat-x; background-image: url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAEAAAAQCAYAAADXnxW3AAAAAXNSR0IArs4c6QAAAAZiS0dEAP8A/wD/oL2nkwAAAAlwSFlzAAALEwAACxMBAJqcGAAAAAd0SU1FB9sDEBUkDq8pxjkAAAAdaVRYdENvbW1lbnQAAAAAAENyZWF0ZWQgd2l0aCBHSU1QZC5lBwAAADdJREFUCNdVxrERwDAMAzGK0v47eS6Z927SpMFBAAbkvSvnRk5+7K5cVfLMyN39bWakJAjA5xw9R94jN3tVhVEAAAAASUVORK5CYII=) }
  28. #right-panel .zebra { background-repeat: repeat; padding: 0 0 0 18px; background-image: url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAEAAAAmCAYAAAAFvPEHAAAAAXNSR0IArs4c6QAAAAZiS0dEAP8A/wD/oL2nkwAAAAlwSFlzAAALEwAACxMBAJqcGAAAAAd0SU1FB9sDEBYWFlNztEcAAAAdaVRYdENvbW1lbnQAAAAAAENyZWF0ZWQgd2l0aCBHSU1QZC5lBwAAABdJREFUCNdjYKAtePv5338mBgYGBpoQAGy1BAJlb/y6AAAAAElFTkSuQmCC) }
  29. #right-panel .data { line-height: 19px; white-space: nowrap }
  30. #right-panel pre.data { white-space: pre }
  31. #right-panel tbody.failure { color: red }
  32. #right-panel td.key { min-width: 108px }
  33. #right-panel td.delimiter { min-width: 18px }
  34. #right-panel td.assertion:before { counter-increment: assertion; content: counter(assertion) ". " }
  35. #right-panel td.assertion { color: black }
  36. #right-panel .trail { border-top: 1px solid #b4b4b4 }
  37. ]]></style>
  38. <script type="text/javascript"><![CDATA[
  39. var onclick_li = (function() {
  40. var last_selected = null;
  41. return function(li) {
  42. if( last_selected == li )
  43. return;
  44. if( last_selected )
  45. last_selected.className = "";
  46. last_selected = li;
  47. last_selected.className = "selected";
  48. document.getElementById("right-panel").innerHTML = last_selected.firstChild.nextSibling.innerHTML;
  49. return false;
  50. };
  51. })();
  52. var patch_timestamp = function() {
  53. var spans = document.getElementsByTagName("span");
  54. var len = spans.length;
  55. for( var i = 0; i < len; ++i ) {
  56. var span = spans[i];
  57. if( "patch_timestamp" == span.className )
  58. span.innerHTML = new Date( parseInt( span.innerHTML ) );
  59. }
  60. };
  61. var patch_navigation_class = (function() {
  62. var set_class = function(el, flag) {
  63. if(el) {
  64. el.className += flag ? " success" : " failure";
  65. }
  66. };
  67. var traverse = function(el, group_el, flag) {
  68. while(1) {
  69. if(el) {
  70. if(el.className == 'navigation') {
  71. set_class(group_el, flag);
  72. group_el = el;
  73. flag = true;
  74. } else {
  75. var o = el.firstChild;
  76. o = o ? o.className : null;
  77. flag = flag ? (o == 'success') : false;
  78. }
  79. el = el.nextSibling;
  80. } else {
  81. set_class(group_el, flag);
  82. break;
  83. }
  84. }
  85. };
  86. return function() {
  87. var o = document.getElementById("result-list");
  88. o = o ? o.firstChild : null;
  89. if(o)
  90. traverse(o, null, true);
  91. };
  92. })();
  93. window.onload = function() {
  94. patch_timestamp();
  95. patch_navigation_class();
  96. var o = document.getElementById("result-list");
  97. o = o ? o.firstChild : null;
  98. o = o ? o.nextSibling : null;
  99. if(o)
  100. onclick_li(o);
  101. };
  102. ]]></script>
  103. </head>
  104. <body>
  105. <div id="left-panel">
  106. <ol id="result-list">
  107. <xsl:for-each select="*">
  108. <!-- group with the previous sibling -->
  109. <xsl:if test="position() = 1 or @tn != preceding-sibling::*[1]/@tn">
  110. <li class="navigation">Thread: <xsl:value-of select="@tn"/></li>
  111. </xsl:if>
  112. <li onclick="return onclick_li(this);">
  113. <div>
  114. <xsl:attribute name="class">
  115. <xsl:choose>
  116. <xsl:when test="@s = 'true'">success</xsl:when>
  117. <xsl:otherwise>failure</xsl:otherwise>
  118. </xsl:choose>
  119. </xsl:attribute>
  120. <xsl:value-of select="@lb"/>
  121. </div><div class="detail">
  122. <div class="group">Sampler</div>
  123. <div class="zebra">
  124. <table>
  125. <tr><td class="data key">Thread Name</td><td class="data delimiter">:</td><td class="data"><xsl:value-of select="@tn"/></td></tr>
  126. <tr><td class="data key">Timestamp</td><td class="data delimiter">:</td><td class="data"><span class="patch_timestamp"><xsl:value-of select="@ts"/></span></td></tr>
  127. <tr><td class="data key">Time</td><td class="data delimiter">:</td><td class="data"><xsl:value-of select="@t"/> ms</td></tr>
  128. <tr><td class="data key">Latency</td><td class="data delimiter">:</td><td class="data"><xsl:value-of select="@lt"/> ms</td></tr>
  129. <tr><td class="data key">Bytes</td><td class="data delimiter">:</td><td class="data"><xsl:value-of select="@by"/></td></tr>
  130. <tr><td class="data key">Sample Count</td><td class="data delimiter">:</td><td class="data"><xsl:value-of select="@sc"/></td></tr>
  131. <tr><td class="data key">Error Count</td><td class="data delimiter">:</td><td class="data"><xsl:value-of select="@ec"/></td></tr>
  132. <tr><td class="data key">Response Code</td><td class="data delimiter">:</td><td class="data"><xsl:value-of select="@rc"/></td></tr>
  133. <tr><td class="data key">Response Message</td><td class="data delimiter">:</td><td class="data"><xsl:value-of select="@rm"/></td></tr>
  134. </table>
  135. </div>
  136. <div class="trail"></div>
  137. <xsl:if test="count(assertionResult) &gt; 0">
  138. <div class="group">Assertion</div>
  139. <div class="zebra">
  140. <table>
  141. <xsl:for-each select="assertionResult">
  142. <tbody>
  143. <xsl:attribute name="class">
  144. <xsl:choose>
  145. <xsl:when test="failure = 'true'">failure</xsl:when>
  146. <xsl:when test="error = 'true'">failure</xsl:when>
  147. </xsl:choose>
  148. </xsl:attribute>
  149. <tr><td class="data assertion" colspan="3"><xsl:value-of select="name"/></td></tr>
  150. <tr><td class="data key">Failure</td><td class="data delimiter">:</td><td class="data"><xsl:value-of select="failure"/></td></tr>
  151. <tr><td class="data key">Error</td><td class="data delimiter">:</td><td class="data"><xsl:value-of select="error"/></td></tr>
  152. <tr><td class="data key">Failure Message</td><td class="data delimiter">:</td><td class="data"><xsl:value-of select="failureMessage"/></td></tr>
  153. </tbody>
  154. </xsl:for-each>
  155. </table>
  156. </div>
  157. <div class="trail"></div>
  158. </xsl:if>
  159. <div class="group">Request</div>
  160. <div class="zebra">
  161. <table>
  162. <tr><td class="data key">Method/Url</td><td class="data delimiter">:</td><td class="data"><pre class="data"><xsl:value-of select="method"/><xsl:text> </xsl:text><xsl:value-of select="java.net.URL"/></pre></td></tr>
  163. <tr><td class="data key">Query String</td><td class="data delimiter">:</td><td class="data"><pre class="data"><xsl:value-of select="queryString"/></pre></td></tr>
  164. <tr><td class="data key">Cookies</td><td class="data delimiter">:</td><td class="data"><pre class="data"><xsl:value-of select="cookies"/></pre></td></tr>
  165. <tr><td class="data key">Request Headers</td><td class="data delimiter">:</td><td class="data"><pre class="data"><xsl:value-of select="requestHeader"/></pre></td></tr>
  166. </table>
  167. </div>
  168. <div class="trail"></div>
  169. <div class="group">Response</div>
  170. <div class="zebra">
  171. <table>
  172. <tr><td class="data key">Response Headers</td><td class="data delimiter">:</td><td class="data"><pre class="data"><xsl:value-of select="responseHeader"/></pre></td></tr>
  173. <tr><td class="data key">Response Data</td><td class="data delimiter">:</td><td class="data"><pre class="data"><xsl:value-of select="responseData"/></pre></td></tr>
  174. <tr><td class="data key">Response File</td><td class="data delimiter">:</td><td class="data"><pre class="data"><xsl:value-of select="responseFile"/></pre></td></tr>
  175. </table>
  176. </div>
  177. <div class="trail"></div>
  178. </div>
  179. </li>
  180. </xsl:for-each>
  181. </ol>
  182. </div>
  183. <div id="right-panel"></div>
  184. </body>
  185. </html>
  186. </xsl:template>
  187. </xsl:stylesheet>
复制代码

2、在jmeter的bin目次下找到jmeter.properties,设置需要输出的内容为true,并去掉前面的解释符号#

3、将build.xml中报告模板更换成jmeter.results.shanhe.me.xsl



  • 此目次(Jenkins需提前新增一个jmeter+ant+git+jenkins接口自动化案例名称的使命)执行脚本并在html和jtl文件夹天生报告,至此Jenkins容器内jmeter+ant环境就搭建完成了。

3.4 jenkins配置


  • 新建使命




  • 配置使命-源码管理
通过jenkins的git插件拉取长途仓库的接口测试脚本。



  • 配置使命-构建触发器



  • 配置使命-构建环境

(jdk和ant需要先到体系管理->全局工具配置进行配置)



  • 配置使命-构建步骤1
通过执行shell脚本删除历史构建天生的报告文件。
  1. [/code] [list=1]
  2. [*] #!/bin/bash
  3. [*] cd ${WORKSPACE}
  4. [*] rm -rf ./jtl/*.jtl
  5. [*] rm -rf ./html/*.html
  6. [/list] [align=center][img=606,315]https://i-blog.csdnimg.cn/blog_migrate/020d7c5d40277a56861c33d3666b288b.png[/img][/align]
  7. [list]
  8. [*] 配置使命-构建步骤2
  9. [/list]  执行ant下令进行构建。
  10. [align=center][img=439,441]https://i-blog.csdnimg.cn/blog_migrate/89e706ee665143e989f03ac76da1f3f3.png[/img][/align]
  11. [list]
  12. [*] 配置使命-构建步骤3
  13. [/list] 防止在jenkins中打开html测试报告无样式的问题,需要执行下方Groovy代码。
  14. [align=center][img=503,522]https://i-blog.csdnimg.cn/blog_migrate/18cc6448a396708b86d559cb6e10c719.png[/img][/align]
  15. [code]System.setProperty("hudson.model.DirectoryBrowserSupport.CSP", "")
复制代码


  • 构建后操作-天生测试报告
jenkins插件管理中安装HTML Publisher plugin,需要配置已存储的html报告位置以及报告标题。



  • 构建后操作-发送邮件
发送邮件可参考上篇文章《postman+newman+git+jenkins》的邮件发送配置,配置方法是同等的。
3.3 执行Jenkins使命


  • 执行使命



  • 查看使命执行日志



  • 查看测试报告


至此,一个简单的基于工具(jmeter+ant+git+jenkins)的接口自动化测试案例就完成了!
最后感谢每一个认真阅读我文章的人,投桃报李总是要有的,固然不是什么很值钱的东西,如果你用得到的话可以直接拿走:【文末自行领取】


这些资料,对于【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也伴随上万个测试工程师们走过最艰巨的路程,盼望也能帮助到你!


免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

徐锦洪

论坛元老
这个人很懒什么都没写!
快速回复 返回顶部 返回列表