如何使用 JMeter 调用你的 Restful Web Service?进行简单的压力测试和自动化测试

本文涉及的产品
性能测试 PTS,5000VUM额度
简介: 如何使用 JMeter 调用你的 Restful Web Service?进行简单的压力测试和自动化测试 表述性状态传输(REST)作为对基于 SOAP 和 Web 服务描述语言(WSDL)的 Web 服务的简单替代,在 Web 开发上得到了广泛的接受。

如何使用 JMeter 调用你的 Restful Web Service?进行简单的压力测试和自动化测试

表述性状态传输(REST)作为对基于 SOAP 和 Web 服务描述语言(WSDL)的 Web 服务的简单替代,在 Web 开发上得到了广泛的接受。
能够充分证明这点的是主流 Web 2.0 服务提供商在接口设计中对 REST 的普遍采用 - 包括雅虎、谷歌以及脸谱 - 出于简单易用、以面向资源的模型发布自己的服务的偏好他们都已经抛弃了 SOAP 和基于 WSDL 的接口。在你必须要对你的 RESTFul web service 进行测试的时候,你可能会有这两个选择:

  • 使用 URL 对你的 WebService 进行简单访问测试
  • 使用 JMeter 来对 WebService 进行循环访问测试。这种测试也能起到简单性能测试的效果。

本文示例中将会演示如何创建简单 Hello World WebService,并使用 JMeter 访问测试该 WebService。
如果你碰到了以下问题之一,那么恭喜你,你来对地方了:

  • 使用 JMeter 对 SOAP/REST WebService 进行功能测试
  • 使用 JMeter 对一个 RESTFul API 执行性能测试
  • 使用 JMeter 对 Rest API 进行自动化测试 - 也就是性能测试
  • 如何使用 JMeter 测试 REST API
  • 使用 JMeter 来测试一个 RESTful web service(Jersey)

步骤概述:

1. 先决条件:RESTFul web service 的完整实现。
2. 创建简单 Java 类:CrunchifyJMeterTest.java(我们将使用 JMeter 访问的服务)。
3. 在 Tomcat 上对该应用进行重新部署。
4. 运行 JMeter 并打开下文提供的 .jmx 文件。
5. 执行并对你的测试结果进行分析。

步骤 1

先决条件:RESTFul web service 的完整实现。部署并运行应用程序。

步骤 2

创建 CrunchifyJMeterTest.java 文件

[java] view plain copy

print?

  1. package com.crunchify.restjersey;
  2. import java.io.FileNotFoundException;
  3. import java.io.IOException;
  4. import javax.ws.rs.GET;
  5. import javax.ws.rs.Path;
  6. import javax.ws.rs.Produces;
  7. /**
  8.  * @author Crunchify
  9.  * 
  10.  */
  11. @Path("/index")
  12. public class CrunchifyJMeterTest {
  13.     @GET
  14.     @Produces("text/html")
  15.     public String checkECV() throws InterruptedException, FileNotFoundException, IOException {
  16.         String result = "<br><div align='center'><h2>Hey This is Crunchify's JMeter Test...</h2></div>";
  17.         System.out.println(result);
  18.         Thread.sleep(1000);
  19.         return result;
  20.     }
  21. }

之后应该看到下面这种 Eclipse 目录结构:
Crunchify-REST-Jersey-Example.png

步骤 3

将 CrunchifyRESTJerseyExample 在 Tomcat 上重新部署,使用这个 URL 测试你的 REST 服务:
URL:http://localhost:8080/CrunchifyRESTJerseyExample/crunchify/index/
Crunchify-REST-Jersey-Example-URL-test.png

步骤 4

将以下脚本拷贝到文本文件并另存为 Crunchify-JMeter-Test.jmx:

[html] view plain copy

