jmeter接口自动化测试

安装

jdk安装
  • 下载安装:https://www.oracle.com/cn/java/technologies/downloads/#jdk17-windows

  • 配置环境变量:

    1. 新建系统变量JAVA_HOME,输入jdk安装路径 D:\Program Files\Java\jdk-17
    2. 新建系统变量CLASS_PATH,输入 .;%JAVA_HOME%\lib\dt.jar;%JAVA_HOME%\lib\tools.jar;
    3. 编辑系统变量PATH, 点击新建 %JAVA_HOME%\bin
  • 验证,cmd输入命令java,javac,java -version

jmeter安装
  • 下载安装:https://jmeter.apache.org/download_jmeter.cgi

  • 配置环境变量:

    1. 新建系统变量,JMETER_HOME,输入jmeter安装路径 D:\apache-jmeter-5.6.2
    2. 编辑系统变量CLSS_PATH,在后面追加 %JMETER_HOME%\lib\ext\ApacheJMeter_core.jar;%JMETER_HOME%\lib\jorphan.jar;
    3. 编辑系统变量PATH, 点击新建 %JMETER_HOME%\bin
  • 验证,cmd输入命令jmeter, jmeter -v

  • 修改默认语言为中文:在安装目录的bin目录中找到 jmeter.properties,修改第39行为 language=zh_CN

接口关联

{
  "code": 200,
  "data": {
    "access_token": "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJmcmVzaCI6ZmFsc2UsImlhdCI6MTcwMTY3ODE3NywianRpIjoiNjk5ZWQwN2ItYWFhNS00NzJkLWJjZDUtOTljNjFiOThlZTM1IiwidHlwZSI6ImFjY2VzcyIsInN1YiI6InRlc3RAcXEuY29tIiwibmJmIjoxNzAxNjc4MTc3LCJleHAiOjE3MDE2NzkwNzd9.AkVdti-TFOTa-3ALan9-cr-NmPwJtvfP8Mc_6ykr9mw",
    "message": "Login successful"
  },
  "message": "success"
}
正则表达式提取器

在这里插入图片描述

使用第一个正则匹配到的第一个值

使用: ${access_token}

匹配多个

正则表达式:“access_token”: “(.*?)”,\n.*?“message”: “(.*?)”

模板:$1$ ,匹配access_token,$2$匹配message

逗号拼接:$1$,$2$

JSON提取器

json path表达式:$.data.access_token

添加调试取样器查看提取的变量

Cookie关联

用正则从信息头提取session:Set-Cookie: session=(.*?);

使用HTTP信息头管理器:名称: Cookie,值: session=${session}

或使用HTTP Cookie管理器:名称: session,值: ${session}, 域:127.0.0.1

跨线程组关联

使用beanshell后置处理器将提取变量设置为全局变量

设置

${__setProperty(new_token,${access_token},)}

使用

${__property(new_token,,)}

断言

响应断言

断言包含文本

JSON断言

断言json path 存在,同时断言值

BeanShell断言
断言状态码
if(ResponseCode.equals("200")){
	Failure=false;
	FailureMessage="断言状态码成功";
}
else{
	Failure=true;
	FailureMessage="断言状态码200失败,状态码:"+ResponseCode;
}
断言响应内容
var resp = prev.getResponseDataAsString();
if(resp.contains("access_token")){
	Failure=false;
	FailureMessage="断言响应内容成功";
}
else{
	Failure=true;
	FailureMessage="断言响应内容失败,响应不包含access_token";
}
断言数据库
  1. 下载mysql数据库连接jar包:https://dev.mysql.com/downloads/connector/j/
  2. 将 mysql-connector-j-8.2.0.jar,放到jmeter的lib目录,重启jmeter。(数据库版本不同可能不兼容)
  3. 添加配置元件 :JDBC Connection Configuration
  4. 填值:
    • 数据库连接池名字:Variable Name for create pool: conn_db
    • 最大连接数:Max Number of Connections: 10
    • 数据库地址:Database URL: jdbc:mysql://127.0.0.1:3306/
    • 数据库驱动:JDBC Driver class: com.msql.jdbc.Driver
    • 用户名
    • 密码
  5. 添加取样器:JDBC Request
  6. 填值:
    • 数据库连接池名字:Variable Name of Pool declared in JDBC Connection Configuration: conn_db
    • 选择Query Type
    • 填写sql语句:eg: select * from user where email=“test@qq.com”;
    • 查询结果变量名: Result variable name: result
  7. 获取值: 获取第一行,第password列的值,与 预期值断言
