基于Jmeter和Jenkins接口自动化测试框架搭建详细过程_oaa608868的博客-程序员ITS301

技术标签: Jenkins  Jmeter  



1. 下载地址

Jmeter: http://jmeter.apache.org/download_jmeter.cgi

Ant:http://ant.apache.org/bindownload.cgi

Jenkins:https://jenkins.io/index.html

2. Jmeter+Ant安装配置

  • Jmeter解压后运行Jmeter.bat即可使用
  • Ant下载后解压到目录,按照下列配置系统变量即可:

     1) ANT_HOME:D:\apache-ant-1.9.4-bin\apache-ant-1.9.4(此为ANT安装路径)

     2) PATH(后面添加以下内容):;%ANT_HOME%\bin

     3) 测试Ant:在CMD中输入ant –version,如果返回ant版本,则表示安装成功

  • 将JMeter目录下extras子目录里ant-JMeter-1.1.1.jar复制到Ant的lib目录下,这样Ant运行时才能找到,"org.programmerplanet.ant.taskdefs.jmeter.JMeterTask"这个类,从而成功触发JMeter脚本。
  • jmeter默认保存的是.csv格式的文件,所以我们先要设置一下bin/jmeter.properties文件内容,保存

    jmeter.save.saveservice.output_format=xml

  • 配置ant构建文件build.xml(见附录)

3. Jenkins安装与配置

Jenkins安装比较简单,点击msi文件按照要求操作即可,过程若出现问题,可重启下jenkins服务后,再继续安装;

接下来主要介绍Jenkins构建ant项目的配置:

浏览器输入地址http://localhost:8080/显示界面,则表示安装成功

4. shell运行环境搭建

安装cygwin后,配置环境变量

l 按照下列配置系统变量即可:

1) JAVA_HOME:D:\ Program Files\Java\jdk1.8.0_11(此为JDK安装路径)

2) ANT_HOME:D:\apache-ant-1.9.4-bin\apache-ant-1.9.4(此为ANT安装路径)

3) PATH(后面添加以下内容):

;%JAVA_HOME%\bin;%JAVA_HOME%\jre\bin;%ANT_HOME%\bin;E:\Program Files\cygwin\bin

4) 测试Ant:在CMD中输入ant –version,如果返回ant版本,则表示安装成功

5) 测试Java:在CMD中输入 java –version,如果返回java版本,则表示安装成功

6) 测试shell运行环境:在CMD中输入sh,如果为提示错误,则表示安装成功

5. Testlink与Jenkins对接

1、先安装过程所需要的插件:Testlink Plugin、TAP Plugin

直接在系统管理—>管理插件—>可选插件,中搜索后,点击安装即可

注意:TAP插件必须用1.25版本的,新版本发布后tap文件链接失效

2、 系统管理—>系统设置,Testlink配置如下:

Name:TestLink 1.9.10

URL:http://192.168.4.56/testlink/lib/api/xmlrpc/v1/xmlrpc.php

Developer Key:登录TestLink后,选择个人账号—生成新的秘钥,API Key拷贝到这里即可

6. TestLink配置

Testlink需要增加自定义字段,如AutoTest,具体的用例如下:

注意两点:
1)测试方式:自动的

2)AutoTest:内容为对应该条用例的.tap文件的文件名

7. 构建项目

7.1. 准备测试脚本到SVN

上图是SVN中对应的文件结构,下面会介绍jenkins构建项目的详细配置。

7.2. 新建一个自由风格的软件项目

旧的构建保存了你在一次构建中的所有临时文件,如果构建没有保留的必要,就勾选丢弃旧的构建,同时设置保持构建天使和保持构建的最大个数两个参数。注意:三个设置必须同时设置,否则无效

7.3. 源码管理

选择subversion

Repository URL:SVN中脚本目录

Credentials:SVN账号用户名和密码,可点击后面的Add添加

Local module directory:SVN下载到本地的目录,配置为【.】默认项目根目录下【注:这个与Ant的build.xml文件中的配置相关联,默认的文件是根目录,若这里有修改,需要修改对应的build.xml文件

7.4. 触发器配置

这里是让脚本每天晚上自动构建执行,若不需要重复执行,则不要勾选该项接口

7.5. 构建Invoke TestLink

注:

1、 TestLink Configuration中TestProjectName和TestPlan Name需要与testlink中的【测试项目】和测【测试计划】一致

2、 TestLink Configuration中BuildName:

1) 可以与Testlink设置一致(这样每次测试结果会在同一个测试版本中记录多次结果)