print?

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <jmeterTestPlan version="1.2" properties="2.4" jmeter="2.9 r1437961">
  3.   <hashTree>
  4.     <TestPlan guiclass="TestPlanGui" testclass="TestPlan" testname="App Shah Desktop Test" enabled="true">
  5.       <stringProp name="TestPlan.comments"></stringProp>
  6.       <boolProp name="TestPlan.functional_mode">false</boolProp>
  7.       <boolProp name="TestPlan.serialize_threadgroups">false</boolProp>
  8.       <elementProp name="TestPlan.user_defined_variables" elementType="Arguments" guiclass="ArgumentsPanel" testclass="Arguments" testname="User Defined Variables" enabled="true">
  9.         <collectionProp name="Arguments.arguments"/>
  10.       </elementProp>
  11.       <stringProp name="TestPlan.user_define_classpath"></stringProp>
  12.     </TestPlan>
  13.     <hashTree>
  14.       <ThreadGroup guiclass="ThreadGroupGui" testclass="ThreadGroup" testname="Crunchify's REST Service JMeter Test" enabled="true">
  15.         <stringProp name="ThreadGroup.on_sample_error">continue</stringProp>
  16.         <elementProp name="ThreadGroup.main_controller" elementType="LoopController" guiclass="LoopControlPanel" testclass="LoopController" testname="Loop Controller" enabled="true">
  17.           <boolProp name="LoopController.continue_forever">false</boolProp>
  18.           <intProp name="LoopController.loops">-1</intProp>
  19.         </elementProp>
  20.         <stringProp name="ThreadGroup.num_threads">5</stringProp>
  21.         <stringProp name="ThreadGroup.ramp_time">2</stringProp>
  22.         <longProp name="ThreadGroup.start_time">1367432020000</longProp>
  23.         <longProp name="ThreadGroup.end_time">1367432020000</longProp>
  24.         <boolProp name="ThreadGroup.scheduler">false</boolProp>
  25.         <stringProp name="ThreadGroup.duration"></stringProp>
  26.         <stringProp name="ThreadGroup.delay"></stringProp>
  27.       </ThreadGroup>
  28.       <hashTree>
  29.         <HTTPSamplerProxy guiclass="HttpTestSampleGui" testclass="HTTPSamplerProxy" testname="HTTP Request" enabled="true">
  30.           <boolProp name="HTTPSampler.postBodyRaw">true</boolProp>
  31.           <elementProp name="HTTPsampler.Arguments" elementType="Arguments">
  32.             <collectionProp name="Arguments.arguments">
  33.               <elementProp name="" elementType="HTTPArgument">
  34.                 <boolProp name="HTTPArgument.always_encode">false</boolProp>
  35.                 <stringProp name="Argument.value"></stringProp>
  36.                 <stringProp name="Argument.metadata">=</stringProp>
  37.               </elementProp>
  38.             </collectionProp>
  39.           </elementProp>
  40.           <stringProp name="HTTPSampler.domain">localhost</stringProp>
  41.           <stringProp name="HTTPSampler.port">8080</stringProp>
  42.           <stringProp name="HTTPSampler.connect_timeout">10000</stringProp>
  43.           <stringProp name="HTTPSampler.response_timeout">10000</stringProp>
  44.           <stringProp name="HTTPSampler.protocol"></stringProp>
  45.           <stringProp name="HTTPSampler.contentEncoding"></stringProp>
  46.           <stringProp name="HTTPSampler.path">/CrunchifyRESTJerseyExample/crunchify/index</stringProp>
  47.           <stringProp name="HTTPSampler.method">GET</stringProp>
  48.           <boolProp name="HTTPSampler.follow_redirects">true</boolProp>
  49.           <boolProp name="HTTPSampler.auto_redirects">false</boolProp>
  50.           <boolProp name="HTTPSampler.use_keepalive">true</boolProp>
  51.           <boolProp name="HTTPSampler.DO_MULTIPART_POST">false</boolProp>
  52.           <boolProp name="HTTPSampler.monitor">false</boolProp>
  53.           <stringProp name="HTTPSampler.embedded_url_re"></stringProp>
  54.         </HTTPSamplerProxy>
  55.         <hashTree/>
  56.         <ResultCollector guiclass="StatGraphVisualizer" testclass="ResultCollector" testname="Aggregate Graph" enabled="true">
  57.           <boolProp name="ResultCollector.error_logging">false</boolProp>
  58.           <objProp>
  59.             <name>saveConfig</name>
  60.             <value class="SampleSaveConfiguration">
  61.               <time>true</time>
  62.               <latency>true</latency>
  63.               <timestamp>true</timestamp>
  64.               <success>true</success>
  65.               <label>true</label>
  66.               <code>true</code>
  67.               <message>true</message>
  68.               <threadName>true</threadName>
  69.               <dataType>true</dataType>
  70.               <encoding>false</encoding>
  71.               <assertions>true</assertions>
  72.               <subresults>true</subresults>
  73.               <responseData>false</responseData>
  74.               <samplerData>false</samplerData>
  75.               <xml>true</xml>
  76.               <fieldNames>false</fieldNames>
  77.               <responseHeaders>false</responseHeaders>
  78.               <requestHeaders>false</requestHeaders>
  79.               <responseDataOnError>false</responseDataOnError>
  80.               <saveAssertionResultsFailureMessage>false</saveAssertionResultsFailureMessage>
  81.               <assertionsResultsToSave>0</assertionsResultsToSave>
  82.               <bytes>true</bytes>
  83.             </value>
  84.           </objProp>
  85.           <stringProp name="filename"></stringProp>
  86.         </ResultCollector>
  87.         <hashTree/>
  88.         <ResultCollector guiclass="StatVisualizer" testclass="ResultCollector" testname="Aggregate Report" enabled="true">
  89.           <boolProp name="ResultCollector.error_logging">false</boolProp>
  90.           <objProp>
  91.             <name>saveConfig</name>
  92.             <value class="SampleSaveConfiguration">
  93.               <time>true</time>
  94.               <latency>true</latency>
  95.               <timestamp>true</timestamp>
  96.               <success>true</success>
  97.               <label>true</label>
  98.               <code>true</code>
  99.               <message>true</message>
  100.               <threadName>true</threadName>
  101.               <dataType>true</dataType>
  102.               <encoding>false</encoding>
  103.               <assertions>true</assertions>
  104.               <subresults>true</subresults>
  105.               <responseData>false</responseData>
  106.               <samplerData>false</samplerData>
  107.               <xml>false</xml>
  108.               <fieldNames>false</fieldNames>
  109.               <responseHeaders>false</responseHeaders>
  110.               <requestHeaders>false</requestHeaders>
  111.               <responseDataOnError>false</responseDataOnError>
  112.               <saveAssertionResultsFailureMessage>false</saveAssertionResultsFailureMessage>
  113.               <assertionsResultsToSave>0</assertionsResultsToSave>
  114.               <bytes>true</bytes>
  115.             </value>
  116.           </objProp>
  117.           <stringProp name="filename"></stringProp>
  118.         </ResultCollector>
  119.         <hashTree/>
  120.       </hashTree>
  121.     </hashTree>
  122.   </hashTree>
  123. </jmeterTestPlan>

 