var pwd = vars.getObject("result").get(0).get("password");
log.info(pwd+"");
...

数据驱动

  1. 创建文件 data.csv

    email,password,assert_str
    "",123456,401
    test@qq.com,"",401
    test@qq.com,123456,access_token
    
  2. 添加逻辑控制器-循环控制器,循环次数3

  3. 添加 配置元件:csv 数据文件设置,填文件名:E:\workspace\h_demo\data.csv,编码:UTF-8

  4. 填写请求和断言参数:${email},${password},${assert_str}

BeanShell脚本

日志打印
log.info("测试info日志");
log.info(123+"");
log.error("测试error日志");
System.out.println("java打印");
vars操作jmeter局部变量(单个线程组)
vars.put("name","test123");
log.info(vars.get("name"));
props操作全局变量
props.put("name","test123"); 
log.info(props.get("name")); 
prev获取接口响应数据
log.info(prev.getResponseDataAsString());
log.info(prev.getResponseCode());
调用java文件
// 调用java文件
source("C:\\Users\\Administrator\\IdeaProjects\\test\\src\\demo.java");
log.info(new demo().addNum(6,8) + "");
调用jar包

idea导出jar包

  • 打开模块设置-工件-添加JAR-来自具有依赖项的模块-选择主类-复制到输出目录-确定-确定

  • 构建-构建工件-选择JAR-构建

在测试计划添加jar包

// 调用jar包
import test.demo;
log.info(new demo().addNum(6,8) + "");
调用python代码

python代码使用print返回值,beanshell通过dos命令调用python脚本

无参数调用

python代码

print('123456')

java代码

import java.io.BufferedReader;
import java.io.InputStreamReader;
 
String command = "cmd /c python C:/Users/Administrator/Desktop/1120_xxxx/demo.py";
Runtime rt = Runtime.getRuntime();
Process pr = rt.exec(command);
pr.waitFor();
BufferedReader br = new BufferedReader(new InputStreamReader(pr.getInputStream()));
String line = "";
StringBuilder response = new StringBuilder();
while ((line = br.readLine()) != null) {
    response.append(line);
} 
response_data = response.toString();
br.close();
log.info("调用py文件返回值:"+response_data);
单个参数

python代码

import argparse

parser = argparse.ArgumentParser()
parser.add_argument("-t", "--ticket")
args = parser.parse_args()
ticket = args.ticket
print(ticket+"单个参数")

java代码

import java.io.BufferedReader;
import java.io.InputStreamReader;
 
// -t传参
String command = "cmd /c python C:/Users/Administrator/Desktop/1120_xxxx/demo1.py -t aaa";
Runtime rt = Runtime.getRuntime();
Process pr = rt.exec(command);
pr.waitFor();
BufferedReader br = new BufferedReader(new InputStreamReader(pr.getInputStream()));
String line = "";
StringBuilder response = new StringBuilder();
while ((line = br.readLine()) != null) {
    response.append(line);
} 
response_data = response.toString();
br.close();
log.info("调用py文件返回值:"+response_data);
多个参数

python代码

import argparse

parser = argparse.ArgumentParser()
parser.add_argument("-t", "--ticket", action="append")
args = parser.parse_args()
# 多个值拼接列表
ticket = args.ticket
print(ticket[0] + ticket[1] + "多个参数")

java代码

String command = "cmd /c python C:/Users/Administrator/Desktop/1120_xxxx/demo2.py -t aaa -t bbb";
Runtime rt = Runtime.getRuntime();
Process pr = rt.exec(command);
pr.waitFor();
BufferedReader br = new BufferedReader(new InputStreamReader(pr.getInputStream()));
String line = "";
StringBuilder response = new StringBuilder();
while ((line = br.readLine()) != null) {
    response.append(line);
} 
response_data = response.toString();
br.close();
log.info("调用py文件返回值:"+response_data);
加密参数
Base64加密
import java.util.Base64;
//初始化一个Base64对象,调用encodeToString()方法
String username_encrypt = Base64.getEncoder().encodeToString("admin".getBytes("UTF-8"));
String password_encrypt = Base64.getEncoder().encodeToString("123456".getBytes("UTF-8"));
log.info(username_encrypt);
log.info(password_encrypt);
vars.put("username_encrypt",username_encrypt);
vars.put("password_encrypt",password_encrypt);
MD5加密

