Jmeter之Beanshell详解

一、 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

    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}