Jmeter之Beanshell详解

发布于:2024-04-27 ⋅ 阅读:(18) ⋅ 点赞:(0)

一、 Beanshell概念

 Beanshell:

  • BeanShell是一种完全符合Java语法规范的脚本语言,并且又拥有自己的一些语法和方法;
  • BeanShell是一种松散类型的脚本语言(这点和JS类似);
  • BeanShell是用Java写成的,一个小型的、免费的、可以下载的、嵌入式的Java源代码解释器,具有对象脚本语言特性,非常精简的解释器jar文件大小为175k。
  • BeanShell执行标准Java语句和表达式,另外包括一些脚本命令和语法。

官网:http://www.BeanShell.org/

 二、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.获取数据库数据并入参

jmeter数据库连接 

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}