步骤 5

  • 下载 Apache JMeter
  • 使用 jmeter.bat 或 jmeter.sh 文件启动 JMeter
  • 点击 File -> Open
  • 切换到你保存 Crunchify-JMeter-Test.jmx 文件的目录并选择该文件

步骤 6

  • 将 Crunchify 测试案例扩展开你会看到下图所示界面
  • 点击 HTTP Request
  • 确认各项参数是否正确

Crunchify-JMeter-and-RESTful-service-Load-test.png

步骤 7

分析你的测试结果:
Crunchify-Tutorial-JMeter-and-RESTService-Result.png
原文链接:How to Call Your Restful Web Service Using JMeter? Perform a Simple Load Testing and Automation。

译者续:《对一个基于 Jersey 框架实现的 RESTful web service 进行压力测试》

Jersey 项目

一个基于 Jersey 框架实现的 RESTful web service 项目如下:
一个Jersey实现的RESTful web service项目.png

将其部署后,访问 wadl 地址 http://192.168.23.204/uas/resource/application.wadl:

wadl地址.png

Web Service 方法

我们要对其这个方法进行压测:

[java] view plain copy

print?

  1. /**
  2.  * 数据初始化
  3.  * @param request
  4.  * @return
  5.  */
  6. @Override
  7. @POST
  8. @Produces(MediaType.APPLICATION_JSON)
  9. @Path("push")
  10. public UserPushResponse push( UserPushRequest request) {
  11.     logger.info(">>>>用户" + request.getToken() + "信息初始化!");
  12.     logger.info(">>>>数据初始化请求报文:{}",
  13.             new Object[]{JsonUtils.object2jsonString(request)});
  14.     //返回报文
  15.     UserPushResponse response;
  16.     try {
  17.         //请求报文校验
  18.         userPushService.validateRequest(request);
  19.         //数据库操作
  20.         redisDao.set(request.getToken(), request.getValue(),
  21.                      new Long(request.getTimeOut()));
  22.         //组织成功报文
  23.         response = userPushService.buildSucessResponse(request);
  24.     } catch (Exception e) {
  25.         logger.info(e.getMessage(), e);
  26.         response = userPushService.buildFailResposne(e);
  27.     }
  28.     logger.info("用户信息初始化响应报文:{}"
  29.             , new Object[]{JsonUtils.object2jsonString(response)});
  30.     return response;
  31. }

