实际业务中,会要求 HTTP 协议中附加 MD5 校验字段, 防止请求参数被恶意篡改, 对于开发同学来说, 这是个很简单的需求。 但是给自动化测试增加了难度, Jmeter 原生不支持这个功能,应测试同学要求,开发了一个简单的小插件。

一、功能分析

  1. MD5 值生成规则:

    Step1, HTTP 请求增加新参数 ct, 值为当前时间毫秒数;

    Step2, 获取HTTP 请求所有参数,放入 list;

    Step3, List 增加新参数 key, 值根据通信协议的双方约定(不在 http请求中传递 );

    Step4, 把 List 按参数名进行排序,把排序后的参数值进行拼接, 对拼接结果执行 MD5 运算;

    Step5, Http请求增加新参数 code, 值是 Step4 计算得出的值。

  注:不同业务此规则可能略有差异,请根据实际情况实现。

  2. MD5 值需要在 HTTP 请求之前添加 (属于 Jmeter 前置处理器范围)。

二、需要解决的问题

  1. 在 Jmeter 的 HTTP 请求中获取参数、 增加参数

  2. 实现 MD5 的具体逻辑

  3. 在合适的时机把生成的 MD5 校验值附加到 HTTP 请求参数中

  4. 集成到 Jmeter 菜单中,方便使用。

三、解决步骤

  1. 因之前没有 Jmeter 插件开发经验, 需要先学习一下如何开发。 网上搜集了一下资料没有太明确的方案;不过Jmeter本身有很多插件,就从这里入手,分析一下现有插件的实现方式,然后用相同的方式来完成插件。

  2. 到 Apache 官网 下载 Jmeter 的源代码(我使用的2.11版), 根据 Jmeter 界面中看到的关键字来定位插件代码位置。

  3. 开发的是 Http Sampler 插件,那么,我们先搜索一下Http Sampler关键字, 可以找到 HttpSampler.java;阅读代码可以得到问题1的答案。

  4. 问题2不在赘述。

  5. 分析问题3, 我们需要在 HTTP 请求发送前, 进行插件调用, 可以得知这个插件应该是属于前置处理器范围的。 通过分析前置处理器的代码应该可以解决这个问题。Jmeter 插件中第一个前置处理器是 BeanShell PreProcessor, 用 BeanShell 和 PreProcessor 关键字查找源代码, 查到了类 BeanShellPreProcessor.java;阅读这个类及父类代码可以得到问题3、问题4的答案。

四、具体代码