2) 也可以设置由Jenkins配置返回,如:上图中的$BUILD_TAG,则表示将构建这次项目的标记作为版本返回到TestLink上,这样每一次构建,Testlink上都会新增一个版本

3、 Targets:这里是指build.xml文件中的<target name=“xxx”>,可设置为空,则默认全部执行

4、 这里的Build File是指Ant的build.xml所在目录,若前面SVN目录有修改,这里也需要一并修改

5、 After iterating all test cases Build Steps:这里运行jmx.sh文件,将Jmeter运行结果XML格式,转换为TAP格式,以便TestLink转换为测试结果

这个配置是将TAP结果返回到Testlink,

注意:

Include Pattern:这里的路径与build.xml文件中输出的路径一致。

7.6. TAP结果发布

Test results:结果路径配置,与build.xml中配置输出路径一致

7.7. 构建后邮件发送

增加构建后操作:Editable Email Notification

7.8. 系统配置:邮件的全局参数配置

未截图的按默认配置

备注:

Default Subject:用例执行结果:$PROJECT_NAME - Build # $BUILD_NUMBER - $BUILD_STATUS!

Default Content

复制代码
${FILE,path="sum/sum.html"}<hr/> 

(本邮件是程序自动下发的,请勿回复!)<br/> 

项目名称:$PROJECT_NAME<br/> 

构建编号:$BUILD_NUMBER<br/> 

svn版本号:${SVN_REVISION}<br/> 

构建状态:$BUILD_STATUS<br/> 

触发原因:${CAUSE}<br/> 

构建日志地址:<a href="${BUILD_URL}console">${BUILD_URL}console</a><br/><hr/> 

构建地址:<a href="$BUILD_URL">$BUILD_URL</a><br/><hr/> 

变更集:${JELLY_SCRIPT,template="html"}<br/><hr/>
复制代码

邮件通知配置,根据邮箱实际情况进行配置即可,如下:

 

8.附录

