Jenkins 是一个开源软件项目,是基于Java开发的一种持续集成工具,用于监控持续重复的工作;
Jmeter 可以做接口测试,也可以做压力测试,而且是开源软件;
Ant 是基于Java的构建工具,完成脚本执行并收集结果生成报告,可以跨平台。
注:以下操作均基于Windows环境,且已经安装配置好Jmeter。
ANT_HOME
,值填ant的解压路径D:\ant\apache-ant-1.10.12
;在系统变量Path下添加%ANT_HOME%\bin
;ant -v
,如返回版本信息则证明安装成功;
<project name="ant-jmeter-test" default="run" basedir=".">
<tstamp>
<format property="time" pattern="yyyyMMddhhmm" />
tstamp>
<property name="jmeter.home" value="D:\apache-jmeter-5.4.3" />
<property name="jmeter.result.jtl.dir" value="D:\ant\jtl-report" />
<property name="jmeter.result.html.dir" value="D:\ant\html-report" />
<property name="ReportName" value="天气api接口测试报告" />
<property name="jmeter.result.jtlName" value="${jmeter.result.jtl.dir}/${ReportName}${time}.jtl" />
<property name="jmeter.result.htmlName" value="${jmeter.result.html.dir}/${ReportName}${time}.html" />
<target name="run">
<antcall target="test" />
<antcall target="report" />
target>
<target name="test">
<taskdef name="jmeter" classname="org.programmerplanet.ant.taskdefs.jmeter.JMeterTask" />
<jmeter jmeterhome="${jmeter.home}" resultlog="${jmeter.result.jtlName}">
<testplans dir="D:\ant\test" includes="*.jmx" />
<property name="jmeter.save.saveservice.output_format" value="xml"/>
jmeter>
target>
<path id="xslt.classpath">
<fileset dir="${jmeter.home}/lib" includes="xalan*.jar"/>
<fileset dir="${jmeter.home}/lib" includes="serializer*.jar"/>
path>
<target name="report">
<tstamp><format property="report.datestamp" pattern="yyyy/MM/dd HH:mm"/>tstamp>
<xslt
classpathref="xslt.classpath"
force="true"
in="${jmeter.result.jtlName}"
out="${jmeter.result.htmlName}"
style="${jmeter.home}/extras/jmeter-results-detail-report_21.xsl" >
<param name="dateReport" expression="${report.datestamp}"/>
xslt>
<copy todir="${jmeter.result.html.dir}">
<fileset dir="${jmeter.home}/extras">
<include name="collapse.png" />
<include name="expand.png" />
fileset>
copy>
target>
project>
注意:以下地址修改为自己本地的路径
①jmeter安装的路径:D:\apache-jmeter-5.4.3
②生成jtl测试文件的路径:D:\ant\jtl-report
(jtl-report为自己新建的文件夹)
③生成html报告的路径:D:\ant\html-report
(html-report为自己新建的文件夹)
④jmeter脚本存放路径:D:\ant\test
ant run
;由于jmeter自带的report报告模板样式太单一,不能看到接口响应结果、耗时等信息,所以修改报告模板样式为更加丰富的jmeter-results-shanhe-me.xsl。
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0">
<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"/>
<xsl:strip-space elements="*"/>
<xsl:template match="/testResults">
<html lang="en">
<head>
<meta name="Author" content="shanhe.me"/>
<title>JMeter Test Resultstitle>
<style type="text/css">
* { margin: 0; padding: 0 }
html, body { width: 100%; height: 100%; background: #b4b4b4; font-size: 12px }
table { border: none; border-collapse: collapse; table-layout: fixed }
td { vertical-align: baseline; font-size: 12px }
#left-panel { position: absolute; left: 0; top: 0; bottom: 0; width: 300px; overflow: auto; background: #dee4ea }
#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() }
#left-panel li.success { color: #565b60 }
#left-panel li.failure { color: red }
#left-panel li { list-style: none; color: black; cursor: pointer }
#left-panel li.selected { background-repeat: repeat-x; color: white; background: url() }
#left-panel div { line-height: 20px; background-position: 25px 3px; background-repeat: no-repeat; padding: 0 0 0 45px }
#left-panel div.success { background-image: url() }
#left-panel div.failure { background-image: url() }
#left-panel div.detail { display: none }
#right-panel { position: absolute; right: 0; top: 0; bottom: 0; left: 301px; overflow: auto; background: white }
#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() }
#right-panel .zebra { background-repeat: repeat; padding: 0 0 0 18px; background-image: url() }
#right-panel .data { line-height: 19px; white-space: nowrap }
#right-panel pre.data { white-space: pre }
#right-panel tbody.failure { color: red }
#right-panel td.key { min-width: 108px }
#right-panel td.delimiter { min-width: 18px }
#right-panel td.assertion:before { counter-increment: assertion; content: counter(assertion) ". " }
#right-panel td.assertion { color: black }
#right-panel .trail { border-top: 1px solid #b4b4b4 }
]]>style>
<script type="text/javascript">
var onclick_li = (function() {
var last_selected = null;
return function(li) {
if( last_selected == li )
return;
if( last_selected )
last_selected.className = "";
last_selected = li;
last_selected.className = "selected";
document.getElementById("right-panel").innerHTML = last_selected.firstChild.nextSibling.innerHTML;
return false;
};
})();
var patch_timestamp = function() {
var spans = document.getElementsByTagName("span");
var len = spans.length;
for( var i = 0; i < len; ++i ) {
var span = spans[i];
if( "patch_timestamp" == span.className )
span.innerHTML = new Date( parseInt( span.innerHTML ) );
}
};
var patch_navigation_class = (function() {
var set_class = function(el, flag) {
if(el) {
el.className += flag ? " success" : " failure";
}
};
var traverse = function(el, group_el, flag) {
while(1) {
if(el) {
if(el.className == 'navigation') {
set_class(group_el, flag);
group_el = el;
flag = true;
} else {
var o = el.firstChild;
o = o ? o.className : null;
flag = flag ? (o == 'success') : false;
}
el = el.nextSibling;
} else {
set_class(group_el, flag);
break;
}
}
};
return function() {
var o = document.getElementById("result-list");
o = o ? o.firstChild : null;
if(o)
traverse(o, null, true);
};
})();
window.onload = function() {
patch_timestamp();
patch_navigation_class();
var o = document.getElementById("result-list");
o = o ? o.firstChild : null;
o = o ? o.nextSibling : null;
if(o)
onclick_li(o);
};
]]>script>
head>
<body>
<div id="left-panel">
<ol id="result-list">
<xsl:for-each select="*">
<xsl:if test="position() = 1 or @tn != preceding-sibling::*[1]/@tn">
<li class="navigation">Thread: <xsl:value-of select="@tn"/>li>
xsl:if>
<li onclick="return onclick_li(this);">
<div>
<xsl:attribute name="class">
<xsl:choose>
<xsl:when test="@s = 'true'">successxsl:when>
<xsl:otherwise>failurexsl:otherwise>
xsl:choose>
xsl:attribute>
<xsl:value-of select="@lb"/>
div><div class="detail">
<div class="group">Samplerdiv>
<div class="zebra">
<table>
<tr><td class="data key">Thread Nametd><td class="data delimiter">:td><td class="data"><xsl:value-of select="@tn"/>td>tr>
<tr><td class="data key">Timestamptd><td class="data delimiter">:td><td class="data"><span class="patch_timestamp"><xsl:value-of select="@ts"/>span>td>tr>
<tr><td class="data key">Timetd><td class="data delimiter">:td><td class="data"><xsl:value-of select="@t"/> mstd>tr>
<tr><td class="data key">Latencytd><td class="data delimiter">:td><td class="data"><xsl:value-of select="@lt"/> mstd>tr>
<tr><td class="data key">Bytestd><td class="data delimiter">:td><td class="data"><xsl:value-of select="@by"/>td>tr>
<tr><td class="data key">Sample Counttd><td class="data delimiter">:td><td class="data"><xsl:value-of select="@sc"/>td>tr>
<tr><td class="data key">Error Counttd><td class="data delimiter">:td><td class="data"><xsl:value-of select="@ec"/>td>tr>
<tr><td class="data key">Response Codetd><td class="data delimiter">:td><td class="data"><xsl:value-of select="@rc"/>td>tr>
<tr><td class="data key">Response Messagetd><td class="data delimiter">:td><td class="data"><xsl:value-of select="@rm"/>td>tr>
table>
div>
<div class="trail">div>
<xsl:if test="count(assertionResult) > 0">
<div class="group">Assertiondiv>
<div class="zebra">
<table>
<xsl:for-each select="assertionResult">
<tbody>
<xsl:attribute name="class">
<xsl:choose>
<xsl:when test="failure = 'true'">failurexsl:when>
<xsl:when test="error = 'true'">failurexsl:when>
xsl:choose>
xsl:attribute>
<tr><td class="data assertion" colspan="3"><xsl:value-of select="name"/>td>tr>
<tr><td class="data key">Failuretd><td class="data delimiter">:td><td class="data"><xsl:value-of select="failure"/>td>tr>
<tr><td class="data key">Errortd><td class="data delimiter">:td><td class="data"><xsl:value-of select="error"/>td>tr>
<tr><td class="data key">Failure Messagetd><td class="data delimiter">:td><td class="data"><xsl:value-of select="failureMessage"/>td>tr>
tbody>
xsl:for-each>
table>
div>
<div class="trail">div>
xsl:if>
<div class="group">Requestdiv>
<div class="zebra">
<table>
<tr><td class="data key">Method/Urltd><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>
<tr><td class="data key">Query Stringtd><td class="data delimiter">:td><td class="data"><pre class="data"><xsl:value-of select="queryString"/>pre>td>tr>
<tr><td class="data key">Cookiestd><td class="data delimiter">:td><td class="data"><pre class="data"><xsl:value-of select="cookies"/>pre>td>tr>
<tr><td class="data key">Request Headerstd><td class="data delimiter">:td><td class="data"><pre class="data"><xsl:value-of select="requestHeader"/>pre>td>tr>
table>
div>
<div class="trail">div>
<div class="group">Responsediv>
<div class="zebra">
<table>
<tr><td class="data key">Response Headerstd><td class="data delimiter">:td><td class="data"><pre class="data"><xsl:value-of select="responseHeader"/>pre>td>tr>
<tr><td class="data key">Response Datatd><td class="data delimiter">:td><td class="data"><pre class="data"><xsl:value-of select="responseData"/>pre>td>tr>
<tr><td class="data key">Response Filetd><td class="data delimiter">:td><td class="data"><pre class="data"><xsl:value-of select="responseFile"/>pre>td>tr>
table>
div>
<div class="trail">div>
div>
li>
xsl:for-each>
ol>
div>
<div id="right-panel">div>
body>
html>
xsl:template>
xsl:stylesheet>
ant run
,并用浏览器打开输出的HTML报告,结果如下前提:已在jenkins上部署好项目。
Ant Plugin
;Jenkins支持通过SMTP协议发送邮件,所以要想发送邮件,首先得有一个支持SMTP协议的邮箱账号,市面上主流的邮箱都是支持该协议的,比如QQ邮箱、网易163邮箱等。
前提:我这里使用的是网易163邮箱。
POP3/SMTP/IMAP
,点击开启以后会提示我们发送信息,然后返回一个授权码;smtp.163.com
,QQ邮箱为smtp.qq.com
465
,不加密的端口为25
类型
:选择 Username with passwd
用户名
:输入邮箱地址密码
:输入邮箱的授权码,而不是邮箱登录密码Default user e-mail suffix
:默认邮箱后缀Default Content Type
:邮件发送格式Default Recipients
:邮件接收地址E-mail Notification
(邮件通知),按照上面的smtp服务设置依次进行填写,勾选“通过发送测试邮件测试配置”,最后点击Test;Editable Email Notification
,进行如下配置;Default Subject 模板
:【自动化构建通知】$PROJECT_NAME - Build # $BUILD_NUMBER - $BUILD_STATUS!
Default Content 模板如下
:
DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>${ENV, var="JOB_NAME"}-第${BUILD_NUMBER}次构建日志title>
head>
<body leftmargin="8" marginwidth="0" topmargin="8" marginheight="4"
offset="0">
<table width="95%" cellpadding="0" cellspacing="0" style="font-size: 11pt; font-family: Tahoma, Arial, Helvetica, sans-serif">
<tr>
(本邮件由系统自动发出,无需回复!)<br/>br>
大家好,以下为 <strong>${PROJECT_NAME }strong> 项目构建信息:br>br>
<strong>构建结果:<font color="#CC0000">${BUILD_STATUS}strong>font>
tr>
<tr>
<td><br />
<b><font color="#0B610B">构建信息font>b>
<hr size="2" width="100%" align="center" />td>
tr>
<tr>
<td>
<ul>
<li>项目名称 : ${PROJECT_NAME}li>
<li>构建编号 : 第${BUILD_NUMBER}次构建li>
<li>触发原因: ${CAUSE}li>
<li>构建状态: ${BUILD_STATUS}li>
<li>构建日志: <a href="${BUILD_URL}console">${BUILD_URL}consolea>li>
<li>构建URL: <a href="${BUILD_URL}">${BUILD_URL}a>li>
ul>
<h4><font color="#0B610B">测试报告font>h4>
<hr size="2" width="100%" />
<a href="${PROJECT_URL}HTML_20Report">${PROJECT_URL}HTML_20Reporta>
<h4><font color="#0B610B">失败用例font>h4>
<hr size="2" width="100%" />
$FAILED_TESTS<br/>
<h4><font color="#0B610B">最近提交(#${GIT_REVISION})font>h4>
<hr size="2" width="100%" />
<ul>
${CHANGES_SINCE_LAST_SUCCESS, reverse=true, format="%c", changesFormat="<li>%d [%a] %mli>"}
ul>
td>
tr>
table>
body>
html>