1. MD5PreProcessor.java

 public class MD5PreProcessor  extends AbstractTestElement implements TestStateListener, PreProcessor, Serializable {

     private static final long serialVersionUID = -1L;
private static final Logger LOGGER = LoggingManager.getLoggerForClass(); /**
* Default constructor.
*/
public MD5PreProcessor() {
} @Override
public void testStarted() {
} @Override
public void testStarted(String host) {
} @Override
public void testEnded() {
} @Override
public void testEnded(String host) {
} /*
* ------------------------------------------------------------------------
* Methods implemented from interface org.apache.jmeter.config.Modifier
* ------------------------------------------------------------------------
*/ @Override
public void process() {
Sampler sam = getThreadContext().getCurrentSampler();
HTTPSamplerBase sampler = null;
if (!(sam instanceof HTTPSamplerBase)) {
return;
} else {
sampler = (HTTPSamplerBase) sam;
} // 排序参数和header(appversion, os)
java.util.List<KeyValue> list = new ArrayList<KeyValue>(); sampler.addArgument("ct", System.currentTimeMillis() + ""); // 添加参数列表
Arguments arguments = sampler.getArguments();
PropertyIterator iter = arguments.iterator();
while(iter.hasNext()) {
Argument arg = getFromJMeterProperty(iter.next());
KeyValue kv = new KeyValue(arg.getName(), arg.getValue());
list.add(kv);
} // 添加header: os appVersion
HeaderManager headerManager = sampler.getHeaderManager();
for(int i = 0; i < headerManager.getHeaders().size(); i++) {
Header a = headerManager.getHeader(i);
if("os".equals(a.getName()) || "appVersion".equals(a.getName())) {
list.add(new KeyValue(a.getName(), a.getValue()));
}
} // 加盐
list.add(new KeyValue("key", "+MrK}6seb#$E")); StringBuffer sbu = new StringBuffer(); // 按顺序拼接所有参数值
Collections.sort(list);
for(KeyValue kv : list) {
//LOGGER.info("key: "+kv.getKey()+", value:"+kv.getValue());
sbu.append(kv.getValue());
} try {
MessageDigest md = MessageDigest.getInstance("MD5");
byte[] md5sum = md.digest(URLEncoder.encode(sbu.toString(), "UTF-8").getBytes());
String md5Code = new BigInteger(1, md5sum).toString(16); while(md5Code.length() < 32 ){
md5Code = "0"+md5Code;
} LOGGER.info("before: "+sbu.toString()+",encode:"+URLEncoder.encode(sbu.toString(), "UTF-8")+",end:"+md5Code);
sampler.addArgument("code", md5Code);
} catch(Exception e) {
e.printStackTrace();
} } private Argument getFromJMeterProperty(JMeterProperty o) {
return (Argument)o.getObjectValue();
} /*
* ------------------------------------------------------------------------
* Methods
* ------------------------------------------------------------------------
*/
private class KeyValue implements Comparable<KeyValue> {
private String key;
private String value; public KeyValue(String key, String value) {
this.key = key;
this.value = value;
} @Override
public int compareTo(KeyValue o) {
int result = this.key.compareTo(o.key);
if(result == 0) {
result = this.value.compareTo(o.value);
} return result;
} public String getKey() {
return key;
} public String getValue() {
return value;
}
} }

2. Md5PreProcessorGui.java

public class Md5PreProcessorGui extends AbstractPreProcessorGui {
private static final long serialVersionUID = 100L;
private static final Logger LOGGER = LoggingManager.getLoggerForClass(); public Md5PreProcessorGui() {
createGui();
} public String getStaticLabel() {
return "MD5 Encode";
} public String getLabelResource() {
return getClass().getCanonicalName();
} public void configure(TestElement element) {
super.configure(element);
} public TestElement createTestElement() {
MD5PreProcessor sampler = new MD5PreProcessor();
modifyTestElement(sampler);
return sampler;
} public void modifyTestElement(TestElement element) {
element.clear();
configureTestElement(element);
} public void clearGui() {
super.clearGui(); } private void createGui() {
setLayout(new BorderLayout(0, 5));
setBorder(makeBorder()); this.setLayout(new BorderLayout(0, 5));
this.setBorder(this.makeBorder());
this.add(this.makeTitlePanel(), "North"); VerticalPanel mainPanel = new VerticalPanel();
add(mainPanel, "Center"); }
}