函数助手:${__digest(MD5,admin,)}

//导包
import org.apache.commons.codec.digest.DigestUtils;
//md5加密
String username_encrypt = DigestUtils.md5Hex("admin");
String password_encrypt  = DigestUtils.md5Hex("123");
log.info(username_encrypt);
log.info(password_encrypt);
vars.put("username_encrypt",username_encrypt);
vars.put("password_encrypt",password_encrypt);
SHA加密

函数助手:${__digest(sha-1,admin,)}

import org.apache.commons.codec.digest.DigestUtils;
String username_encrypt = DigestUtils.sha512Hex("admin");
log.info("加密结果为:"+username_encrypt);
RSA加密

java代码

import java.io.BufferedReader;
import java.io.InputStreamReader;
 
String command = "cmd /c python C:/Users/Administrator/Desktop/1120_xxxx/demo2.py -t aaa -t bbb";

Runtime rt = Runtime.getRuntime();
Process pr = rt.exec(command);
 
pr.waitFor();
 
BufferedReader br = new BufferedReader(new InputStreamReader(pr.getInputStream()));
String line = "";
StringBuilder response = new StringBuilder();
while ((line = br.readLine()) != null) {
    response.append(line);
 
} 
response_data = response.toString();
br.close();
 
log.info("调用py文件返回值:"+response_data);
String[] res = response_data.split("##");
log.info(res[0]);
log.info(res[1]);

python代码

import argparse
import base64
import rsa


def rsa_encrypt():
    parser = argparse.ArgumentParser()
    parser.add_argument("-t", "--ticket", action="append")
    args = parser.parse_args()
    ticket = args.ticket
    # 公钥
    public_key_str = """-----BEGIN RSA PUBLIC KEY-----
    MIGJAoGBALO7UPE26anTGHND2Q54zYYPusDx+tbO1Yia7zoxpZediw+Baea7aFZCJ+ZvWd5ZBTopuWvb8hN
    kY24eBHcXN0pU32WjsH9REp1kXhxbndnw+u3diaoUFqVc66xl+LXEo1Y9oDWfkGCir2JnN0aieUiPlHDLhmc+
    LII/ZDspITKDAgMBAAE=-----END RSA PUBLIC KEY-----"""
    pubkey = rsa.PublicKey.load_pkcs1(public_key_str.encode())
    # 加密用户名
    username_str = rsa.encrypt(str(ticket[0]).encode("utf-8"), pubkey)
    # 把二进制转化成字符串格式
    username_encrypt = base64.b64encode(username_str).decode("utf-8")
    print(username_encrypt)
    print("##")
    # 加密密码
    password_str = rsa.encrypt(str(ticket[1]).encode("utf-8"), pubkey)
    # 把二进制转化成字符串格式
    password_encrypt = base64.b64encode(password_str).decode("utf-8")
    print(password_encrypt)

if __name__ == '__main__':
    rsa_encrypt()

Jmeter+Ant+Jenkins持续集成

命令行运行

-n(命令行执行)-t (指定jmx脚本)

jmeter -n -t 测试计划.jmx

-l (生成jtl报告)

jmeter -n -t 测试计划.jmx -l result.jtl

-e (生成html报告) -o (输出报告位置,位置必须是空文件夹)