该方法单元测试相关代码:

[java] view plain copy

print?

  1. UserResource userResource = RESTfulJsonClientFactory.createClient(UserResource.class, VIP);
  2. Logger logger = LoggerFactory.getLogger(this.getClass());
  3. @Test
  4. public void testResource() {
  5.     UserPushRequest pushRequest = new UserPushRequest();
  6.     pushRequest.setToken("mytest004");
  7.     UserInfoAndMenu userInfoAndMenu = new UserInfoAndMenu();
  8.     UserInfo userInfo = new UserInfo();
  9.     userInfo.setMerCode("102239");
  10.     userInfo.setAccountList(getAccountList());
  11.     userInfoAndMenu.setUserInfo(userInfo);
  12.     pushRequest.setValue(JsonUtils.object2jsonString(userInfoAndMenu));
  13.     pushRequest.setTimeOut(10000);
  14.     System.out.println("请求报文:{}"+JsonUtils.object2jsonString(pushRequest));
  15.     UserPushResponse response = userResource.push(pushRequest);
  16.     System.out.println("响应报文:{}"+JsonUtils.object2jsonString(response));
  17. }

RESTfulJsonClientFactory 是一个 WebService 的远程客户端创建类。执行该单元测试,输出:
请求报文:{}{"msgType":null,"tranCode":null,"sysCode":"1001","brcCode":"1000","srcCode":"01","frontNo":"dfs-02040668","frontTime":"2017-01-20 15:27:19","repCode":null,"repMsg":null,"remark":null,"version":"1.0","signature":null,"cerVersion":null,"channelNo":"01","token":"mytest004","value":"{\"csrfToken\":null,\"userInfo\":{\"loginName\":null,\"merCode\":\"102239\",\"merName\":null,\"accountList\":[{\"acccountCode\":\"1022380016\",\"accountName\":\"redis测试账户名称\"}],\"operator\":null,\"role\":null},\"menuTree\":null}","timeOut":"10000","reqDate":null}
响应报文:{}{"msgType":null,"tranCode":null,"sysCode":"1001","brcCode":"1000","srcCode":"01","frontNo":"channel.web.dfs.local","frontTime":"2017-01-20 15:26:24","repCode":"000000","repMsg":"成功","remark":null,"version":"1.0","signature":null,"cerVersion":null,"channelNo":"01","token":"mytest004","result":"1","reqDate":null}

http 报文截取