Jmeter MD5插件的更多相关文章

  1. JMeter Plugin插件配置及远程监控

    下载jmeter plugins 插件: https://jmeter-plugins.org/downloads/all/ 官网建议我们直接下载plugins-manager.jar,然后进行可选择 ...

  2. 【JMeter】插件安装

    安装插件的方法有两种,一种是传统的方式,即官网下载,本地配置,重启jmeter.现在有一种快捷的方法可以自定义安装插件-插件管理器 JMeter 插件管理器的使用方法很简单:不要手动安装各种插件,它提 ...

  3. JMeter开发插件——图片验证码识别

    我们在性能测试中总会时不时地遭遇到来自于应用系统的各种阻碍,图片验证码就是一类最常见的束缚,登录或交易时需要按照图片中的内容输入正确的验证信息后,数据才可以提交成功,这使得许多性能测试工具只能望而却步 ...

  4. 如何为Apache JMeter开发插件(一)

    本文转载于http://blog.csdn.net/column/details/12925.html,作者:xreztento 作者写的很精华,我打算在此系列操作一遍后,加多点截图,便于更多人更快上 ...

  5. 如何为Apache JMeter开发插件(二)—第一个JMeter插件

    文章内容转载于:http://lib.csdn.net/article/softwaretest/25700,并且加上个人一些截图 本篇将开启为JMeter开发插件之旅,我们选择以Function(函 ...

  6. JMeter扩展插件实现对自定义协议进行支持 转

    本文版权归xmeter.net 所有.欢迎转载,转载请注明出处. 摘要## JMeter本身提供了插件机制,允许第三方扩展JMeter以支持JMeter不支持的协议的测试.本文以扩展一个简单的Apac ...

  7. Jmeter websocket插件安装与使用

    Jmeter websocket插件安装与使用 Apache JMeter是Apache组织开发的基于Java的压力测试工具.用于对软件做压力测试,它最初被设计用于Web应用测试,但后来扩展到其他测试 ...

  8. Jmeter官方插件实现Dubbo接口测试

    目前主流的分布式框架有Dubbo和SpringCloud, SpringCloud是基于Http协议的分布式框架,Dubbo是基于RPC的分布式框架,Jmeter没有内置对Dubbo接口的支持,很难直 ...

  9. 四、Jmeter安装插件

    Jmeter安装插件方法和使用 1-下载Jmeter管理jar包 下载地址:https://jmeter-plugins.org/install/Install/ 2-点击下载 plugins-man ...

随机推荐

  1. java IO(四):键盘录入

    */ .hljs { display: block; overflow-x: auto; padding: 0.5em; color: #333; background: #f8f8f8; } .hl ...

  2. 在UITableView顶部制作简单的UISegmentControl实例方法

    // http://www.tuicool.com/articles/yUfURj 使用方法 如上图: 分两步: 一.创建320长度背景,高不超过40最好 //segment背景图 UIImageVi ...

  3. mysql(4)—— 表连接查询与where后使用子查询的性能分析。

    子查询就是在一条查询语句中还有其它的查询语句,主查询得到的结果依赖于子查询的结果. 子查询的子语句可以在一条sql语句的FROM,JOIN,和WHERE后面,本文主要针对在WHERE后面使用子查询与表 ...

  4. How to create and start VirtualBox VM without GUI

    Suppose you want to create and run virtual machines (VMs) on VirtualBox. However, a host machine doe ...

  5. [转载]完全版线段树 by notonlysuccess大牛

    原文出处:http://www.notonlysuccess.com/ (好像现在这个博客已经挂掉了,在网上找到的全部都是转载) 今天在清北学堂听课,听到了一些很令人吃惊的消息.至于这消息具体是啥,等 ...

  6. BZOJ 4514: [Sdoi2016]数字配对 [费用流 数论]

    4514: [Sdoi2016]数字配对 题意: 有 n 种数字,第 i 种数字是 ai.有 bi 个,权值是 ci. 若两个数字 ai.aj 满足,ai 是 aj 的倍数,且 ai/aj 是一个质数 ...

  7. 【转】PE详解

    参考网址: http://blog.tianya.cn/listcate-4259222-2269876-1.shtml PE文件

  8. 安装Spring Tool Suite(STS)

    JAVA开发工具中,常用工具就是Eclipse,IntelliJ IDEA. 现在使用spring boot&cloud框架进行开发的时候,虽然可以使用上面两个工具,但都未必就真的量身定制,I ...

  9. u-boot核心初始化

    异常向量表:异常:因为内部或者外部的一些事件,导致处理器停下正在处理的工作,转而去处理这些发生的事件.ARM Architecture Reference Manual p54页.7种异常的类型:Re ...

  10. 关于目前自己iOS项目使用的第三方开源库

    1.AFNetworking 目前比较推荐的iOS网络请求组件,默认网络请求是异步,通过block回调的方式对返回数据进行处理. 2.FMDB 对sqlite数据库操作进行了封装,demo也比较简单. ...