【超详细实战攻略】Jmeter逻辑控制器中if控制器、模块控制器、测试片段的使用方法
- [1 搭建测试对象](#1 搭建测试对象)
-
- [1.1 禅道下载](#1.1 禅道下载)
- [1.2 禅道安装](#1.2 禅道安装)
- [1.3 运行禅道](#1.3 运行禅道)
- [1.4 接口查看](#1.4 接口查看)
- [1.5 接口选择](#1.5 接口选择)
- [2 Jmeter前置操作](#2 Jmeter前置操作)
-
- [2.1 创建Jmeter线程组](#2.1 创建Jmeter线程组)
- [2.2 创建信息头管理器和请求默认值](#2.2 创建信息头管理器和请求默认值)
- [2.3 添加获取token接口](#2.3 添加获取token接口)
- [2.4 添加监听器](#2.4 添加监听器)
- [3 获取token并存入系统变量](#3 获取token并存入系统变量)
- [4 测试片段](#4 测试片段)
- [5 模块控制器](#5 模块控制器)
- [6 if控制器](#6 if控制器)
-
- [6.1 需求分析](#6.1 需求分析)
- [6.2 if控制器说明](#6.2 if控制器说明)
- [6.3 获取系统变量](#6.3 获取系统变量)
- [6.4 添加获取用户列表接口](#6.4 添加获取用户列表接口)
- [6.5 提取账号和id](#6.5 提取账号和id)
- [6.6 if控制器-判断用户存在](#6.6 if控制器-判断用户存在)
- [6.7 if控制器-判断用户不存在](#6.7 if控制器-判断用户不存在)
- [7 测试计划效果](#7 测试计划效果)
写作目的:
1、本文主要是通过搭建禅道为测试对象,然后介绍Jmeter的if控制器、模块控制器、测试片段的使用方法;
2、仅作为简单的介绍,不作为项目的实际参考依据,具体的依据项目测试为准。
1 搭建测试对象
1.1 禅道下载
- 直接进入官网,选择开源版进行下载;
- 根据自身电脑环境选择合适的版本,本文是Windows版本:
- 选择"Windows一键安装"下载即可。
1.2 禅道安装
- 下载后如下:
- 双击运行,选择安装目录,我们安装在如下目录:
- 点击Extract,等待安装:
1.3 运行禅道
-
安装完后,在安装目录下有个exe文件双击打开:
-
双击后会打开如下页面:
-
点击"访问禅道",或者输入:
http://127.0.0.1/index.php
:
-
点击【开源版】,进入登陆页面:
1.4 接口查看
- 首次登录时,登录信息如下:
python
# 账号:admin
# 密码:123456
- 首次登录需要修改密码,修改成功后进入系统首页,如下:
- 点击左侧的导航栏中的【后台】,点击【二次开发】:
- 可以看到相关的API接口:
1.5 接口选择
- 我们用以下几个接口来进行说明;
- 获取用户列表;
- 首先是获取登录token,用于后续接口登录权限认证;
- 创建用户接口后获取用户的id;
- 删除用户接口时根据创建的用户id进行删除;
- 公共接口地址为:
python
http://127.0.0.1/zentao/api.php/v1
接口 | 说明 |
---|---|
token获取 | /tokens |
创建用户 | /users |
删除用户 | /users/:id |
2 Jmeter前置操作
2.1 创建Jmeter线程组
-
以此选择【测试计划】-【添加】-【线程(用户)】-【线程组】:
-
命名为"线程组-登录获取token":
2.2 创建信息头管理器和请求默认值
- 创建http信息头管理器,以此为【测试计划】-【添加】-【配置元件】-【HTTP信息头管理器】:
- 创建后如下:
- 点击底部的【添加】按钮,添加头信息:
- 创建http请求默认值,以此为:【测试计划】-【添加】-【配置元件】-【HTTP请求默认值】:
- 创建后如下:
- 设置协议、ip、端口等信息:
2.3 添加获取token接口
- 在线程组下新建http请求,以此为:【线程组-登录获取token】-【添加】-【取样器】-【HTTP请求】:
- 添加后如下,并设置接口信息:
2.4 添加监听器
- 在测试计划下添加监听器,以此为:【测试计划】-【监听器】-【查看结果树】:
- 添加运行后如下:
3 获取token并存入系统变量
- 在以上请求【HTTP请求-登录获取token】下创建【json提取器】;
- 以此为:【HTTP请求-登录获取token】-【添加】-【后置处理器】-【JSON提取器】:
- 提取规则如下,目的是提取登录后返回的token值:
- 添加后置处理器,将提取的变量token加入系统变量,以此为:【HTTP请求-登录获取token】-【添加】-【后置处理器】-【BeanShell 后置处理程序】:
- 写入如下内容,将token加入系统变量中:
python
${__setProperty(token,${token},)}
4 测试片段
-
测试片段的作用类似分组的功能;
-
我们可以把一些业务功能单元进行封装起来;
-
比如这里创建一个名为【用户管理】的测试片段,可以在模块控制器中进行调用;
-
创建测试片段步骤以此为:【线程组-登录获取token】-【添加】-【测试片段】-【测试片段】:
-
创建后如下:
-
注意:创建的测试片段默认是禁止状态,建议使用默认值,只有当测试片段被模块控制器调用时,才会执行,不调用执行;另外如果测试片段被启用,那么它自身下的接口也会执行(即使模块控制器没有调用测试片段)。
5 模块控制器
- 以上创建了测试片段,片段下可以放很多的接口数据;
- 那么如何让测试片段执行,一种是启用测试片段(不建议);另一种就是通过模块控制器进行调用;
- 那模块控制器就是来调用测试片段的,可以指定对应的测试片段进行执行;
- 我们创建一个模块控制器,以此为:【线程组-登录获取token】-【添加】-【逻辑控制器】-【模块控制器】:
- 创建后,要在下方选中执行的测试片段,如下:
6 if控制器
6.1 需求分析
- 前边基本把所有的步骤已经进行完了,接下来就是添加需要执行的接口数据;
- 首先是获取登录后token,传入接口的请求头,这样才有权限;
- 这里获取token其实就是我们之前设置好的系统变量;
- 其次是添加需要执行的接口数据,我们的逻辑是:
① 先判断创建的用户名是不是存在;
② 如果存在,先调用删除用户接口;
③ 然后再调用创建用户接口;
④ 如果用户不存在,直接调用创建用户接口。
6.2 if控制器说明
- i控制器其实就是一个条件判断,和代码中的if语句类似;
- if控制器在这里的作用就是判断用户是否存在。
6.3 获取系统变量
- 在测试片段下添加请求头,主要是获取系统变量token;
- 以此为:【测试片段-用户管理】-【添加】-【配置元件】-【HTTP信息头管理器】:
- 创建后,点击下方的【添加】按钮获取系统变量token:
6.4 添加获取用户列表接口
- 我们在测试片段下先获取用户列表,以此为:【测试片段-用户管理】-【添加】-【取样器】-【HTTP请求】:
- 添加获取用户列表接口数据:
6.5 提取账号和id
- 我们通过获取用户列表接口可以查看系统中有多少个用户;
- 比如运行上边的获取用户列表接口后,可以看出有两个用户:
python
admin、NoamaNelson
- 用户列表接口的返回值如下:
python
{
"page":1,
"total":2,
"limit":20,
"users":[
{
"id":2,
"dept":0,
"account":"NoamaNelson",
"realname":"\u866b\u65e0\u6daf",
"role":"",
"pinyin":"",
"email":""
},
{
"id":1,
"dept":0,
"account":"admin",
"realname":"admin",
"role":"",
"pinyin":"admin a",
"email":""
}
]
}
- 添加后置处理器提取账号和id,提取的账号主要是用于判断这个账号是否存在;提取id是如果账号存在,后续通过id来删除用户;
- 以此为:【HTTP请求-获取用户列表】-【添加】-【后置处理器】-【JSON提取器】:
- 提取规则如下:
python
$.users[?(@.account=="NoamaNelson")].account
- 同样的方法添加提取器,提取id:
python
$.users[?(@.account=="NoamaNelson")].id
6.6 if控制器-判断用户存在
- 添加if控制器,假如用户存在,以此为:【测试片段-用户管理】-【添加】-【逻辑控制器】-【IF控制器】:
- 控制器中添加如下内容:
python
${__jexl3("${account}" == "NoamaNelson")}
-
如果用户存在,我们先通过id删除用户,以此添加删除用户的接口:【IF 控制器-用户存在】-【取样器】-【HTTP请求】:
-
添加删除接口数据:
-
删除用户后再添加用户:
-
添加用户数据为:
python
{
"account":"NoamaNelson",
"password":"123456",
"realname":"虫无涯"
}
6.7 if控制器-判断用户不存在
-
添加if控制器,假如用户不存在,以此为:【测试片段-用户管理】-【添加】-【逻辑控制器】-【IF控制器】;
-
注意这里的用户不存在,判断依据就是提取的用户账号是error,这个error描述是我们在提取用户账号的时候自己写的,比如:
-
那么如果用户用户存在,就直接创建用户就行了:
7 测试计划效果
- 通过以上我们就可以放心的去添加删除用户了;
- 使用if控制器在这里的好处是:如果用户存在我们先删除后添加,避免重复添加或报错的;
- 我们最后的整个测试计划如下:
- 或者我们可以设计如下的效果:
- 完整的脚本如下:
python
<?xml version="1.0" encoding="UTF-8"?>
<jmeterTestPlan version="1.2" properties="5.0" jmeter="5.6.3">
<hashTree>
<TestPlan guiclass="TestPlanGui" testclass="TestPlan" testname="测试计划">
<elementProp name="TestPlan.user_defined_variables" elementType="Arguments" guiclass="ArgumentsPanel" testclass="Arguments" testname="用户定义的变量">
<collectionProp name="Arguments.arguments"/>
</elementProp>
</TestPlan>
<hashTree>
<HeaderManager guiclass="HeaderPanel" testclass="HeaderManager" testname="HTTP信息头管理器">
<collectionProp name="HeaderManager.headers">
<elementProp name="" elementType="Header">
<stringProp name="Header.name">content-type</stringProp>
<stringProp name="Header.value">application/json</stringProp>
</elementProp>
</collectionProp>
</HeaderManager>
<hashTree/>
<ConfigTestElement guiclass="HttpDefaultsGui" testclass="ConfigTestElement" testname="HTTP请求默认值">
<stringProp name="HTTPSampler.domain">127.0.0.1</stringProp>
<stringProp name="HTTPSampler.protocol">http</stringProp>
<elementProp name="HTTPsampler.Arguments" elementType="Arguments" guiclass="HTTPArgumentsPanel" testclass="Arguments" testname="用户定义的变量">
<collectionProp name="Arguments.arguments"/>
</elementProp>
<stringProp name="HTTPSampler.implementation">HttpClient4</stringProp>
</ConfigTestElement>
<hashTree/>
<ResultCollector guiclass="ViewResultsFullVisualizer" testclass="ResultCollector" testname="查看结果树">
<boolProp name="ResultCollector.error_logging">false</boolProp>
<objProp>
<name>saveConfig</name>
<value class="SampleSaveConfiguration">
<time>true</time>
<latency>true</latency>
<timestamp>true</timestamp>
<success>true</success>
<label>true</label>
<code>true</code>
<message>true</message>
<threadName>true</threadName>
<dataType>true</dataType>
<encoding>false</encoding>
<assertions>true</assertions>
<subresults>true</subresults>
<responseData>false</responseData>
<samplerData>false</samplerData>
<xml>false</xml>
<fieldNames>true</fieldNames>
<responseHeaders>false</responseHeaders>
<requestHeaders>false</requestHeaders>
<responseDataOnError>false</responseDataOnError>
<saveAssertionResultsFailureMessage>true</saveAssertionResultsFailureMessage>
<assertionsResultsToSave>0</assertionsResultsToSave>
<bytes>true</bytes>
<sentBytes>true</sentBytes>
<url>true</url>
<threadCounts>true</threadCounts>
<idleTime>true</idleTime>
<connectTime>true</connectTime>
</value>
</objProp>
<stringProp name="filename"></stringProp>
</ResultCollector>
<hashTree/>
<SetupThreadGroup guiclass="SetupThreadGroupGui" testclass="SetupThreadGroup" testname="setUp 线程组-登录获取token">
<intProp name="ThreadGroup.num_threads">1</intProp>
<intProp name="ThreadGroup.ramp_time">1</intProp>
<boolProp name="ThreadGroup.same_user_on_next_iteration">true</boolProp>
<stringProp name="ThreadGroup.on_sample_error">continue</stringProp>
<elementProp name="ThreadGroup.main_controller" elementType="LoopController" guiclass="LoopControlPanel" testclass="LoopController" testname="循环控制器">
<stringProp name="LoopController.loops">1</stringProp>
<boolProp name="LoopController.continue_forever">false</boolProp>
</elementProp>
</SetupThreadGroup>
<hashTree>
<HTTPSamplerProxy guiclass="HttpTestSampleGui" testclass="HTTPSamplerProxy" testname="HTTP请求-登录获取token">
<stringProp name="HTTPSampler.path">/zentao/api.php/v1/tokens</stringProp>
<boolProp name="HTTPSampler.follow_redirects">true</boolProp>
<stringProp name="HTTPSampler.method">POST</stringProp>
<boolProp name="HTTPSampler.use_keepalive">true</boolProp>
<boolProp name="HTTPSampler.postBodyRaw">true</boolProp>
<elementProp name="HTTPsampler.Arguments" elementType="Arguments">
<collectionProp name="Arguments.arguments">
<elementProp name="" elementType="HTTPArgument">
<boolProp name="HTTPArgument.always_encode">false</boolProp>
<stringProp name="Argument.value">{"account": "admin", "password": "Zb918110"}</stringProp>
<stringProp name="Argument.metadata">=</stringProp>
</elementProp>
</collectionProp>
</elementProp>
</HTTPSamplerProxy>
<hashTree>
<JSONPostProcessor guiclass="JSONPostProcessorGui" testclass="JSONPostProcessor" testname="JSON提取器-提取token">
<stringProp name="JSONPostProcessor.referenceNames">token</stringProp>
<stringProp name="JSONPostProcessor.jsonPathExprs">$.token</stringProp>
<stringProp name="JSONPostProcessor.match_numbers">1</stringProp>
<stringProp name="JSONPostProcessor.defaultValues">error</stringProp>
</JSONPostProcessor>
<hashTree/>
<BeanShellPostProcessor guiclass="TestBeanGUI" testclass="BeanShellPostProcessor" testname="BeanShell 后置处理程序">
<stringProp name="filename"></stringProp>
<stringProp name="parameters"></stringProp>
<boolProp name="resetInterpreter">false</boolProp>
<stringProp name="script">${__setProperty(token,${token},)}</stringProp>
</BeanShellPostProcessor>
<hashTree/>
</hashTree>
</hashTree>
<ThreadGroup guiclass="ThreadGroupGui" testclass="ThreadGroup" testname="线程组-用户管理">
<intProp name="ThreadGroup.num_threads">1</intProp>
<intProp name="ThreadGroup.ramp_time">1</intProp>
<boolProp name="ThreadGroup.same_user_on_next_iteration">true</boolProp>
<stringProp name="ThreadGroup.on_sample_error">continue</stringProp>
<elementProp name="ThreadGroup.main_controller" elementType="LoopController" guiclass="LoopControlPanel" testclass="LoopController" testname="循环控制器">
<stringProp name="LoopController.loops">1</stringProp>
<boolProp name="LoopController.continue_forever">false</boolProp>
</elementProp>
</ThreadGroup>
<hashTree>
<HeaderManager guiclass="HeaderPanel" testclass="HeaderManager" testname="HTTP信息头管理器">
<collectionProp name="HeaderManager.headers">
<elementProp name="" elementType="Header">
<stringProp name="Header.name">Token</stringProp>
<stringProp name="Header.value">${__property(token,,)}</stringProp>
</elementProp>
</collectionProp>
</HeaderManager>
<hashTree/>
<ModuleController guiclass="ModuleControllerGui" testclass="ModuleController" testname="模块控制器-用户管理">
<collectionProp name="ModuleController.node_path">
<stringProp name="869052411">测试计划</stringProp>
<stringProp name="869052411">测试计划</stringProp>
<stringProp name="-578308503">线程组-用户管理</stringProp>
<stringProp name="2078815177">测试片段-用户管理</stringProp>
</collectionProp>
</ModuleController>
<hashTree/>
<TestFragmentController guiclass="TestFragmentControllerGui" testclass="TestFragmentController" testname="测试片段-用户管理" enabled="false"/>
<hashTree>
<HTTPSamplerProxy guiclass="HttpTestSampleGui" testclass="HTTPSamplerProxy" testname="HTTP请求-获取用户列表">
<stringProp name="HTTPSampler.path">/zentao/api.php/v1/users</stringProp>
<boolProp name="HTTPSampler.follow_redirects">true</boolProp>
<stringProp name="HTTPSampler.method">GET</stringProp>
<boolProp name="HTTPSampler.use_keepalive">true</boolProp>
<boolProp name="HTTPSampler.postBodyRaw">false</boolProp>
<elementProp name="HTTPsampler.Arguments" elementType="Arguments" guiclass="HTTPArgumentsPanel" testclass="Arguments" testname="用户定义的变量">
<collectionProp name="Arguments.arguments"/>
</elementProp>
</HTTPSamplerProxy>
<hashTree>
<JSONPostProcessor guiclass="JSONPostProcessorGui" testclass="JSONPostProcessor" testname="JSON提取器-提取账号">
<stringProp name="JSONPostProcessor.referenceNames">account</stringProp>
<stringProp name="JSONPostProcessor.jsonPathExprs">$.users[?(@.account=="NoamaNelson")].account</stringProp>
<stringProp name="JSONPostProcessor.match_numbers">1</stringProp>
<stringProp name="JSONPostProcessor.defaultValues">error</stringProp>
</JSONPostProcessor>
<hashTree/>
<JSONPostProcessor guiclass="JSONPostProcessorGui" testclass="JSONPostProcessor" testname="JSON提取器-提取id">
<stringProp name="JSONPostProcessor.referenceNames">id</stringProp>
<stringProp name="JSONPostProcessor.jsonPathExprs">$.users[?(@.account=="NoamaNelson")].id</stringProp>
<stringProp name="JSONPostProcessor.match_numbers">1</stringProp>
<stringProp name="JSONPostProcessor.defaultValues">error</stringProp>
</JSONPostProcessor>
<hashTree/>
</hashTree>
<IfController guiclass="IfControllerPanel" testclass="IfController" testname="IF 控制器-用户存在">
<stringProp name="IfController.condition">${__jexl3("${account}" == "NoamaNelson")}</stringProp>
<boolProp name="IfController.evaluateAll">false</boolProp>
<boolProp name="IfController.useExpression">true</boolProp>
</IfController>
<hashTree>
<HTTPSamplerProxy guiclass="HttpTestSampleGui" testclass="HTTPSamplerProxy" testname="HTTP请求-通过id删除用户">
<stringProp name="HTTPSampler.path">/zentao/api.php/v1/users/:${id}</stringProp>
<boolProp name="HTTPSampler.follow_redirects">true</boolProp>
<stringProp name="HTTPSampler.method">DELETE</stringProp>
<boolProp name="HTTPSampler.use_keepalive">true</boolProp>
<boolProp name="HTTPSampler.postBodyRaw">false</boolProp>
<elementProp name="HTTPsampler.Arguments" elementType="Arguments" guiclass="HTTPArgumentsPanel" testclass="Arguments" testname="用户定义的变量">
<collectionProp name="Arguments.arguments"/>
</elementProp>
</HTTPSamplerProxy>
<hashTree/>
<HTTPSamplerProxy guiclass="HttpTestSampleGui" testclass="HTTPSamplerProxy" testname="HTTP请求-创建用户">
<stringProp name="HTTPSampler.path">/zentao/api.php/v1/users</stringProp>
<boolProp name="HTTPSampler.follow_redirects">true</boolProp>
<stringProp name="HTTPSampler.method">POST</stringProp>
<boolProp name="HTTPSampler.use_keepalive">true</boolProp>
<boolProp name="HTTPSampler.postBodyRaw">true</boolProp>
<elementProp name="HTTPsampler.Arguments" elementType="Arguments">
<collectionProp name="Arguments.arguments">
<elementProp name="" elementType="HTTPArgument">
<boolProp name="HTTPArgument.always_encode">false</boolProp>
<stringProp name="Argument.value">{"account": "NoamaNelson", "password": "123456", "realname": "虫无涯"}</stringProp>
<stringProp name="Argument.metadata">=</stringProp>
</elementProp>
</collectionProp>
</elementProp>
</HTTPSamplerProxy>
<hashTree/>
</hashTree>
<IfController guiclass="IfControllerPanel" testclass="IfController" testname="IF 控制器-用户不存在">
<stringProp name="IfController.condition">${__jexl3("${account}" == "error")}</stringProp>
<boolProp name="IfController.evaluateAll">false</boolProp>
<boolProp name="IfController.useExpression">true</boolProp>
</IfController>
<hashTree>
<HTTPSamplerProxy guiclass="HttpTestSampleGui" testclass="HTTPSamplerProxy" testname="HTTP请求-创建用户">
<stringProp name="HTTPSampler.path">/zentao/api.php/v1/users</stringProp>
<boolProp name="HTTPSampler.follow_redirects">true</boolProp>
<stringProp name="HTTPSampler.method">POST</stringProp>
<boolProp name="HTTPSampler.use_keepalive">true</boolProp>
<boolProp name="HTTPSampler.postBodyRaw">true</boolProp>
<elementProp name="HTTPsampler.Arguments" elementType="Arguments">
<collectionProp name="Arguments.arguments">
<elementProp name="" elementType="HTTPArgument">
<boolProp name="HTTPArgument.always_encode">false</boolProp>
<stringProp name="Argument.value">{"account": "NoamaNelson", "password": "123456", "realname": "虫无涯"}</stringProp>
<stringProp name="Argument.metadata">=</stringProp>
</elementProp>
</collectionProp>
</elementProp>
</HTTPSamplerProxy>
<hashTree/>
</hashTree>
</hashTree>
</hashTree>
</hashTree>
</hashTree>
</jmeterTestPlan>