一、 Beanshell概念
Beanshell:
- BeanShell是一种完全符合Java语法规范的脚本语言,并且又拥有自己的一些语法和方法;
- BeanShell是一种松散类型的脚本语言(这点和JS类似);
- BeanShell是用Java写成的,一个小型的、免费的、可以下载的、嵌入式的Java源代码解释器,具有对象脚本语言特性,非常精简的解释器jar文件大小为175k。
- BeanShell执行标准Java语句和表达式,另外包括一些脚本命令和语法。
二、Jmeter常用Bean Shell
-
定时器: BeanShell Timer
-
前置处理器:BeanShell PreProcessor
-
采样器: BeanShell Sampler
-
后置处理器:BeanShell PostProcessor
-
断言: BeanShell断言
-
监听器: BeanShell Listener
三、Jmeter常用内置变量
1.内置变量
BeanShell脚本中不用定义,可以直接使用的变量,常用的内置变量和方法如下:
log:写日志到控制台和jmeter.log,如log.info("XXX");
vars:操作变量vars.get("skuId"); 从jmeter中获取${skuId}变量的值
vars.put("name","test"); 将"test"保存到${name}变量中
prev:获取前面sampler返回的信息getResponseDataAsString(); 获取响应信息
getResponseCode(); 获取响应
更多内置变量参考:https://jmeter.apache.org
举例:
2.添加 BeanShell取样器
log.info(".....<<<<<");
log.error("xxxx");
String myip=vars.get("ip");//获取参数值(在用户定义的变量中设置了"ip"值)
log.info(myip);//日志打印
vars.put("ip","192.168.0.0");//赋值
String myip=vars.get("ip");
log.info(myip);//日志打印
3.后置处理器的应用
log.info("前一个返回结果为:"+prev.getResponseDataAsString());
log.info("前一个请求的状态码为:"+prev.getResponseCode())
四、BeanShell的用法实例
1.BeanShell面板上写脚本
需求:
1、调用接口获取sku信息
2、判断库存,如果库存大于500,调用buy接口购买10个商品,否则购买5个商品
// 获取接口返回的库存值 String myStock = vars.get("p_stock");//转换为整数 int iStock = Integer.parselnt(myStock);//判断库存 if (iStock>500){ // 重新保存参数 vars.put("buyNum","10"); }else{ vars.put("buyNum","5"); }
先是get接口json提取器提取库存字段然后用到后面脚本里去做判断
2.引用外部 java源文件
引用外部源码文件然后实现md5加密完成签名接口
源码文件:
链接: https://pan.baidu.com/s/1JQlgeHGVHl8NOdSiTwNMkg?pwd=uqrs 提取码: uqrs
在beanShel中通过source("代码路径")方法引入java,然后调用方法和java一样,new一个class,再调用里面的方法
前置处理器
//引入源代码
source("/Users/mac/Documents/study23/jmeter/md5/Md5Util.java");
//生成随机手机号
String phone = "135${__Random(10000000,99999999,myPhone)}";
String code = "testmay";
//生成时间戳
String time = "${__time(,myTime)}";
// 调用外部函数进行加密
String md5 = Md5Util.getMd5Hex(phone+code+time);
// 将数据另存为新的变量
vars.put("phone",phone);
vars.put("md5",md5);
${phone} ${code} ${md5}可被调用
3.调用jar包
使用beahshell调用自己写的工具类,工具类实现了密码的加、解密功能
在eclipse写好代码,然后把该类打成jar包(在类上点击右键->Export->jar file)
法一:
1、将jar包放到jmeter目录/Users/mac/Documents/apache-jmeter-5.6.2/lib/ext下
3、打开jmeter,添加一个http sampler(调用登录接口),在sampler下添加一个BeanShell PreProcessor(如果jmeter已经打开了,步骤2中jar包要生效,必须才重启jmeter)
4、在beanshell PreProcessor中导入我们的jar包,调用里面的加、解密码方法,把结果保存在jmeter变量中
法二:
调用jar包
1、测试计划,Add directory or jar to classpath
2、import 所需要的类名
查看jar包内容:
进入jar路径,输入以下指令:
jar tf testfan-md5.jar
然后调用方式用import
import com.lee.util.Md5Util // 生成随机手机号 String phone = "135${_Random(10000000,99999999,myPhone)}"; String code = "testmay"; // 生成时间戳 String time = "${__time(,myTime)}"; //调用外部函数进行加密 String md5 = Md5Util.getMd5Hex(phone+code+time); //将数据另存为新的变量 vars.put("phone",phone); vars.put("md5",md5);
4.提取 json值
需求:提取sample返回json数据中所有name字段值,返回的json格式如下:
{"body":{"apps":[{"name":"111"},{"name":"222"}]}}
jmeter中添加后置处理器BeanShell PostProcessor
说明:脚本中的导入的json包需要自己去网络下载后放到\lib\ext
链接: https://pan.baidu.com/s/1-knIb9_NulF81mIkortvoQ?pwd=h4p4 提取码: h4p4
import org.json.*;
String response_data = prev.getResponseDataAsString();
JSONObject data_obj = new JSONObject(response_data);
String apps_str = data_obj.get("body").get("apps").toString();
JSONArray apps_array = new JSONArray(apps_str);
String[] result = new String[apps_array.length()];
for(int i=0;i<apps_array.length();i++){
JSONObject app_obj = new JSONObject(apps_array.get(i).toString());
String name = app_obj.get("name").toString();
result[i] = name;
}
vars.put("result", Arrays.toString(result));
5.BeanShell断言
5.1 数据断言
内置变量
Failure:是否失败, boolean类型
FailureMessage:失败日志,在断言失败时显示
int iStock = Integer.parselnt(vars.get("p_stock")); if (iStock >1500){ Failure =true; FailureMessage ="库存数量超过了1500"; // ResponseData是服务器返回的byte[]类型的数据 // 如果想打印,必须转换为String类型的,用new String(ResponseData) log.info(new String(ResponseData)); //打印当前请求的url,SamplerData是String类型的数据 log.info(SamplerData); }
5.2 状态断言
//状态码断言
log.info("状态码:" + ResponseCode);
if(ResponseCode.equals("200")){
Failure=false;
}
else{
Failure=true;
FailureMessage="响应状态码非200"; //指定失败原因
}
5.3 响应体断言
//获取响应数据
Stringresponse= prev.getResponseDataAsString();
log.info("响应体:" + response);
//响应数据包含if(response.contains("登录成功")){
Failure=false;
}
else{
Failure=true;
FailureMessage="响应数据不包含登录成功";
}
5.4 json值断言
//JSON响应断言
import org.json.*; //导入org.json包
Stringresponse= prev.getResponseDataAsString();
//获取响应数据
JSONObjectresponseJson=newJSONObject(response);
//转为JSON对象
Stringmessage= responseJson.getString("message");
log.info("响应message字段:" + message);
if(message.equals("成功")){
Failure=false;
}
else{
Failure=true;
FailureMessage="响应message字段非成功";
}
6.BeanShell写数据到文件
这如果保存后就可以用到其他线程里当参数了,很实用
以后只需要更改下面的部分,这个文件就可以用了
"csrf_token" "/Users/mac/Documents/study23/jmeter/output.txt"
需求
1、调用登录接口,获取token值
2、将token值保存到一个文件里
String line = vars.get("csrf_token"); try { BufferedWriter writer = new BufferedWriter(new FileWriter("/Users/mac/Documents/study23/jmeter/output.txt",true)); writer.write(line); writer.newLine(); writer.close(); } catch (IOException e) { e.printStackTrace(); }
运行会自动生成一个 txt文件
7.引用外部class文件
1、直接把上例中的java文件编译成class文件
进入java路径下,使用javac Md5Util.java就会编译成.class文件,同目录自动生成.class文件
【注意】macos上此处有大坑
这里有一个大坑,调用class文件时:addClassPath()方法中的路径,在macos上不能写成文件的绝对路径。这个路径只能写到包名的前一级,不能包含包名。
如果拿到的是一个单独的class文件,一定要反编译,检查包名,将这个class文件的包名层级新建出来,再将class文件放进去
2、BeanShell使用代码如下:
用addClassPath(path)方法引入 class文件,在用import导入包及类,然后就可以像java一样调用了
//引入class文件
addClassPath("/Users/mac/Documents/study23/jmeter/md5");
//导入类名
import Md5Util
// import 包名.类名
//生成随机手机号
String phone = "135${__Random(10000000,99999999,myPhone)}";
String code = "testmay";
//生成时间戳
String time = "${__time(,myTime)}";
// 调用外部函数进行加密
String md5 = Md5Util.getMd5Hex(phone+code+time);
// 将数据另存为新的变量
vars.put("phone",phone);
vars.put("md5",md5);
log.info(md5)
8.获取数据库数据并入参
8.1 添加配置元件 JDBC Connection Configuration
8.2 添加取样器JDBC Request
8.3 添加后置处理器 JDBC Postprocessor
import java.util.Random;
Random random=new Random();
Object object=vars.getObject("object"); //获取sql查询结果
int size=object.size(); //获取查询结果数量
line_0 = object.get(0); //获取object的第一个元素
filed_name = line_0.get("title"); //获取title字段的值
vars.put("size",size.toString()); //将size转换成字符串,存到变量size中
vars.put("line_0",line_0.toString());
vars.put("filed_name",filed_name.toString());
String[] fields={"title","price"}; //创建一个字符数组,里面为需要作为入参的字段
log.info(" ");
log.info("========================开始打印日志====================");
log.info("size:" + vars.get("size"));
log.info("line_0:" + vars.get("line_0"));
log.info("filed_name:" + vars.get("filed_name"));
运行
8.4 参数调用
${file_name}
${size}