jmeter -n -t 测试计划.jmx -l result.jtl -e -o result
# jmeter.save.saveservice.output_format=csv
集成Ant
  1. 下载安装:https://ant.apache.org/bindownload.cgi ,下载zip包,解压后将bin目录添加到系统环境变量Path

  2. 将jmeter.results.shanhe.me.xsl拷贝到jmeter的extras目录下

    <?xml version="1.0" encoding="UTF-8"?>
    <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 Results</title>
                <style type="text/css"><![CDATA[
                
                    * { 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(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAkAAAAICAYAAAArzdW1AAAAAXNSR0IArs4c6QAAAAZiS0dEAP8A/wD/oL2nkwAAAAlwSFlzAAALEwAACxMBAJqcGAAAAAd0SU1FB9sDEBQqGbO7BEcAAAAdaVRYdENvbW1lbnQAAAAAAENyZWF0ZWQgd2l0aCBHSU1QZC5lBwAAAKRJREFUGNN1zM0KgkAYheF3RvtXSsGyWhRNaILS7bdt11W0KgJvoPwZp0UlBPUtz3nOJw7Hk7necv5dOA2Qaazo2vZP0LEt9olCVtqQROufKNmuqBuBNAYW4QzXGX6B0bDPcjGnMQYJ8Cg12U59oSzaUJQa4IUAXMclDHwAAn/MxPMw765FZd2QRgopBWmsKCrdfhXnS/4ZYElBXdyxewN008Y8AephLAkqz613AAAAAElFTkSuQmCC) }
                    #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(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAEAAAAUCAYAAABMDlehAAAAAXNSR0IArs4c6QAAAAZiS0dEAP8A/wD/oL2nkwAAAAlwSFlzAAALEwAACxMBAJqcGAAAAAd0SU1FB9sDEBQxLTs5O2gAAAAdaVRYdENvbW1lbnQAAAAAAENyZWF0ZWQgd2l0aCBHSU1QZC5lBwAAAEdJREFUCNc1y7ERgEAMA0GNUhIyGqM2uqKgtyWZhE9v53A/7/A6D7BkMDNgy2AroB2wHTCZv5UMOgFLG1bvd7XBckBlwCXjA5wMOF5iOX/MAAAAAElFTkSuQmCC) }
                    #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(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=) }
                    #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) }
                    #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(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAEAAAAQCAYAAADXnxW3AAAAAXNSR0IArs4c6QAAAAZiS0dEAP8A/wD/oL2nkwAAAAlwSFlzAAALEwAACxMBAJqcGAAAAAd0SU1FB9sDEBUkDq8pxjkAAAAdaVRYdENvbW1lbnQAAAAAAENyZWF0ZWQgd2l0aCBHSU1QZC5lBwAAADdJREFUCNdVxrERwDAMAzGK0v47eS6Z927SpMFBAAbkvSvnRk5+7K5cVfLMyN39bWakJAjA5xw9R94jN3tVhVEAAAAASUVORK5CYII=) }
                    #right-panel .zebra { background-repeat: repeat; padding: 0 0 0 18px; background-image: url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAEAAAAmCAYAAAAFvPEHAAAAAXNSR0IArs4c6QAAAAZiS0dEAP8A/wD/oL2nkwAAAAlwSFlzAAALEwAACxMBAJqcGAAAAAd0SU1FB9sDEBYWFlNztEcAAAAdaVRYdENvbW1lbnQAAAAAAENyZWF0ZWQgd2l0aCBHSU1QZC5lBwAAABdJREFUCNdjYKAtePv5338mBgYGBpoQAGy1BAJlb/y6AAAAAElFTkSuQmCC) }
                    #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"><![CDATA[
                
                    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="*">
                            <!-- group with the previous sibling -->
                            <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'">success</xsl:when>
                                            <xsl:otherwise>failure</xsl:otherwise>
                                        </xsl:choose>
                                    </xsl:attribute>
                                    <xsl:value-of select="@lb"/>
                                </div><div class="detail">
                                    <div class="group">Sampler</div>
                                    <div class="zebra">
                                        <table>
                                            <tr><td class="data key">Thread Name</td><td class="data delimiter">:</td><td class="data"><xsl:value-of select="@tn"/></td></tr>
                                            <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>
                                            <tr><td class="data key">Time</td><td class="data delimiter">:</td><td class="data"><xsl:value-of select="@t"/> ms</td></tr>
                                            <tr><td class="data key">Latency</td><td class="data delimiter">:</td><td class="data"><xsl:value-of select="@lt"/> ms</td></tr>
                                            <tr><td class="data key">Bytes</td><td class="data delimiter">:</td><td class="data"><xsl:value-of select="@by"/></td></tr>
                                            <tr><td class="data key">Sample Count</td><td class="data delimiter">:</td><td class="data"><xsl:value-of select="@sc"/></td></tr>
                                            <tr><td class="data key">Error Count</td><td class="data delimiter">:</td><td class="data"><xsl:value-of select="@ec"/></td></tr>
                                            <tr><td class="data key">Response Code</td><td class="data delimiter">:</td><td class="data"><xsl:value-of select="@rc"/></td></tr>
                                            <tr><td class="data key">Response Message</td><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) &gt; 0">
                                        <div class="group">Assertion</div>
                                        <div class="zebra">
                                            <table>
                                                <xsl:for-each select="assertionResult">
                                                    <tbody>
                                                        <xsl:attribute name="class">
                                                            <xsl:choose>
                                                                <xsl:when test="failure = 'true'">failure</xsl:when>
                                                                <xsl:when test="error = 'true'">failure</xsl:when>
                                                            </xsl:choose>
                                                        </xsl:attribute>
                                                        <tr><td class="data assertion" colspan="3"><xsl:value-of select="name"/></td></tr>
                                                        <tr><td class="data key">Failure</td><td class="data delimiter">:</td><td class="data"><xsl:value-of select="failure"/></td></tr>
                                                        <tr><td class="data key">Error</td><td class="data delimiter">:</td><td class="data"><xsl:value-of select="error"/></td></tr>
                                                        <tr><td class="data key">Failure Message</td><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">Request</div>
                                    <div class="zebra">
                                        <table>
                                            <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>
                                            <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>
                                            <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>
                                            <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>
                                        </table>
                                    </div>
                                    <div class="trail"></div>
                                    <div class="group">Response</div>
                                    <div class="zebra">
                                        <table>
                                            <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>
                                            <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>
                                            <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>
                                        </table>
                                    </div>
                                    <div class="trail"></div>
                                </div>
                            </li>
                        </xsl:for-each>
                    </ol>
                </div>
                <div id="right-panel"></div>
            </body>
            </html>
        </xsl:template>
    </xsl:stylesheet>
    
  3. 将jmeter的extras目录下ant-jmeter-1.1.1.jar拷贝到ant的lib目录下

  4. 修改jmeter.propties全局配置文件

    jmeter.save.saveservice.output_format=xml
    jmeter.save.saveservice.response_data=true
    jmeter.save.saveservice.samplerData=true
    jmeter.save.saveservice.responseHeaders=true
    jmeter.save.saveservice.requestHeaders=true
    
  5. 修改build.xml文件

    <?xml version="1.0" encoding="UTF-8"?>
    <project name="ant-jmeter-test" default="run" basedir=".">
    <property environment="env"/>
    <tstamp>
            <format property="time" pattern="yyyy_MM_dd_hh_mm" />
        </tstamp>
        <!-- 需要调用的jmeter目录,根据需要进行修改-->  
        <property name="jmeter.home" value="D:\apache-jmeter-5.6.2" />
        <property name="report.title" value="接口自动化测试"/>
        <!-- jmeter生成jtl格式的结果报告的路径--> 
        <property name="jmeter.result.jtl.dir" value="E:\ants\jtl" />
        <!-- jmeter生成html格式的结果报告的路径-->
        <property name="jmeter.result.html.dir" value="E:\ants\html" />
    	<!-- 【详细报告】jmeter生成html格式的详细报告的路径-->
    	<property name="jmeter.result.html.dir1" value="report" />
        <!-- 生成的报告的前缀-->  
        <property name="ReportName" value="接口自动化汇总报告" />
    	<property name="ReportName1" value="接口自动化详细报告" />
        <property name="jmeter.result.jtlName" value="${jmeter.result.jtl.dir}/${ReportName}.jtl" />
        <property name="jmeter.result.htmlName" value="${jmeter.result.html.dir}/${ReportName}.html" />
    	<!-- 【详细报告】详细报告的文件名-->
        <property name="jmeter.result.htmlName1" value="${jmeter.result.html.dir1}/${ReportName1}.html" />
        <target name="run">
    	   <!--antcall target="delete" /-->
            <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}">
                <!-- 声明要运行的脚本路径"*.jmx"指包含此目录下的所有jmeter脚本-->
                <testplans dir="E:\ants" 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-report_21.xsl">
                  <param name="dateReport" expression="${report.datestamp}"/>
           </xslt>
    	   <xslt 
                  classpathref="xslt.classpath"  
                  force="true"
                  in="${jmeter.result.jtlName}"
                  out="${jmeter.result.htmlName1}"
                  style="${jmeter.home}/extras/jmeter.results.shanhe.me.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>
    		<copy todir="${jmeter.result.html.dir1}">
                <fileset dir="${jmeter.home}/extras">
                    <include name="collapse.png" />
                    <include name="expand.png" />
                </fileset>
            </copy>
        </target>
    </project>
    
  6. cmd在当前目录执行ant,查看结果

通过jenkins执行Ant命令
  • 安装插件 HTML Publisher plugin

  • 安装插件Groovy,在Script Console执行

    System.setProperty("hudson.model.DirectoryBrowserSupport.CSP","")
    
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值