SmartSniff 为我们截取到了这次 Web Service 调用,包括一个完整的 http 请求报文(蓝字部分)和返回报文(红字部分):
SmartSniff帮我们截取到了这次WebService调用.png
根据 SmartSniff 的结果我们拿到了这次 Web Service 调用的完整 URL:http:192.168.23.204/uas/resource/com.dfs.uas.biz.resource.UserResource/push,以及完整报文:
POST /uas/resource/com.dfs.uas.biz.resource.UserResource/push HTTP/1.1
Content-Type: application/json
User-Agent: Java/1.6.0_29
Host: 192.168.23.204
Accept: text/html, image/gif, image/jpeg, *; q=.2, */*; q=.2
Connection: keep-alive
Content-Length: 429

{"sysCode":"1001","brcCode":"1000","srcCode":"01","frontNo":"dfs-02040668","frontTime":"2017-01-20 17:12:29","version":"1.0","channelNo":"01","token":"mytest004","value":"{\"csrfToken\":null,\"userInfo\":{\"loginName\":null,\"merCode\":\"102239\",\"merName\":null,\"accountList\":[{\"acccountCode\":\"1022380016\",\"accountName\":\"redis..................\"}],\"operator\":null,\"role\":null},\"menuTree\":null}","timeOut":10000}
HTTP/1.1 200 OK
Server: nginx/1.10.2
Date: Fri, 20 Jan 2017 09:11:28 GMT
Content-Type: application/json
Transfer-Encoding: chunked
Connection: keep-alive

db
{"sysCode":"1001","brcCode":"1000","srcCode":"01","frontNo":"channel.web.dfs.local","frontTime":"2017-01-20 17:11:28","repCode":"000000","repMsg":"......","version":"1.0","channelNo":"01","token":"mytest004","result":1}
0
这些都是很有用的信息,URL 在我们写 HTTP 采样器配置路径的时候用,请求报文在我们组织请求报文体时用,返回报文可以用来参考写断言。

HTTP 信息头管理器

JMeter 新建测试线程组、新建 HTTP 信息头,信息头加入名称为 Content-Type 值为 application/json 项:
JMeter新建测试线程组、新建HTTP信息头.png

HTTP 请求采样器

根据截取信息编辑 HTTP 请求采样器,注意标框部分:
根据截取信息编辑 HTTP 请求采样器.png

HTTP 响应断言

根据截取信息编辑 HTTP 响应断言:
HTTP 响应断言.png

执行压力测试

使用察看结果树对脚本调试成功之后,再配置线程组并发数、循环次数等,开始压测,一个小时后的 TRT 结果:
TRT.png

TPS 数据:

TPS.png

译者后记

Web Service 的压力测试脚本其实很好写,但是如果从建立 Web Service 的客户端比如写 BeanShell 实现本地执行远程服务调用的角度出发往往会落入复杂繁琐而且遥遥无期的脚本开发调试之中(用《神探狄仁杰》里狄仁杰经常说的一句经典台词说:落入彀中)。换个思路,比如从协议的角度出发,Web Service 再复杂也脱离不了 HTTP 吧?然后在借助合理的协议工具的帮助下,看似复杂的脚本编写就会变得非常简单了。

相关实践学习
通过性能测试PTS对云服务器ECS进行规格选择与性能压测
本文为您介绍如何利用性能测试PTS对云服务器ECS进行规格选择与性能压测。
相关文章
|
6天前
|
前端开发 机器人 测试技术
【RF案例】Web自动化测试弹窗处理
在进行Web自动化测试时,常会遇到不同类型的弹窗,如ajax、iframe、新窗口及alert/Confirm等。这些弹窗可通过Selenium进行定位与处理。其中,ajax弹窗直接定位处理;iframe需先选中再操作;新窗口类似iframe处理;而alert/Confirm则需特殊方法应对。在Robot Framework中,需先定义并获取窗口后使用特定关键字处理。此外,还有部分div弹窗需在消失前快速定位。希望本文能帮助大家更好地处理各类弹窗。
17 6
【RF案例】Web自动化测试弹窗处理
|
2天前
|
机器学习/深度学习 人工智能 测试技术
软件测试中的自动化测试实践与挑战
本文深入探讨了软件测试领域中的自动化测试,从基本概念到实际应用案例,揭示了自动化测试在提升软件开发效率和质量中的关键作用。同时,文章也分析了在实施自动化测试过程中面临的主要挑战,并提出了相应的解决策略。
14 5
|
4天前
|
监控 jenkins 测试技术
软件测试中的自动化测试策略与实践
本文将深入探讨自动化测试在软件开发中的重要性及其实施策略。我们将从自动化测试的基本概念入手,分析其在提高软件质量、缩短开发周期和降低维护成本方面的优势。通过具体案例,展示如何有效地规划和执行自动化测试,以及如何评估其效果。
13 1
|
18天前
|
测试技术 C# 图形学
掌握Unity调试与测试的终极指南:从内置调试工具到自动化测试框架,全方位保障游戏品质不踩坑,打造流畅游戏体验的必备技能大揭秘!
【9月更文挑战第1天】在开发游戏时,Unity 引擎让创意变为现实。但软件开发中难免遇到 Bug,若不解决,将严重影响用户体验。调试与测试成为确保游戏质量的最后一道防线。本文介绍如何利用 Unity 的调试工具高效排查问题,并通过 Profiler 分析性能瓶颈。此外,Unity Test Framework 支持自动化测试,提高开发效率。结合单元测试与集成测试,确保游戏逻辑正确无误。对于在线游戏,还需进行压力测试以验证服务器稳定性。总之,调试与测试贯穿游戏开发全流程,确保最终作品既好玩又稳定。
40 4
|
19天前
|
Web App开发 测试技术 API
自动化测试之美:使用Selenium和Python进行Web应用测试
【8月更文挑战第31天】在软件开发的快节奏世界中,自动化测试如同一束明灯,照亮了质量保证之路。本文将引导你通过Selenium和Python的强大组合,探索如何构建高效的Web应用测试框架。我们不仅会讨论理论,还会深入代码,从一个简单的示例开始,逐步扩展至更复杂的场景。无论你是初学者还是有经验的开发者,这篇文章都将为你提供宝贵的见解和实用的技巧。让我们一同揭开自动化测试的神秘面纱,体验它的魅力所在。
|
3天前
|
测试技术 持续交付
软件测试中的自动化测试实践与探索
在软件开发生命周期中,测试阶段是确保产品质量和稳定性的关键环节。随着技术的快速发展,自动化测试逐渐成为提升测试效率和覆盖率的重要手段。本文将探讨自动化测试的基本概念、工具选择、实施策略以及面临的挑战,旨在为读者提供关于如何在项目中有效应用自动化测试的指导性见解。
10 0
|
5天前
|
监控 Java 测试技术
探索软件测试中的自动化测试策略
在软件开发的生命周期中,测试阶段是确保产品质量和性能的关键步骤。随着技术的进步,自动化测试已成为提高测试效率、减少人为错误的重要手段。本文将探讨自动化测试的基本概念、优势以及实施策略,旨在为读者提供一个关于如何有效利用自动化测试来提升软件开发质量的清晰视角。
|
5天前
|
敏捷开发 测试技术 持续交付
自动化测试之美:如何用Selenium和Python打造高效测试脚本
【9月更文挑战第13天】在软件开发的海洋中,自动化测试是那抹不可或缺的亮色。它不仅提升了测试效率,还保障了产品质量。本文将带你领略使用Selenium和Python构建自动化测试脚本的魅力所在,从环境的搭建到脚本的编写,再到问题的排查,每一步都是对软件质量把控的深刻理解和实践。让我们开始这段探索之旅,解锁自动化测试的秘密吧!
7 0
|
5天前
|
jenkins 测试技术 持续交付
自动化测试的高效之路:如何利用Python进行Web应用测试
【9月更文挑战第13天】在软件开发的快节奏中,自动化测试是确保质量和效率的关键。本文将引导你了解如何使用Python语言及其强大的测试框架来提升Web应用的测试效率。我们将一起探索编写简洁而强大的测试脚本的技巧,以及如何通过持续集成(CI)实现自动化测试流程。准备好让你的测试工作飞一般的感觉!
|
18天前
|
API C# 开发框架
WPF与Web服务集成大揭秘:手把手教你调用RESTful API,客户端与服务器端优劣对比全解析!
【8月更文挑战第31天】在现代软件开发中,WPF 和 Web 服务各具特色。WPF 以其出色的界面展示能力受到欢迎,而 Web 服务则凭借跨平台和易维护性在互联网应用中占有一席之地。本文探讨了 WPF 如何通过 HttpClient 类调用 RESTful API,并展示了基于 ASP.NET Core 的 Web 服务如何实现同样的功能。通过对比分析,揭示了两者各自的优缺点:WPF 客户端直接处理数据,减轻服务器负担,但需处理网络异常;Web 服务则能利用服务器端功能如缓存和权限验证,但可能增加服务器负载。希望本文能帮助开发者根据具体需求选择合适的技术方案。
53 0