JMX2TAP.SH内容:

 1 #!/bin/bash
 2 LANG=zh_CN.UTF-8
 3 export LANG
 4 # Parse each result file from the tests, created by JMeter when run by ant.
 5 # Every file created by JMeter will be parsed consecutively by the following for loop.
 6 for XML_resultfile in $(ls -1 results/*.xml); do
 7     echo "XML_resultfile: ${XML_resultfile}"
 8     BASE_filename=$(basename ${XML_resultfile} .xml)  
 9     TestSamples=( $(xpath  ${XML_resultfile} '//@lb' 2>/dev/null) ) 
10     num_of_TestSamples=${#TestSamples[@]}
11      echo "num_of_TestSamples: ${num_of_TestSamples}"
12   # Reference to TAP13 file standard:
13   # link:http://www.4pan.net
14   # result_selector=0 
15   if [ ${num_of_TestSamples} -gt 0 ]; then
16       for ((line=0; ${line} < ${num_of_TestSamples}; line++)); do    
17         echo "------------------CASE ${line}--------------------------"    
18         echo "TestCases fullname: ${TestSamples[${line}]}"
19         isFlag=$( echo ${TestSamples[${line}]} | grep '.*-.*-.*(.\{1,\}):.*')
20         if [ "${isFlag}" != "" ]; then
21             filename=$( echo ${TestSamples[${line}]} | cut -d '"' -f 2 | cut -d '(' -f 1 )
22             TAP_filename=${filename}.tap
23             isExistTap=""
24             isExistTap=$(ls -R | grep "${TAP_filename}")
25             if [ "${isExistTap}" == "" ]; then
26                 TestCase=$( echo ${TestSamples[${line}]} | cut -d '"' -f 2 | cut -d ':' -f 1 )
27                 num_of_testcases=1
28                 TestCase_count1=( $(xpath ${XML_resultfile} "//httpSample[contains(@lb,'${filename}')]/@rc" 2>/dev/null) )
29                 TestCase_count2=( $(xpath ${XML_resultfile} "//sample[contains(@lb,'${filename}')]/@rc" 2>/dev/null) )
30                 ((num_of_testcases=${#TestCase_count1[@]}+${#TestCase_count2[@]}))
31                 echo "TestCase:${TestCase}"
32                 echo "filename:${TAP_filename}"
33                 echo "TestCase_count:${num_of_testcases}"        
34                 label="${BASE_filename} - ${TestCase}"        
35                 echo "TAP version 13" > results/${TAP_filename}
36                 echo ">>>${BASE_filename}.xml Covert to : ${TAP_filename}"    
37                 echo "1..${num_of_testcases}" >> results/${TAP_filename}
38                 # Create an array with testcases labels and results
39                 result_if1=( $(xpath ${XML_resultfile} "//httpSample[contains(@lb,'${filename}')]/assertionResult[failure='true']/failure" 2>/dev/null | tr -d '"' | tr ' ' '_' | sed 's/^_//') )
40                 result_if2=( $(xpath ${XML_resultfile} "//sample[contains(@lb,'${filename}')]/assertionResult[failure='true']/failure" 2>/dev/null | tr -d '"' | tr ' ' '_' | sed 's/^_//') )
41                 ((num_of_resultif_count=${#result_if1[@]}+${#result_if2[@]}))
42                 echo ">TestResult:${num_of_resultif_count}"
43                 #echo "number_result_if:${#result_if[@]}"
44                 if [ ${num_of_resultif_count} -gt 0 ]; then
45                     msg_texts=( $(xpath ${XML_resultfile} "//httpSample[contains(@lb,'${filename}')]/assertionResult[failure='true']/failureMessage" 2>/dev/null | tr -d '"' | tr ' ' '_' | sed 's/^_//') )
46                     result="not ok ${num_of_testcases}  ${label} - error[${num_of_resultif_count}]"
47                     echo "${result}">> results/${TAP_filename}
48                     echo "failureMessage:${msg_texts}">> results/${TAP_filename}                    
49                     #echo "${result}"
50                     #echo "failureMessage:${msg_texts}"
51                     #(( line=${line}+${num_of_testcases}-1 ))            
52                             #echo "msg_text:${msg_texts}">> results/${TAP_filename}
53                             #num_of_msg_texts=${#msg_texts[@]}
54                             #echo "num_of_msg_texts:${num_of_msg_texts}"
55                             #echo "num_of_msg_texts:${msg_texts}"
56                             #for ((msgnum=0; ${msgnum} <${num_of_msg_texts}; msgnum++)); do                
57                             #msg_text="message[${num}]:'$(echo ${msg_texts[${msgnum}]}| cut -d '>' -f 2 | cut -d '<' -f 1)'"
58                             #echo "${msg_texts}"
59                             #done                
60                     else                
61                         if [ ${num_of_testcases} -gt 0 ]; then
62                             result="ok ${num_of_testcases}  ${label} - error[${#result_if[@]}]"
63                             echo "${result}">> results/${TAP_filename}
64                             #echo "${result}"
65                             #(( line=${line}+${num_of_testcases}-1 ))
66                         else
67                             result="not ok 1 - ${num_of_testcases}  ${label} - error[XML Covert to TAP ERROR:Xpath query find 0 result]"
68                             echo "${result}">> results/${TAP_filename}
69                             #echo "${result}"
70                             #(( line=${line}+${num_of_testcases} ))
71                         fi                                    
72                 fi
73             else
74                 echo ">tap already converted"
75             fi
76         else
77             echo "sample name  is not standard"
78         fi
79     done
80     else
81         echo "over" >> results/${TAP_filename}
82     fi
83 done
View Code

build.xml内容

 1 <?xml version="1.0" encoding="UTF-8"?>
 2 <project name="ant-jmeter-test" default="run" basedir=".">
 3     <tstamp> <format property="time" pattern="yyyyMMddhhmm" />    </tstamp>
 4     <tstamp> <format property="report.datestamp" pattern="yyyy/MM/dd HH:mm" /></tstamp>
 5     <property environment="env"/>
 6     <property name="report.title" value="接口测试"/>   
 7     <property name="jmeter.home" value="${env.JMETER_HOME}" /><!-- 需要改成自己本地的 Jmeter 目录-->     
 8     <property name="jmeter.result.dir" value="${env.WORKSPACE}/results" /><!-- jmeter测试结果放置目录,这里是放在构建项目下的result目录-->
 9     <property name="jmeter.sum.dir" value="${env.WORKSPACE}/sum" /><!-- jmeter测试概要放置目录(主要用于邮件发送时,作正文内容),这里是放在构建项目下的result目录-->
10     <property name="testplan.dir" value="${env.WORKSPACE}" /><!-- 运行的jmx脚本目录-->
11    
12    
13     <property name="ReportName" value="TestReport" /><!-- 生成的报告的前缀 -->
14     <property name="jmeter.result.jtlName" value="${jmeter.result.dir}/${ReportName}.xml" /><!-- 生成的jtl报告名称 -->
15     <property name="jmeter.result.htmlName" value="${jmeter.result.dir}/${ReportName}.html" /><!-- 生成的html报告名称缀 -->
16     <property name="jmeter.sum.htmlName" value="${jmeter.sum.dir}/sum.html" />     <!-- 生成的sum报告的名称 -->    
17 
18     <target name="run">
19         <echo message="start..."/>
20         <antcall target="clean" />
21         <antcall target="test" />
22         <antcall target="report" />
23     </target>
24     <target name="clean">
25         <delete dir="${env.WORKSPACE}/results/" />
26         <mkdir dir="${env.WORKSPACE}/results" />
27         <delete dir="${env.WORKSPACE}/sum/" />
28         <mkdir dir="${env.WORKSPACE}/sum" />        
29     </target>
30     <target name="test">
31         <taskdef name="jmeter" classname="org.programmerplanet.ant.taskdefs.jmeter.JMeterTask" />
32         <jmeter jmeterhome="${jmeter.home}" resultlog="${jmeter.result.jtlName}">
33             <!-- 声明要运行的脚本。"*.jmx"指包含此目录下的所有jmeter脚本 -->
34             <!-- 配置构建参数TESTLINK_TESTCASE_TESTFILEPATH在Jenkins上    -->
35             <!-- ant -DGUI=false -DTEST_ENVIRONMENT=2 -DTESTLINK_TESTCASE_TESTFILEPATH=test.jmx -->
36             <testplans dir="${testplan.dir}" includes="${env.TESTLINK_TESTCASE_TESTFILEPATH}.jmx" />
37             <property name="jmeter.save.saveservice.output_format" value="xml"/>
38             <property name="jmeter.save.saveservice.assertion_results" value="all"/>
39             <property name="jmeter.save.saveservice.bytes" value="true"/>             
40         </jmeter>
41     </target>
42      <path id="xslt.classpath">
43         <fileset dir="${jmeter.home}/lib" includes="xalan*.jar"/>
44         <fileset dir="${jmeter.home}/lib" includes="serializer*.jar"/>
45      </path>
46 
47       <target name="report">
48        
49         <xslt  classpathref="xslt.classpath"
50                 force="true"
51                 in="${jmeter.result.jtlName}" 
52                 out="${jmeter.result.htmlName}" 
53                 style="${jmeter.home}/extras/jmeter.results.shanhe.me.xsl"> <!-- jtl测试报告转化为html报告 -->
54         <param name="dateReport" expression="${report.datestamp}"/>
55         </xslt>
56         <xslt  classpathref="xslt.classpath"
57                 force="true"
58                 in="${jmeter.result.jtlName}" 
59                 out="${jmeter.sum.htmlName}" 
60                 style="${jmeter.home}/extras/jmeter-results-detail-report_21-lj.xsl"> <!-- sum.html报告,该xsl文件在上个文件的body中保留summary部分的显示,用于jenkins构建后邮件发送的正文展示 -->
61         <param name="dateReport" expression="${report.datestamp}"/>
62         </xslt>
63         <!-- 因为上面生成报告的时候,不会将相关的图片也一起拷贝至目标目录,所以,需要手动拷贝 -->
64         <copy todir="${jmeter.result.dir}">
65             <fileset dir="${jmeter.home}/extras">
66                 <include name="collapse.png"/>
67                 <include name="expand.png"/>
68             </fileset>
69         </copy>
70     </target>
71 </project>
72  
View Code

 

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/oaa608868/article/details/53523845

智能推荐

移动客户端面经_江南:-)的博客-程序员ITS301

前言:对我来说,人生最重要的转折点,也许没有之一。一切都来的措不及防,我在原先实习单位在谈提前转正的时候,闹了些不愉快,我以为我要离职了,我在上周四的晚上投递了简历到腾讯官网,想给自己再一次可以选择的机会。第二天我收到了腾讯的电话,周六一面、周二二面、周四hr面,周五收到offer call和邮件,整个流程我都跟做梦一样,一周的时间,我拿到了腾讯校招的offer,我感觉自己再云中飘,对于我这种...

Docker入门_铃伏的博客-程序员ITS301

Docker入门Docker概述Docker为何要出现docker为什么这么多人喜欢用Docker安装Docker的基本组成Docker的常用命令帮助命令镜像命令docker pull下载镜像docker rmi删除镜像容器命令新建容器并启动查看当前正在运行的容器退出容器删除容器启动和停止容器的操作常用的其它命令后台启动命令查看日志查看镜像的元数据进入当前正在运行的容器从容器拷贝到主机上Docker镜像讲解镜像是什么Docker镜像加载原理理解commit镜像Docker概述Docker为何要出现由于

保护模式中特权级_成仔不说话的博客-程序员ITS301

学习过程中遇到一个对保护模式总结很好的Blog,转来分享一下。先说下特权级的概念,在保护模式下,系统依靠特权级来实施代码和数据的保护,相当于权限啦。特权级共有4个级别,0,1,2,3,数字越小表示权限越高。如图:    较为核心的代码和数据放在较高(靠内)的层级中,处理器用此来防止较低特权的任务在不被允许的情况下访问处于高特权级的段。为了防止概念混淆,我们不用特权级大小来说

endnote如何导入和导出文献_Bright_Sshine的博客-程序员ITS301_endnote导出

在日常写一些报告时,常常用用到参考文献,此时如果手动编辑,耗时太长,这时就会用到一些文献管理的软件,endnote就是其中之一。这里将写一下如何用endnote导入和导出一些文献。 1)endnote文献导入(中国知网或者万方) 点击右下角更多引用格式 选择endnote,并点击导出,下载txt格式文件 2)在endnote中点击导入,然后选择相应文献,相关选项如下,点击导入便将文献导入了endnote中,如果时p...

DAGAN调试_国安永远争第一的博客-程序员ITS301

代码:https://github.com/nebulaV/DAGAN数据:https://docs.google.com/forms/d/e/1FAIpQLSeJ0XywEejM_YA_Od7IbZkPhDrHRjgyHGTaXvNXys3M9OdySQ/view强调:我用的是学校的服务器,没有安装Ubuntu 16.04 。按照教程装好库之后,出现错误1:内核错误解决办法:...

随便推点

网络安全初创公司Illumio获1.25亿美元D轮融资_weixin_34220963的博客-程序员ITS301

在网络无孔不入的生活中,网络信息安全显的越来越重要,尤其是像勒索病毒出现的时候。而那些从事网络信息安全业务的公司也确实得到了资本的青睐。进入2017年以来,全球各地的多家公司获得投资,比如微软1亿美元收购以色列网络安全公司Hexadite,英国网络安全公司Wandera获得 2750万美元投资,网络安全公司Crowdstrike完成1亿美元融资,网络安...

升讯威ADO.NET增强组件(源码):送给喜欢原生ADO.NET的你_weixin_30485799的博客-程序员ITS301

GitHub:https://github.com/iccb1013/sheng.ADO.NET.Plus早前分享过,当时没有把代码上传到Github,只是通过邮件的形式分享给了部分需要的朋友,最近终于有时间简单整理一下直接上传到 Github。目前上传的最新版本有一些新功能特性,还有一些细节调整有兴趣的自己看一下代码。...

自定义gradle插件进阶__ext的博客-程序员ITS301

上次写过自定义gradle插件入门的博客, 介绍了最基本的gradle构建和简单的自定义gradle, 运用生命周期函数、task依赖和扩展配置等。在上篇博客的基础上, 这篇博客继续研究下自定义gradle插件。动态编译技术在开源框架中的应用非常的广泛,现在市面上的插件化框架,热修复框架几乎都使用了动态编译技术,原理几乎都是在编译期间动态的在class文件中注入或者修改代码。AOP技术应...

feature_select_Olaf-li的博客-程序员ITS301_feature select

import pandas as pdimport numpy as npimport matplotlib.pyplot as pltimport seaborn as snsimport datetimefrom tqdm import tqdmfrom sklearn.preprocessing import LabelEncoderfrom sklearn.feature_selection import SelectKBestfrom sklearn.feature_selecti

最大流Dinic算法的一些优化 [网络流][最大流]_dirushi6322的博客-程序员ITS301

明天省夏要讲网络流啦!晚上翻出自己的模板发现是蓝书模板QwQ。。拿出以前的提交代码(AC过的?)曾经的提交记录在luogu上重新提交一遍,结果gg...OVO没有去除多余的inline去除了多余的inline论强数据练考验模板的好处?于是决定自造一份正常的模板。。。主要的优化有三——(1) 当前弧优化,防止因重复访问一条边造成效率降低。...

Flex与.NET互操作(四):使用HttpService、URLReqeust和URLLoader加载/传输数据_技术论坛的博客-程序员ITS301

ActionScript 3.0中提供的数据加载请求类主要是HTTPService,URLLoader和URLRequest,可以通过他们协同来完成数据加载和请求。下面我么便来看看这三个类是怎么来完成数据加载工作。     在本地IIS服务器上有如下定义的XML文件: 1 xml version="1.0" encoding="utf-8" ?> 2 Root> 3   Book>

推荐文章

热门文章

相关标签