jmeter学习记录--04--Beanshell
一、什么是Bean Shell
- BeanShell是一种完全符合Java语法规范的脚本语言,并且又拥有自己的一些语法和方法;BeanShell是一种松散类型的脚本语言(这点和JS类似);
- BeanShell是用Java写成的,一个小型的、免费的、可以下载的、嵌入式的Java源代码解释器
- BeanShell能够动态地执行标准JAVA语法,并利用在JavaScript和Perl中常见的的松散类型、命令、闭包等通用脚本来对其进行拓展。BeanShell不仅仅可以通过运行其内部的脚本来处理Java应用程序,还可以在运行过程中动态执行java应用程序执行java代码。因为BeanShell是用java写的,运行在同一个虚拟机的应用程序,因此可以自由地引用对象脚本并返回结果。
- BeanShell执行标准Java语句和表达式,另外包括一些脚本命令和语法。
jmeter中,Beanshell使用与java请求的区别?
要根据自己的使用场景来选择适合自己的自定义脚本的方式。
BeanShell脚本编写类似于Java脚本。它可以获取、修改系统定义或用户定义的变量值,同时也可以进行一些相应的测试数据处理工作(例如:前置处理器、后置处理器及响应断言中均存在响应的BeanShell应用)。
使用建议 BeanShell是JMeter内置的功能,但是由于它是脚本语言,动态加载执行的,因此效率不是很高,不太适合于在经常执行的场景下,比如将BeanShell放在循环内部,不断地被执行。比较适合的应用场景是放在执行一次、或者少数几次的地方,比如在循环外部读取配置文件内容等。
Java扩展JMeter的实现方式的效率比较高,适合于放在经常执行的测试步骤中,但是由于它不是JMeter内置的功能,扩展起来需要有些工作量,而且部署的时候也比较麻烦(分布式运行的时候需要将自定义的JAR拷贝至所有的机器上)。
从java请求与beanshell请求的原理来考虑:
1、比如测试java接口,确不想继承AbstractJavaSamplerClient,可以考虑用beanshell请求来完成。
2、对数据进行参数化时,比如要对字符串进行加密操作、拼成unique类型的ID、定制时间类型等等,可以用beanShell来完成。
3、线程之间想共享一个变量,可以用beanShell实现。
二、Jmeter有哪些Bean Shell
-
定时器: BeanShell Timer 可用来设置线程延迟间隔
-
前置处理器:BeanShell PreProcessor
-
采样器: BeanShell Sampler
-
后置处理器:BeanShell PostProcessor
-
断言: BeanShell断言
-
监听器: BeanShell Listener
这几种在使用是时比较类似,只需要用一种重点练习用法就可以。
三、用法
BeanShell PreProcessor用法(对参数值加密处理返回结果)
常用内置变量
目录
一、操作变量
二、操作属性
三、自定义函数
四、引用外部java文件
五、引用外部class文件
六、引用外部Jar包
七、其它用法(接受参数, log等)
3、用法简单总结
1)可使用用户自定义的变量,在simple中对其进行处理(诸如对其加密、类型转换等),后续请求值使用这些变量与定义函数值;
2)可以在simple中java直接自定义函数,对其一系列处理后,后续请求值使用这些变量与定义函数值;
四、使用案例
案例一:使用Jmeter的BeanShell断言,把响应数据中的JSON跟数据库中的记录对比
案例二:jmeter beanShell 修改http请求参数
案例三:JMeter - BeanShellSampler实现本地AES加密、Jmeter接口测试加解密及Bean Shell使用案例
md5加密
import org.apache.commons.codec.digest.DigestUtils; //导入md5加密的包 String str = "1234567"; //要加密的字符串 String md5_after = DigestUtils.md5Hex(str); // md5加密 vars.put("md5_upper",sign.toUpperCase()); //把加密的结果转成大写的,存到md5_upper这个变量里面,参数化的时候直接用${md5_upper}取值就ok了 vars.put("md5_lower",sign.toLowerCase()); //把加密的结果转成小写的,存到md5_lower这个变量里面,参数化的时候直接用${md5_lower}取值就ok了 vars.put("befor_md5",str); //这个是 加密之前的字符串,在beanshell里面定义的变量参数化的时候不能直接使用,需要用vars.put方法才可以用
如果用到签名算法时,可参考:大众点评API
签名算法类似如下:
1. 对除appkey以外的所有请求参数进行字典升序排列;
2. 将以上排序后的参数表进行字符串连接,如key1value1key2value2key3value3...keyNvalueN;
3. 将app key作为前缀,将app secret作为后缀,对该字符串进行SHA-1计算,并转换成16进制编码;
import org.apache.commons.codec.digest.DigestUtils; import java.util.Map; import java.util.HashMap; import java.util.Map.Entry; // 定义Secret String secret = "test"; // 创建参数表 //Map<String, String> paramMap = new HashMap<String, String>(); Map paramMap = new HashMap(); paramMap.put("mobileNumber", "${mobileNumber}"); paramMap.put("loginPword", "test123456"); paramMap.put("time", "${__time(/1000,)}"); paramMap.put("geetest_challenge", ""); paramMap.put("geetest_seccode", ""); paramMap.put("geetest_validate", ""); // 对参数名进行字典排序 String[] keyArray = paramMap.keySet().toArray(new String[0]); Arrays.sort(keyArray); // 拼接有序的参数名-值串 StringBuilder stringBuilder = new StringBuilder(); for (String key : keyArray) { stringBuilder.append(key).append(paramMap.get(key)); } stringBuilder.append(secret); String s = stringBuilder.toString(); // md5加密 String sign = DigestUtils.md5Hex(s); vars.put("sign",sign);
第一步:测试计划添加自定义变量
第二步:添加beanshell请求,处理
第三步:查看结果树,查看结果
result = "\n1:"+bsh.args[0]+"\n"; //1:从参数框中获取参数变量,不适用 result+="2:${testBeanshell}\n"; //2:直接获取变量,操作方便,非常适用 result+="3:"+vars.get("testBeanshell"); //3:从vars内置变量中获取变量,适用 vars.put("testBeanshell","beanshell_change");//改变变量值 result+="\n4:${testBeanshell}";//4:直接获取不生效,因为jmeter还没有对其进行复制 result+="\n5:"+vars.get("testBeanshell");//:5:值改变,容易理解 ResponseCode = 500;//回写请求响应代码,借鉴的http响应码 ResponseMessage = "test ok!!";//回写响应信息 IsSuccess = true;//回写响应结果 log.info("\nlog test:"+result);//调用jmeter内置log对象,日志会打印到jmeter.log中 SampleResult.setResponseData("SampleResult:"+result+"\n");//回写响应数据,SampleResult为jmeter内置回写结果对象,有很多可用方法,可以参看源码 return "return:"+result;//这里的返回值,相当于调用SampleResult.setResponseData()方法,注意使用的return关键字,后面的行都不会执行,且SampleResult.setResponseData("SampleResult:"+result+"\n");也不会执行,这里会被return的结果覆盖。但是用SampleResult.setResponseData()方法回写,后续还可以接代码行,因此建议不用return进行回写响应数据。
红色标注的为入参。可以直接用seling OK中间用空格分开,或用${}来引用。脚本中通过bsh.args来引用,bsh.args[0]表示取的第一个值。
能访问的属性与变量:
SampleResult:访问对应的结果对象,能查询与编辑其中的属性。
ResponseCode:引用自行设置的返回码。
ResponseMessage:设置响应消息内容。
IsSuccess:引用它来决定产生的事务是否成功。如果设置为false那事务会失败。那在结果树与聚合报告中都会显示失败。
案例五:jmeter使用BeanShell Sampler测试自己写的java接口(一)
案例:起多个本地线程jmeter beanshell使用说明
案例:jmeter之编写java请求一beashell调用java代码(上篇)(推